mirror of
https://github.com/GrammaticalFramework/gf-core.git
synced 2026-04-10 21:39:32 -06:00
275 lines
9.5 KiB
Haskell
275 lines
9.5 KiB
Haskell
module GF.GFCC.Raw.ConvertGFCC (toGFCC,fromGFCC) where
|
|
|
|
import GF.GFCC.CId
|
|
import GF.GFCC.DataGFCC
|
|
import GF.GFCC.Raw.AbsGFCCRaw
|
|
|
|
import GF.Infra.PrintClass
|
|
import GF.Data.Assoc
|
|
import GF.Formalism.FCFG
|
|
import GF.Formalism.Utilities (NameProfile(..), Profile(..), SyntaxForest(..))
|
|
import GF.Parsing.FCFG.PInfo (FCFPInfo(..), buildFCFPInfo)
|
|
|
|
import qualified Data.Array as Array
|
|
import Data.Map
|
|
|
|
pgfMajorVersion, pgfMinorVersion :: Integer
|
|
(pgfMajorVersion, pgfMinorVersion) = (1,0)
|
|
|
|
-- convert parsed grammar to internal GFCC
|
|
|
|
toGFCC :: Grammar -> GFCC
|
|
toGFCC (Grm [
|
|
App "pgf" (AInt v1 : AInt v2 : App a []:cs),
|
|
App "flags" gfs,
|
|
ab@(
|
|
App "abstract" [
|
|
App "fun" fs,
|
|
App "cat" cts
|
|
]),
|
|
App "concrete" ccs
|
|
]) = GFCC {
|
|
absname = mkCId a,
|
|
cncnames = [mkCId c | App c [] <- cs],
|
|
gflags = fromAscList [(mkCId f,v) | App f [AStr v] <- gfs],
|
|
abstract =
|
|
let
|
|
aflags = fromAscList [(mkCId f,v) | App f [AStr v] <- gfs]
|
|
lfuns = [(mkCId f,(toType typ,toExp def)) | App f [typ, def] <- fs]
|
|
funs = fromAscList lfuns
|
|
lcats = [(mkCId c, Prelude.map toHypo hyps) | App c hyps <- cts]
|
|
cats = fromAscList lcats
|
|
catfuns = fromAscList
|
|
[(cat,[f | (f, (DTyp _ c _,_)) <- lfuns, c==cat]) | (cat,_) <- lcats]
|
|
in Abstr aflags funs cats catfuns,
|
|
concretes = fromAscList [(mkCId lang, toConcr ts) | App lang ts <- ccs]
|
|
}
|
|
where
|
|
|
|
toConcr :: [RExp] -> Concr
|
|
toConcr = foldl add (Concr {
|
|
cflags = empty,
|
|
lins = empty,
|
|
opers = empty,
|
|
lincats = empty,
|
|
lindefs = empty,
|
|
printnames = empty,
|
|
paramlincats = empty,
|
|
parser = Nothing
|
|
})
|
|
where
|
|
add :: Concr -> RExp -> Concr
|
|
add cnc (App "flags" ts) = cnc { cflags = fromAscList [(mkCId f,v) | App f [AStr v] <- ts] }
|
|
add cnc (App "lin" ts) = cnc { lins = mkTermMap ts }
|
|
add cnc (App "oper" ts) = cnc { opers = mkTermMap ts }
|
|
add cnc (App "lincat" ts) = cnc { lincats = mkTermMap ts }
|
|
add cnc (App "lindef" ts) = cnc { lindefs = mkTermMap ts }
|
|
add cnc (App "printname" ts) = cnc { printnames = mkTermMap ts }
|
|
add cnc (App "param" ts) = cnc { paramlincats = mkTermMap ts }
|
|
add cnc (App "parser" ts) = cnc { parser = Just (toPInfo ts) }
|
|
|
|
toPInfo :: [RExp] -> FCFPInfo
|
|
toPInfo [App "rules" rs, App "startupcats" cs] = buildFCFPInfo (rules, cats)
|
|
where
|
|
rules = lmap toFRule rs
|
|
cats = fromList [(mkCId c, lmap expToInt fs) | App c fs <- cs]
|
|
|
|
toFRule :: RExp -> FRule
|
|
toFRule (App "rule"
|
|
[n,
|
|
App "cats" (rt:at),
|
|
App "R" ls]) = FRule name args res lins
|
|
where
|
|
name = toFName n
|
|
args = lmap expToInt at
|
|
res = expToInt rt
|
|
lins = mkArray [mkArray [toSymbol s | s <- l] | App "S" l <- ls]
|
|
|
|
toFName :: RExp -> FName
|
|
toFName (App "_A" [x]) = Name wildCId [Unify [expToInt x]]
|
|
toFName (App f ts) = Name (mkCId f) (lmap toProfile ts)
|
|
where
|
|
toProfile :: RExp -> Profile (SyntaxForest CId)
|
|
toProfile AMet = Unify []
|
|
toProfile (App "_A" [t]) = Unify [expToInt t]
|
|
toProfile (App "_U" ts) = Unify [expToInt t | App "_A" [t] <- ts]
|
|
toProfile t = Constant (toSyntaxForest t)
|
|
|
|
toSyntaxForest :: RExp -> SyntaxForest CId
|
|
toSyntaxForest AMet = FMeta
|
|
toSyntaxForest (App n ts) = FNode (mkCId n) [lmap toSyntaxForest ts]
|
|
toSyntaxForest (AStr s) = FString s
|
|
toSyntaxForest (AInt i) = FInt i
|
|
toSyntaxForest (AFlt f) = FFloat f
|
|
|
|
toSymbol :: RExp -> FSymbol
|
|
toSymbol (App "P" [c,n,l]) = FSymCat (expToInt c) (expToInt l) (expToInt n)
|
|
toSymbol (AStr t) = FSymTok t
|
|
|
|
toType :: RExp -> Type
|
|
toType e = case e of
|
|
App cat [App "H" hypos, App "X" exps] ->
|
|
DTyp (lmap toHypo hypos) (mkCId cat) (lmap toExp exps)
|
|
_ -> error $ "type " ++ show e
|
|
|
|
toHypo :: RExp -> Hypo
|
|
toHypo e = case e of
|
|
App x [typ] -> Hyp (mkCId x) (toType typ)
|
|
_ -> error $ "hypo " ++ show e
|
|
|
|
toExp :: RExp -> Exp
|
|
toExp e = case e of
|
|
App "App" [App fun [], App "B" xs, App "X" exps] ->
|
|
DTr [mkCId x | App x [] <- xs] (AC (mkCId fun)) (lmap toExp exps)
|
|
App "Eq" eqs ->
|
|
EEq [Equ (lmap toExp ps) (toExp v) | App "E" (v:ps) <- eqs]
|
|
App "Var" [App i []] -> DTr [] (AV (mkCId i)) []
|
|
AMet -> DTr [] (AM 0) []
|
|
AInt i -> DTr [] (AI i) []
|
|
AFlt i -> DTr [] (AF i) []
|
|
AStr i -> DTr [] (AS i) []
|
|
_ -> error $ "exp " ++ show e
|
|
|
|
toTerm :: RExp -> Term
|
|
toTerm e = case e of
|
|
App "R" es -> R (lmap toTerm es)
|
|
App "S" es -> S (lmap toTerm es)
|
|
App "FV" es -> FV (lmap toTerm es)
|
|
App "P" [e,v] -> P (toTerm e) (toTerm v)
|
|
App "RP" [e,v] -> RP (toTerm e) (toTerm v) ----
|
|
App "W" [AStr s,v] -> W s (toTerm v)
|
|
App "A" [AInt i] -> V (fromInteger i)
|
|
App f [] -> F (mkCId f)
|
|
AInt i -> C (fromInteger i)
|
|
AMet -> TM "?"
|
|
AStr s -> K (KS s) ----
|
|
_ -> error $ "term " ++ show e
|
|
|
|
------------------------------
|
|
--- from internal to parser --
|
|
------------------------------
|
|
|
|
fromGFCC :: GFCC -> Grammar
|
|
fromGFCC gfcc0 = Grm [
|
|
App "pgf" (AInt pgfMajorVersion:AInt pgfMinorVersion
|
|
: App (prCId (absname gfcc)) [] : lmap (flip App [] . prCId) (cncnames gfcc)),
|
|
App "flags" [App (prCId f) [AStr v] | (f,v) <- toList (gflags gfcc `union` aflags agfcc)],
|
|
App "abstract" [
|
|
App "fun" [App (prCId f) [fromType t,fromExp d] | (f,(t,d)) <- toList (funs agfcc)],
|
|
App "cat" [App (prCId f) (lmap fromHypo hs) | (f,hs) <- toList (cats agfcc)]
|
|
],
|
|
App "concrete" [App (prCId lang) (fromConcrete c) | (lang,c) <- toList (concretes gfcc)]
|
|
]
|
|
where
|
|
gfcc = utf8GFCC gfcc0
|
|
agfcc = abstract gfcc
|
|
fromConcrete cnc = [
|
|
App "flags" [App (prCId f) [AStr v] | (f,v) <- toList (cflags cnc)],
|
|
App "lin" [App (prCId f) [fromTerm v] | (f,v) <- toList (lins cnc)],
|
|
App "oper" [App (prCId f) [fromTerm v] | (f,v) <- toList (opers cnc)],
|
|
App "lincat" [App (prCId f) [fromTerm v] | (f,v) <- toList (lincats cnc)],
|
|
App "lindef" [App (prCId f) [fromTerm v] | (f,v) <- toList (lindefs cnc)],
|
|
App "printname" [App (prCId f) [fromTerm v] | (f,v) <- toList (printnames cnc)],
|
|
App "param" [App (prCId f) [fromTerm v] | (f,v) <- toList (paramlincats cnc)]
|
|
] ++ maybe [] (\p -> [fromPInfo p]) (parser cnc)
|
|
|
|
fromType :: Type -> RExp
|
|
fromType e = case e of
|
|
DTyp hypos cat exps ->
|
|
App (prCId cat) [
|
|
App "H" (lmap fromHypo hypos),
|
|
App "X" (lmap fromExp exps)]
|
|
|
|
fromHypo :: Hypo -> RExp
|
|
fromHypo e = case e of
|
|
Hyp x typ -> App (prCId x) [fromType typ]
|
|
|
|
fromExp :: Exp -> RExp
|
|
fromExp e = case e of
|
|
DTr xs (AC fun) exps ->
|
|
App "App" [App (prCId fun) [], App "B" (lmap (flip App [] . prCId) xs), App "X" (lmap fromExp exps)]
|
|
DTr [] (AV x) [] -> App "Var" [App (prCId x) []]
|
|
DTr [] (AS s) [] -> AStr s
|
|
DTr [] (AF d) [] -> AFlt d
|
|
DTr [] (AI i) [] -> AInt (toInteger i)
|
|
DTr [] (AM _) [] -> AMet ----
|
|
EEq eqs ->
|
|
App "Eq" [App "E" (lmap fromExp (v:ps)) | Equ ps v <- eqs]
|
|
_ -> error $ "exp " ++ show e
|
|
|
|
fromTerm :: Term -> RExp
|
|
fromTerm e = case e of
|
|
R es -> App "R" (lmap fromTerm es)
|
|
S es -> App "S" (lmap fromTerm es)
|
|
FV es -> App "FV" (lmap fromTerm es)
|
|
P e v -> App "P" [fromTerm e, fromTerm v]
|
|
RP e v -> App "RP" [fromTerm e, fromTerm v] ----
|
|
W s v -> App "W" [AStr s, fromTerm v]
|
|
C i -> AInt (toInteger i)
|
|
TM _ -> AMet
|
|
F f -> App (prCId f) []
|
|
V i -> App "A" [AInt (toInteger i)]
|
|
K (KS s) -> AStr s ----
|
|
K (KP d vs) -> App "FV" (str d : [str v | Var v _ <- vs]) ----
|
|
where
|
|
str v = App "S" (lmap AStr v)
|
|
|
|
-- ** Parsing info
|
|
|
|
fromPInfo :: FCFPInfo -> RExp
|
|
fromPInfo p = App "parser" [
|
|
App "rules" [fromFRule rule | rule <- Array.elems (allRules p)],
|
|
App "startupcats" [App (prCId f) (lmap intToExp cs) | (f,cs) <- toList (startupCats p)]
|
|
]
|
|
|
|
fromFRule :: FRule -> RExp
|
|
fromFRule (FRule n args res lins) =
|
|
App "rule" [fromFName n,
|
|
App "cats" (intToExp res:lmap intToExp args),
|
|
App "R" [App "S" [fromSymbol s | s <- Array.elems l] | l <- Array.elems lins]
|
|
]
|
|
|
|
fromFName :: FName -> RExp
|
|
fromFName n = case n of
|
|
Name f ps | f == wildCId -> fromProfile (head ps)
|
|
| otherwise -> App (prCId f) (lmap fromProfile ps)
|
|
where
|
|
fromProfile :: Profile (SyntaxForest CId) -> RExp
|
|
fromProfile (Unify []) = AMet
|
|
fromProfile (Unify [x]) = daughter x
|
|
fromProfile (Unify args) = App "_U" (lmap daughter args)
|
|
fromProfile (Constant forest) = fromSyntaxForest forest
|
|
|
|
daughter n = App "_A" [intToExp n]
|
|
|
|
fromSyntaxForest :: SyntaxForest CId -> RExp
|
|
fromSyntaxForest FMeta = AMet
|
|
-- FIXME: is there always just one element here?
|
|
fromSyntaxForest (FNode n [args]) = App (prCId n) (lmap fromSyntaxForest args)
|
|
fromSyntaxForest (FString s) = AStr s
|
|
fromSyntaxForest (FInt i) = AInt i
|
|
fromSyntaxForest (FFloat f) = AFlt f
|
|
|
|
fromSymbol :: FSymbol -> RExp
|
|
fromSymbol (FSymCat c l n) = App "P" [intToExp c, intToExp n, intToExp l]
|
|
fromSymbol (FSymTok t) = AStr t
|
|
|
|
-- ** Utilities
|
|
|
|
mkTermMap :: [RExp] -> Map CId Term
|
|
mkTermMap ts = fromAscList [(mkCId f,toTerm v) | App f [v] <- ts]
|
|
|
|
mkArray :: [a] -> Array.Array Int a
|
|
mkArray xs = Array.listArray (0, length xs - 1) xs
|
|
|
|
expToInt :: Integral a => RExp -> a
|
|
expToInt (App "neg" [AInt i]) = fromIntegral (negate i)
|
|
expToInt (AInt i) = fromIntegral i
|
|
|
|
expToStr :: RExp -> String
|
|
expToStr (AStr s) = s
|
|
|
|
intToExp :: Integral a => a -> RExp
|
|
intToExp x | x < 0 = App "neg" [AInt (fromIntegral (negate x))]
|
|
| otherwise = AInt (fromIntegral x)
|