module Idris.Core.Constraints(ucheck) where
import Idris.Core.TT
import Idris.Core.Typecheck
import Control.Applicative
import Control.Arrow
import Control.Monad.RWS
import Control.Monad.State
import qualified Data.Set as S
import Data.List
import Data.Maybe
import qualified Data.Map.Strict as M
import Debug.Trace
ucheck :: [(UConstraint, FC)] -> TC ()
ucheck cs = acyclic rels (map fst (M.toList rels))
  where lhs (ULT l _) = l
        lhs (ULE l _) = l
        cs' = S.toList (S.fromList cs)
        rels = mkRels cs' M.empty
type Relations = M.Map UExp [(UConstraint, FC)]
mkRels :: [(UConstraint, FC)] -> Relations -> Relations
mkRels [] acc = acc
mkRels ((c, f) : cs) acc
    | not (ignore c)
       = case M.lookup (lhs c) acc of
              Nothing -> mkRels cs (M.insert (lhs c) [(c,f)] acc)
              Just rs -> mkRels cs (M.insert (lhs c) ((c,f):rs) acc)
    | otherwise = mkRels cs acc
  where lhs (ULT l _) = l
        lhs (ULE l _) = l
        ignore (ULE l r) = l == r
        ignore _ = False
acyclic :: Relations -> [UExp] -> TC ()
acyclic r cvs = checkCycle (fileFC "root") r [] 0 cvs
  where
    checkCycle :: FC -> Relations -> [(UExp, FC)] -> Int -> [UExp] -> TC ()
    checkCycle fc r path inc [] = return ()
    checkCycle fc r path inc (c : cs)
        = do check fc r path inc c
             
             let r' = M.insert c [] r
             checkCycle fc r' path inc cs
    check fc r path inc (UVar x) | x < 0 = return ()
    check fc r path inc cv
        | inc > 0 && cv `elem` map fst path
            = Error $ At fc UniverseError
                
                
                
                
        
        
        | inc == 0 && cv `elem` map fst path
            = return ()
        | otherwise
             = case M.lookup cv r of
                    Nothing -> return ()
                    Just cs -> mapM_ (next r ((cv, fc):path) inc) cs
    next r path inc (ULT l x, fc) = check fc r path (inc + 1) x
    next r path inc (ULE l x, fc) = check fc r path inc x