summaryrefslogtreecommitdiff
path: root/src/GF/Conversion/MCFGtoFCFG.hs
blob: 70aa4644d0702dddde9bf851bda3e224a5dd7d5b (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
----------------------------------------------------------------------
-- |
-- Maintainer  : PL
-- Stability   : (stable)
-- Portability : (portable)
--
-- > CVS $Date: 2005/05/09 09:28:43 $ 
-- > CVS $Author: peb $
-- > CVS $Revision: 1.6 $
--
-- Converting MCFG grammars to equivalent optimized FCFG
-----------------------------------------------------------------------------


module GF.Conversion.MCFGtoFCFG
    (convertGrammar) where

import Control.Monad
import List (elemIndex)
import Array

import GF.Formalism.Utilities
import GF.Formalism.GCFG
import GF.Formalism.MCFG
import GF.Formalism.FCFG
import GF.Conversion.Types
import GF.Data.SortedList (nubsort)

import GF.Infra.Print

----------------------------------------------------------------------
-- * converting MCFG to optimized FCFG

convertGrammar :: MGrammar -> FGrammar
convertGrammar gram = [ FRule (Abs (fcat cat) (map fcat cats) name) (fcnc cnc) |
                        Rule (Abs cat cats name) cnc <- gram ]
    where mcats = nubsort [ mc | Rule (Abs mcat mcats _) _ <- gram, mc <- mcat:mcats ]

          fcat mcat@(MCat (ECat scat ecns) mlbls) 
              = case elemIndex mcat mcats of
                  Just catid -> FCat catid scat mlbls ecns
                  Nothing -> error ("MCFGtoFCFG.fcat " ++ prt mcat)

          fcnc (Cnc  _ arglbls lins) = listArray (0, length lins-1) (map flin lins)
              where flin (Lin _ syms) = listArray (0, length syms-1) (map fsym syms)
                    fsym (Tok tok) = FSymTok tok
                    fsym (Cat (cat,lbl,arg)) = FSymCat (fcat cat) (flbl arg lbl) arg
                    flbl arg lbl = case elemIndex lbl (arglbls !! arg) of
                                     Just lblid -> lblid
                                     Nothing -> error ("MCFGtoFCFG.flbl " ++ prt arg ++ " " ++ prt lbl)