summaryrefslogtreecommitdiff
path: root/src/GF/Conversion/SimpleToFCFG.hs
blob: fbefd8693a0fb320c79a974b5913f764c2f952a0 (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
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
----------------------------------------------------------------------
-- |
-- Maintainer  : Krasimir Angelov
-- Stability   : (stable)
-- Portability : (portable)
--
-- Converting SimpleGFC grammars to fast nonerasing MCFG grammar.
--
-- the resulting grammars might be /very large/
--
-- the conversion is only equivalent if the GFC grammar has a context-free backbone.
-----------------------------------------------------------------------------


module GF.Conversion.SimpleToFCFG
    (convertGrammar) where

import GF.System.Tracing
import GF.Infra.Print
import GF.Infra.Ident

import Control.Monad

import GF.Formalism.Utilities
import GF.Formalism.GCFG 
import GF.Formalism.FCFG
import GF.Formalism.SimpleGFC 
import GF.Conversion.Types
import GF.Canon.AbsGFC(CIdent(..))

import GF.Data.BacktrackM
import GF.Data.SortedList
import GF.Data.Utilities (updateNthM)

import qualified Data.Map as Map
import qualified Data.Set as Set
import qualified Data.List as List
import Data.Array

----------------------------------------------------------------------
-- main conversion function

convertGrammar :: SGrammar -> FGrammar
convertGrammar srules = getFRules (loop frulesEnv)
  where
    (srulesMap,frulesEnv) = List.foldl' helper (Map.empty,emptyFRulesEnv) srules
      where
        helper (srulesMap,frulesEnv) rule@(Rule (Abs decl _ _) (Cnc ctype _ _)) = 
          let srulesMap' = Map.insertWith (++) (decl2cat decl) [rule] srulesMap
              frulesEnv' = List.foldl' (\env selector -> convertRule selector rule env) 
                                       frulesEnv
                                       (mkSingletonSelectors ctype)
          in srulesMap' `seq` frulesEnv' `seq` (srulesMap',frulesEnv')

    loop frulesEnv =
      let (todo, frulesEnv') = takeToDoRules srulesMap frulesEnv
      in case todo of
           [] -> frulesEnv'
           _  -> loop $! List.foldl' (\env (srules,selector) -> 
                         List.foldl' (\env srule             -> convertRule selector srule env) env srules) frulesEnv' todo


----------------------------------------------------------------------
-- rule conversion

convertRule :: STermSelector -> SRule -> FRulesEnv -> FRulesEnv
convertRule selector (Rule (Abs decl decls (Name fun profile)) (Cnc ctype ctypes (Just term))) frulesEnv =
  foldBM addRule
         frulesEnv
         (convertTerm selector term [Lin emptyPath []])
         (let cat : args = map decl2cat (decl : decls)
          in (initialFCat cat, map (\scat -> (initialFCat scat,[])) args, ctype, ctypes))
  where
    addRule linRec (newCat', newArgs', _, _) env0 =
      let (env1, newCat)          = genFCatHead env0 newCat'
          (env2, newArgs,idxArgs) = foldr (\((fcat@(FCat _ cat rcs tcs),xpaths),ctype,idx) (env,args,all_args) ->
                                       let xargs          = fcat:[FCat 0 cat [path] tcs | path <- reverse xpaths]
                                           (env1, xargs1) = List.mapAccumL (genFCatArg ctype) env xargs
                                       in case fcat of
                                            FCat _ _ [] _ -> (env ,        args,             all_args)
                                            _             -> (env1,xargs1++args,(idx,xargs1):all_args)) (env1,[],[]) (zip3 newArgs' ctypes [0..])
          
          newLinRec = listArray (0,length linRec-1) [translateLin idxArgs path linRec | path <- case newCat of {FCat _ _ rcs _ -> rcs}]

          (_,newProfile) = List.mapAccumL accumProf 0 newArgs'
            where
              accumProf nr (FCat _ _ [] _,_     ) = (nr,       Unify []          )
              accumProf nr (_            ,xpaths) = (nr+cnt+1, Unify [nr..nr+cnt])
                where cnt = length xpaths

          rule = FRule (Abs newCat newArgs (Name fun newProfile)) newLinRec
      in addFCatRule env2 rule
convertRule selector _ frulesEnv = frulesEnv

translateLin idxArgs lbl' []                    = array (0,-1) []
translateLin idxArgs lbl' (Lin lbl syms : lins)
  | lbl' == lbl = listArray (0,length syms-1) (map instSym syms)
  | otherwise   = translateLin idxArgs lbl' lins
  where
    instSym = symbol (\(_, lbl, nr, xnr) -> instCat lbl nr xnr 0 idxArgs) FSymTok
    instCat lbl nr xnr nr' ((idx,xargs):idxArgs)
      | nr == idx = let arg@(FCat _ _ rcs _) = xargs !! xnr
                    in FSymCat arg (index lbl rcs 0) (nr'+xnr)
      | otherwise = instCat lbl nr xnr (nr'+length xargs) idxArgs

    index lbl' (lbl:lbls) idx
      | lbl' == lbl = idx
      | otherwise   = index lbl' lbls $! (idx+1)

----------------------------------------------------------------------
-- term conversion

type CnvMonad a = BacktrackM Env a

type Env    = (FCat, [(FCat,[SPath])], SLinType, [SLinType])
type LinRec = [Lin SCat SPath Token]

data Lin cat lbl tok = Lin lbl [Symbol (cat, lbl, Int, Int) tok]


convertTerm :: STermSelector -> STerm -> LinRec -> CnvMonad LinRec
convertTerm selector (Arg nr cat path) (Lin lbl_path lin : lins) = convertArg selector nr cat path lbl_path lin lins
convertTerm selector (con   :^   args) (Lin lbl_path lin : lins) = convertCon selector con args lbl_path lin lins
convertTerm selector (Rec      record) (Lin lbl_path lin : lins) = convertRec selector record lbl_path lin lins
convertTerm selector (term  :.    lbl)                     lins  = convertTerm (RecPrj lbl selector) term lins
convertTerm selector (Tbl       table) (Lin lbl_path lin : lins) = convertTbl selector table  lbl_path lin lins
convertTerm selector (term  :!    sel)                     lins  = do sel <- evalTerm sel
                                                                      convertTerm (TblPrj sel selector) term lins
convertTerm selector (Variants   vars)                     lins  = do term <- member vars
                                                                      convertTerm selector term lins
convertTerm selector (t1    :++    t2)                     lins  = do lins <- convertTerm selector t2 lins
                                                                      lins <- convertTerm selector t1 lins
                                                                      return lins
convertTerm selector (Token       str) (Lin lbl_path lin : lins) = do projectHead lbl_path
                                                                      return (Lin lbl_path (Tok str : lin) : lins)
convertTerm selector (Empty          ) (Lin lbl_path lin : lins) = do projectHead lbl_path
                                                                      return (Lin lbl_path lin : lins)

convertArg (RecSel record)  nr cat path lbl_path lin lins =
  foldM (\lins (lbl,  selector) -> convertArg selector nr cat (path ++. lbl)  (lbl_path ++.  lbl) lin lins) lins record
convertArg (TblSel cases)   nr cat path lbl_path lin lins =
  foldM (\lins (term, selector) -> convertArg selector nr cat (path ++! term) (lbl_path ++! term) lin lins) lins cases
convertArg (RecPrj lbl  selector) nr cat path lbl_path lin lins = 
  convertArg selector nr cat (path ++. lbl ) lbl_path lin lins
convertArg (TblPrj term selector) nr cat path lbl_path lin lins = 
  convertArg selector nr cat (path ++! term) lbl_path lin lins
convertArg (ConSel terms)   nr cat path lbl_path lin lins = do
  sel <- member terms
  restrictHead   lbl_path sel
  restrictArg nr path sel
  return lins
convertArg StrSel          nr cat path lbl_path lin lins = do
  projectHead lbl_path
  xnr <- projectArg nr path
  return (Lin lbl_path (Cat (cat, path, nr, xnr) : lin) : lins)

convertCon (ConSel terms) con args lbl_path lin lins = do
  args <- mapM evalTerm args
  let term = con :^ args
  guard (term `elem` terms)
  restrictHead lbl_path term
  return lins

convertRec selector                 []                    lbl_path lin lins = return lins
convertRec selector@(RecSel fields) ((label, val):record) lbl_path lin lins = select fields
  where
    select []                          = convertRec selector record lbl_path lin lins
    select ((label',sub_sel) : fields)
      | label == label'                = do lins <- convertTerm sub_sel val (Lin (lbl_path ++. label) lin : lins)
                                            convertRec selector record lbl_path lin lins
      | otherwise                      = select fields
convertRec (RecPrj label sub_sel) record lbl_path lin lins = do
  (label',val) <- member record
  guard (label==label')
  convertTerm sub_sel val (Lin lbl_path lin : lins)

convertTbl selector                []                  lbl_path lin lins = return lins
convertTbl selector@(TblSel cases) ((term, val):table) lbl_path lin lins = case selector of { TblSel cases -> select cases }
  where
    select []                          = convertTbl selector table lbl_path lin lins
    select ((term',sub_sel)  : cases)
      | term == term'                  = do lins <- convertTerm sub_sel val (Lin (lbl_path ++! term) lin : lins)
                                            convertTbl selector table lbl_path lin lins
      | otherwise                      = select cases
convertTbl (TblPrj term sub_sel) table lbl_path lin lins = do
  (term',val) <- member table
  guard (term==term')
  convertTerm sub_sel val (Lin lbl_path lin : lins)


------------------------------------------------------------
-- eval a term to ground terms

evalTerm :: STerm -> CnvMonad STerm
evalTerm arg@(Arg nr _ path) = do ctype <- readArgCType nr
	                          unifyPType arg $ lintypeFollowPath path ctype
evalTerm (con :^ terms)      = do terms <- mapM evalTerm terms
                                  return (con :^ terms)
evalTerm (Rec record)        = do record <- mapM evalAssign record
                                  return (Rec record)
evalTerm (term :. lbl)       = do term <- evalTerm term
                                  evalTerm (term +. lbl)
evalTerm (Tbl table)         = do table <- mapM evalCase table
                                  return (Tbl table)
evalTerm (term :! sel)       = do sel  <- evalTerm sel
	                          evalTerm (term +! sel)
evalTerm (Variants terms)    = member terms >>= evalTerm
evalTerm (t1 :++ t2)         = do t1 <- evalTerm t1
                                  t2 <- evalTerm t2
                                  return (t1 :++ t2)
evalTerm (Token str)         = do return (Token str)
evalTerm Empty               = do return Empty

evalAssign :: (Label, STerm) -> CnvMonad (Label, STerm)
evalAssign (lbl, term) = liftM ((,) lbl) $ evalTerm term

evalCase :: (STerm, STerm) -> CnvMonad (STerm, STerm)
evalCase (pat, term) = liftM2 (,) (evalTerm pat) (evalTerm term)

unifyPType :: STerm -> SLinType -> CnvMonad STerm
unifyPType arg (RecT prec) = 
    liftM Rec $
    sequence [ liftM ((,) lbl) $
	       unifyPType (arg +. lbl) ptype |
	       (lbl, ptype) <- prec ]
unifyPType (Arg nr _ path) (ConT terms) = 
    do (_, args, _, _) <- readState
       let (FCat _ _ _ tcs,_) = args !! nr
       case lookup path tcs of
         Just term -> return term
         Nothing   -> do term <- member terms
		         restrictArg nr path term
		         return term


----------------------------------------------------------------------
-- FRulesEnv

data FRulesEnv = FRulesEnv {-# UNPACK #-} !Int FCatSet [FRule]

type SRulesMap = Map.Map SCat [SRule]
type FCatSet = Map.Map SCat (Map.Map [SPath] (Map.Map [(SPath,STerm)] (Either FCat FCat)))


emptyFRulesEnv = FRulesEnv 0 (ins fcatString (ins fcatInt (ins fcatFloat Map.empty))) []
  where
    ins fcat@(FCat _ cat rcs tcs) fcatSet =
      Map.insertWith (\_ -> Map.insertWith (\_ -> Map.insert tcs x_fcat) rcs tmap_s) cat rmap_s fcatSet
      where
        x_fcat = Right fcat
        tmap_s = Map.singleton tcs x_fcat
        rmap_s = Map.singleton rcs tmap_s

genFCatHead :: FRulesEnv -> FCat -> (FRulesEnv, FCat)
genFCatHead env@(FRulesEnv last_id fcatSet rules) m1@(FCat _ cat rcs tcs) =
  case Map.lookup cat fcatSet >>= Map.lookup rcs >>= Map.lookup tcs of
    Just (Left  fcat) -> (FRulesEnv last_id (ins fcat) rules, fcat)
    Just (Right fcat) -> (env, fcat)
    Nothing           -> let next_id = last_id+1
                             fcat    = FCat next_id cat rcs tcs
                         in (FRulesEnv next_id (ins fcat) rules, fcat)
  where
    ins fcat = Map.insertWith (\_ -> Map.insertWith (\_ -> Map.insert tcs x_fcat) rcs tmap_s) cat rmap_s fcatSet
      where
        x_fcat = Right fcat
        tmap_s = Map.singleton tcs x_fcat
        rmap_s = Map.singleton rcs tmap_s

genFCatArg :: SLinType -> FRulesEnv -> FCat -> (FRulesEnv, FCat)
genFCatArg ctype env@(FRulesEnv last_id fcatSet rules) m1@(FCat _ cat rcs tcs) =
  case Map.lookup cat fcatSet >>= Map.lookup rcs of
    Just tmap -> case Map.lookup tcs tmap of
                   Just (Left  fcat) -> (env,   fcat)
                   Just (Right fcat) -> (env,   fcat)
                   Nothing           -> ins tmap
    Nothing   -> ins Map.empty
  where
    ins tmap =
      let next_id = last_id+1
          fcat    = FCat next_id cat rcs tcs
          (x_fcat,last_id1,tmap1,rules1)
                  = foldBM (\tcs st (x_fcat,last_id,tmap,rules) ->
                                   let (last_id1,tmap1,fcat_arg) = addArg tcs last_id tmap
                                       rule = FRule (Abs fcat  [fcat_arg] coercionName)
                                                    (listArray (0,length rcs-1) [listArray (0,0) [FSymCat fcat_arg lbl 0] | lbl <- [0..length rcs-1]])
                                   in if st
                                        then (Right fcat,last_id1,tmap1,rule:rules)
                                        else (x_fcat,    last_id,  tmap,     rules))
                           (Left fcat,next_id,Map.insert tcs x_fcat tmap,rules)
                           (gen_tcs ctype emptyPath [])
                           False
          rmap1 = Map.singleton rcs tmap1
      in (FRulesEnv last_id1 (Map.insertWith (\_ -> Map.insert rcs tmap1) cat rmap1 fcatSet) rules1, fcat)
      where
        addArg tcs last_id tmap =
	  case Map.lookup tcs tmap of
	    Just (Left  fcat) -> (last_id, tmap, fcat)
	    Just (Right fcat) -> (last_id, tmap, fcat)
	    Nothing           -> let next_id = last_id+1
	                             fcat    = FCat next_id cat rcs tcs
                                 in (next_id, Map.insert tcs (Left fcat) tmap, fcat)

    gen_tcs :: SLinType -> SPath -> [(SPath,STerm)] -> BacktrackM Bool [(SPath,STerm)]
    gen_tcs (RecT record)      path acc = foldM (\acc (label,ctype) -> gen_tcs ctype (path ++. label) acc) acc record
    gen_tcs (TblT terms ctype) path acc = foldM (\acc term          -> gen_tcs ctype (path ++! term ) acc) acc terms
    gen_tcs (StrT)             path acc = return acc
    gen_tcs (ConT terms)       path acc =
      case List.lookup path tcs of
        Just term -> return $! addConstraint path term acc
        Nothing   -> do writeState True
                        term <- member terms
                        return $! addConstraint path term acc
      where
        addConstraint path0 term0 (c@(path,term) : cs)
	  | path0 >  path = c:addConstraint path0 term0 cs
        addConstraint path0 term0 cs  = (path0,term0) : cs

takeToDoRules :: SRulesMap -> FRulesEnv -> ([([SRule], STermSelector)], FRulesEnv)
takeToDoRules srulesMap (FRulesEnv last_id fcatSet rules) = (todo,FRulesEnv last_id fcatSet' rules)
  where
    (todo,fcatSet') =
          Map.mapAccumWithKey (\todo cat rmap ->
                let (todo1,rmap1) = Map.mapAccumWithKey (\todo rcs tmap -> 
                                          let (tcss,tmap') = Map.mapAccumWithKey (\tcss tcs x_fcat ->
                                                                   case x_fcat of
                                                                     Left  fcat -> (tcs:tcss,Right fcat)
                                                                     Right fcat -> (    tcss,    x_fcat)) [] tmap
                                          in case tcss of
                                               [] -> (                               todo,tmap )
                                               _  -> ((srules,mkSelector rcs tcss) : todo,tmap')) todo rmap
                    mb_srules   = Map.lookup cat srulesMap
                    Just srules = mb_srules

                in case mb_srules of
                     Just srules -> (todo1,rmap1)
                     Nothing     -> (todo ,rmap1)) [] fcatSet

addFCatRule :: FRulesEnv -> FRule -> FRulesEnv
addFCatRule (FRulesEnv last_id fcatSet rules) rule = FRulesEnv last_id fcatSet (rule:rules)

getFRules :: FRulesEnv -> [FRule]
getFRules (FRulesEnv last_id fcatSet rules) = rules


------------------------------------------------------------
-- The STermSelector

data STermSelector
  = RecSel [(Label, STermSelector)]
  | TblSel [(STerm, STermSelector)]
  | RecPrj   Label  STermSelector
  | TblPrj   STerm  STermSelector
  | ConSel [STerm]
  | StrSel

mkSingletonSelectors :: SLinType -> [STermSelector]
mkSingletonSelectors ctype = sels0
  where
    (sels0,tcss0) = loop emptyPath ([],[]) ctype
    
    loop path st          (RecT record)      = List.foldl' (\st (lbl,ctype) -> loop (path ++. lbl ) st ctype) st record
    loop path st          (TblT terms ctype) = List.foldl' (\st term        -> loop (path ++! term) st ctype) st terms
    loop path (sels,tcss) (ConT terms)       = (                          sels,map ((,) path) terms : tcss)
    loop path (sels,tcss) (StrT)             = (mkSelector [path] tcss0 : sels,                       tcss)


mkSelector :: [SPath] -> [[(SPath,STerm)]] -> STermSelector
mkSelector rcs tcss =
  List.foldl' addRestriction (case xs of
                                (path:xs) -> List.foldl' addProjection (path2selector StrSel path) xs) ys
  where
    xs = [ reverse path       |               Path path       <- rcs]
    ys = [(reverse path,term) | tcs <- tcss, (Path path,term) <- tcs]
    
    addProjection :: STermSelector -> [Either Label STerm] -> STermSelector
    addProjection StrSel          []                 = StrSel
    addProjection (RecSel fields) (Left  lbl : path) = RecSel (add fields)
      where
        add []                      = [(lbl,path2selector StrSel path)]
        add (field@(lbl',sub_sel):fields)
          | lbl == lbl'             = (lbl',addProjection sub_sel path):fields
          | otherwise               = field : add fields
    addProjection (TblSel cases)  (Right pat : path) = TblSel (add cases)
      where
        add []                      = [(pat,path2selector StrSel path)]
        add (cas@(pat',sub_sel):cases)
          | pat == pat'             = (pat',addProjection sub_sel path):cases
          | otherwise               = cas : add cases

    addRestriction :: STermSelector -> ([Either Label STerm],STerm) -> STermSelector
    addRestriction (ConSel terms)  ([]              ,term) = ConSel (add terms)
      where
        add []                      = [term]
        add (term':terms)
          | term == term'           = term':    terms
          | otherwise               = term':add terms
    addRestriction (RecSel fields) (Left  lbl : path,term) = RecSel (add fields)
      where
        add []                      = [(lbl,path2selector (ConSel [term]) path)]
        add (field@(lbl',sub_sel):fields)
          | lbl == lbl'             = (lbl',addRestriction sub_sel (path,term)):fields
          | otherwise               = field : add fields
    addRestriction (TblSel cases)  (Right pat : path,term) = TblSel (add cases)
      where
        add []                      = [(pat,path2selector (ConSel [term]) path)]
        add (field@(pat',sub_sel):cases)
          | pat == pat'             = (pat',addRestriction sub_sel (path,term)):cases
          | otherwise               = field : add cases

    path2selector base []                 = base
    path2selector base (Left  lbl : path) = RecSel [(lbl,path2selector base path)]
    path2selector base (Right sel : path) = TblSel [(sel,path2selector base path)]


------------------------------------------------------------
-- updating the MCF rule

readArgCType :: Int -> CnvMonad SLinType
readArgCType arg = do (_, _, _, ctypes) <- readState
		      return (ctypes !! arg)

restrictArg :: Int -> SPath -> STerm -> CnvMonad ()
restrictArg nr path term = do
  (head, args, ctype, ctypes) <- readState 
  args' <- updateNthM (\(fcat,xs) -> do fcat <- restrictFCat path term fcat
                                        return (fcat,xs)                   ) nr args
  writeState (head, args', ctype, ctypes)

projectArg :: Int -> SPath -> CnvMonad Int
projectArg nr path = do
  (head, args, ctype, ctypes) <- readState
  (xnr,args') <- updateArgs nr args
  writeState (head, args', ctype, ctypes)
  return xnr
  where
    updateArgs :: Int -> [(FCat,[SPath])] -> CnvMonad (Int,[(FCat,[SPath])])
    updateArgs 0 ((a@(FCat _ _ rcs _),xpaths) : as)
      | path `elem` rcs = return (length xpaths+1,(a,path:xpaths):as)
      | otherwise       = do a <- projectFCat path a
                             return (0,(a,xpaths):as)
    updateArgs n (a : as) = do
      (xnr,as) <- updateArgs (n-1) as
      return (xnr,a:as)

readHeadCType :: CnvMonad SLinType
readHeadCType = do (_, _, ctype, _) <- readState
		   return ctype

restrictHead :: SPath -> STerm -> CnvMonad ()
restrictHead path term
    = do (head, args, ctype, ctypes) <- readState
	 head' <- restrictFCat path term head
	 writeState (head', args, ctype, ctypes)

projectHead :: SPath -> CnvMonad ()
projectHead path
    = do (head, args, ctype, ctypes) <- readState
	 head' <- projectFCat path head
	 writeState (head', args, ctype, ctypes)

restrictFCat :: SPath -> STerm -> FCat -> CnvMonad FCat
restrictFCat path0 term0 (FCat id cat rcs tcs) = do
  tcs <- addConstraint tcs
  return (FCat id cat rcs tcs)
  where
    addConstraint (c@(path,term) : cs)
        | path0 >  path = liftM (c:) (addConstraint cs)
        | path0 == path = guard (term0 == term) >>
                          return (c : cs)
    addConstraint cs    = return ((path0,term0) : cs)

projectFCat :: SPath -> FCat -> CnvMonad FCat
projectFCat path0 (FCat id cat rcs tcs) = do
  return (FCat id cat (addConstraint rcs) tcs)
  where
    addConstraint (path : rcs)
        | path0 >  path = path  : addConstraint rcs
        | path0 == path = path  : rcs
    addConstraint rcs   = path0 : rcs