{-# LANGUAGE CPP #-}
{-# LANGUAGE GeneralizedNewtypeDeriving #-}
{-# LANGUAGE UndecidableInstances #-}
#if __GLASGOW_HASKELL__ >= 802
{-# OPTIONS -Wno-simplifiable-class-constraints #-}
#endif
module Agda.TypeChecking.Primitive where
import Control.Monad
import Control.Monad.Reader (asks)
import Data.Char
import Data.Map (Map)
import qualified Data.Map as Map
import Data.Maybe
import Data.Traversable (traverse)
import Data.Monoid (mempty)
import Data.Word
import Agda.Interaction.Options
import qualified Agda.Interaction.Options.Lenses as Lens
import Agda.Syntax.Position
import Agda.Syntax.Common hiding (Nat)
import Agda.Syntax.Internal
import Agda.Syntax.Internal.Generic (TermLike(..))
import Agda.Syntax.Literal
import Agda.Syntax.Concrete.Pretty ()
import Agda.Syntax.Fixity
import Agda.TypeChecking.Monad hiding (getConstInfo, typeOfConst)
import qualified Agda.TypeChecking.Monad as TCM
import Agda.TypeChecking.Monad.Builtin
import Agda.TypeChecking.Reduce
import Agda.TypeChecking.Reduce.Monad
import Agda.TypeChecking.Substitute
import Agda.TypeChecking.Telescope
import Agda.TypeChecking.Errors
import Agda.TypeChecking.Functions
import Agda.TypeChecking.Level
import Agda.TypeChecking.Quote (QuotingKit, quoteTermWithKit, quoteTypeWithKit, quoteClauseWithKit, quotingKit)
import Agda.TypeChecking.Pretty ()
import Agda.TypeChecking.Warnings
import Agda.Utils.Functor
import Agda.Utils.List
import Agda.Utils.Maybe
import Agda.Utils.Monad
import Agda.Utils.Pretty (pretty, prettyShow)
import Agda.Utils.Size
import Agda.Utils.String ( Str(Str), unStr )
import Agda.Utils.Float
#include "undefined.h"
import Agda.Utils.Impossible
import Debug.Trace
data PrimitiveImpl = PrimImpl Type PrimFun
newtype Nat = Nat { unNat :: Integer }
deriving (Eq, Ord, Num, Enum, Real)
instance Integral Nat where
toInteger = unNat
quotRem (Nat a) (Nat b) = (Nat q, Nat r)
where (q, r) = quotRem a b
instance TermLike Nat where
traverseTermM _ = pure
foldTerm _ = mempty
instance Show Nat where
show = show . toInteger
newtype Lvl = Lvl { unLvl :: Integer }
deriving (Eq, Ord)
instance Show Lvl where
show = show . unLvl
class PrimType a where
primType :: a -> TCM Type
instance (PrimType a, PrimType b) => PrimTerm (a -> b) where
primTerm _ = unEl <$> (primType (undefined :: a) --> primType (undefined :: b))
instance PrimTerm a => PrimType a where
primType _ = el $ primTerm (undefined :: a)
class PrimTerm a where primTerm :: a -> TCM Term
instance PrimTerm Integer where primTerm _ = primInteger
instance PrimTerm Word64 where primTerm _ = primWord64
instance PrimTerm Bool where primTerm _ = primBool
instance PrimTerm Char where primTerm _ = primChar
instance PrimTerm Double where primTerm _ = primFloat
instance PrimTerm Str where primTerm _ = primString
instance PrimTerm Nat where primTerm _ = primNat
instance PrimTerm Lvl where primTerm _ = primLevel
instance PrimTerm QName where primTerm _ = primQName
instance PrimTerm MetaId where primTerm _ = primAgdaMeta
instance PrimTerm Type where primTerm _ = primAgdaTerm
instance PrimTerm Fixity' where primTerm _ = primFixity
instance PrimTerm a => PrimTerm [a] where
primTerm _ = list (primTerm (undefined :: a))
instance PrimTerm a => PrimTerm (IO a) where
primTerm _ = io (primTerm (undefined :: a))
class ToTerm a where
toTerm :: TCM (a -> Term)
toTermR :: TCM (a -> ReduceM Term)
toTermR = (pure .) <$> toTerm
instance ToTerm Nat where toTerm = return $ Lit . LitNat noRange . toInteger
instance ToTerm Word64 where toTerm = return $ Lit . LitWord64 noRange
instance ToTerm Lvl where toTerm = return $ Level . Max . (:[]) . ClosedLevel . unLvl
instance ToTerm Double where toTerm = return $ Lit . LitFloat noRange
instance ToTerm Char where toTerm = return $ Lit . LitChar noRange
instance ToTerm Str where toTerm = return $ Lit . LitString noRange . unStr
instance ToTerm QName where toTerm = return $ Lit . LitQName noRange
instance ToTerm MetaId where
toTerm = do
file <- fromMaybe __IMPOSSIBLE__ <$> asks TCM.envCurrentPath
return $ Lit . LitMeta noRange file
instance ToTerm Integer where
toTerm = do
pos <- primIntegerPos
negsuc <- primIntegerNegSuc
fromNat <- toTerm :: TCM (Nat -> Term)
let intToTerm = fromNat . fromIntegral :: Integer -> Term
let fromInt n | n >= 0 = apply pos [defaultArg $ intToTerm n]
| otherwise = apply negsuc [defaultArg $ intToTerm (-n - 1)]
return fromInt
instance ToTerm Bool where
toTerm = do
true <- primTrue
false <- primFalse
return $ \b -> if b then true else false
instance ToTerm Term where
toTerm = do kit <- quotingKit; runReduceF (quoteTermWithKit kit)
toTermR = do kit <- quotingKit; return (quoteTermWithKit kit)
instance ToTerm Type where
toTerm = do kit <- quotingKit; runReduceF (quoteTypeWithKit kit)
toTermR = do kit <- quotingKit; return (quoteTypeWithKit kit)
instance ToTerm ArgInfo where
toTerm = do
info <- primArgArgInfo
vis <- primVisible
hid <- primHidden
ins <- primInstance
rel <- primRelevant
irr <- primIrrelevant
return $ \ i -> info `applys`
[ case getHiding i of
NotHidden -> vis
Hidden -> hid
Instance{} -> ins
, case getRelevance i of
Relevant -> rel
Irrelevant -> irr
NonStrict -> rel
]
instance ToTerm Fixity' where
toTerm = (. theFixity) <$> toTerm
instance ToTerm Fixity where
toTerm = do
lToTm <- toTerm
aToTm <- toTerm
fixity <- primFixityFixity
return $ \ Fixity{fixityAssoc = a, fixityLevel = l} ->
fixity `apply` [defaultArg (aToTm a), defaultArg (lToTm l)]
instance ToTerm Associativity where
toTerm = do
lassoc <- primAssocLeft
rassoc <- primAssocRight
nassoc <- primAssocNon
return $ \ a ->
case a of
NonAssoc -> nassoc
LeftAssoc -> lassoc
RightAssoc -> rassoc
instance ToTerm PrecedenceLevel where
toTerm = do
(iToTm :: Integer -> Term) <- toTerm
related <- primPrecRelated
unrelated <- primPrecUnrelated
return $ \ p ->
case p of
Unrelated -> unrelated
Related n -> related `apply` [defaultArg $ iToTm n]
buildList :: TCM ([Term] -> Term)
buildList = do
nil' <- primNil
cons' <- primCons
let nil = nil'
cons x xs = cons' `applys` [x, xs]
return $ foldr cons nil
instance ToTerm a => ToTerm [a] where
toTerm = do
mkList <- buildList
fromA <- toTerm
return $ mkList . map fromA
type FromTermFunction a = Arg Term ->
ReduceM (Reduced (MaybeReduced (Arg Term)) a)
class FromTerm a where
fromTerm :: TCM (FromTermFunction a)
instance FromTerm Integer where
fromTerm = do
Con pos _ [] <- primIntegerPos
Con negsuc _ [] <- primIntegerNegSuc
toNat <- fromTerm :: TCM (FromTermFunction Nat)
return $ \ v -> do
b <- reduceB' v
let v' = ignoreBlocking b
arg = (<$ v')
case unArg (ignoreBlocking b) of
Con c ci [Apply u]
| c == pos ->
redBind (toNat u)
(\ u' -> notReduced $ arg $ Con c ci [Apply $ ignoreReduced u']) $ \ n ->
redReturn $ fromIntegral n
| c == negsuc ->
redBind (toNat u)
(\ u' -> notReduced $ arg $ Con c ci [Apply $ ignoreReduced u']) $ \ n ->
redReturn $ fromIntegral $ -n - 1
_ -> return $ NoReduction (reduced b)
instance FromTerm Nat where
fromTerm = fromLiteral $ \l -> case l of
LitNat _ n -> Just $ fromInteger n
_ -> Nothing
instance FromTerm Word64 where
fromTerm = fromLiteral $ \ case
LitWord64 _ n -> Just n
_ -> Nothing
instance FromTerm Lvl where
fromTerm = fromReducedTerm $ \l -> case l of
Level (Max [ClosedLevel n]) -> Just $ Lvl n
_ -> Nothing
instance FromTerm Double where
fromTerm = fromLiteral $ \l -> case l of
LitFloat _ x -> Just x
_ -> Nothing
instance FromTerm Char where
fromTerm = fromLiteral $ \l -> case l of
LitChar _ c -> Just c
_ -> Nothing
instance FromTerm Str where
fromTerm = fromLiteral $ \l -> case l of
LitString _ s -> Just $ Str s
_ -> Nothing
instance FromTerm QName where
fromTerm = fromLiteral $ \l -> case l of
LitQName _ x -> Just x
_ -> Nothing
instance FromTerm MetaId where
fromTerm = fromLiteral $ \l -> case l of
LitMeta _ _ x -> Just x
_ -> Nothing
instance FromTerm Bool where
fromTerm = do
true <- primTrue
false <- primFalse
fromReducedTerm $ \t -> case t of
_ | t =?= true -> Just True
| t =?= false -> Just False
| otherwise -> Nothing
where
a =?= b = a === b
Def x [] === Def y [] = x == y
Con x _ [] === Con y _ [] = x == y
Var n [] === Var m [] = n == m
_ === _ = False
instance (ToTerm a, FromTerm a) => FromTerm [a] where
fromTerm = do
nil' <- primNil
cons' <- primCons
nil <- isCon nil'
cons <- isCon cons'
toA <- fromTerm
fromA <- toTerm
return $ mkList nil cons toA fromA
where
isCon (Lam _ b) = isCon $ absBody b
isCon (Con c _ _)= return c
isCon v = __IMPOSSIBLE__
mkList nil cons toA fromA t = do
b <- reduceB' t
let t = ignoreBlocking b
let arg = (<$ t)
case unArg t of
Con c ci []
| c == nil -> return $ YesReduction NoSimplification []
Con c ci es
| c == cons, Just [x,xs] <- allApplyElims es ->
redBind (toA x)
(\x' -> notReduced $ arg $ Con c ci (map Apply [ignoreReduced x',xs])) $ \y ->
redBind
(mkList nil cons toA fromA xs)
(fmap $ \xs' -> arg $ Con c ci (map Apply [defaultArg $ fromA y, xs'])) $ \ys ->
redReturn (y : ys)
_ -> return $ NoReduction (reduced b)
redBind :: ReduceM (Reduced a a') -> (a -> b) ->
(a' -> ReduceM (Reduced b b')) -> ReduceM (Reduced b b')
redBind ma f k = do
r <- ma
case r of
NoReduction x -> return $ NoReduction $ f x
YesReduction _ y -> k y
redReturn :: a -> ReduceM (Reduced a' a)
redReturn = return . YesReduction YesSimplification
fromReducedTerm :: (Term -> Maybe a) -> TCM (FromTermFunction a)
fromReducedTerm f = return $ \t -> do
b <- reduceB' t
case f $ unArg (ignoreBlocking b) of
Just x -> return $ YesReduction NoSimplification x
Nothing -> return $ NoReduction (reduced b)
fromLiteral :: (Literal -> Maybe a) -> TCM (FromTermFunction a)
fromLiteral f = fromReducedTerm $ \t -> case t of
Lit lit -> f lit
_ -> Nothing
primTrustMe :: TCM PrimitiveImpl
primTrustMe = do
whenM (Lens.getSafeMode <$> commandLineOptions) $ warning SafeFlagPrimTrustMe
eq <- primEqualityName
eqTy <- defType <$> getConstInfo eq
TelV eqTel eqCore <- telView eqTy
let eqSort = case unEl eqCore of
Sort s -> s
_ -> __IMPOSSIBLE__
let t = telePi_ (fmap hide eqTel) $ El eqSort $ Def eq $ map Apply $ teleArgs eqTel
con@(Con rf ci []) <- primRefl
minfo <- fmap (setOrigin Inserted) <$> getReflArgInfo rf
let (refl :: Arg Term -> Term) = case minfo of
Just ai -> Con rf ci . (:[]) . Apply . setArgInfo ai
Nothing -> const con
return $ PrimImpl t $ PrimFun __IMPOSSIBLE__ (size eqTel) $ \ ts -> do
let (u, v) = fromMaybe __IMPOSSIBLE__ $ last2 ts
(u', v') <- normalise' (u, v)
if u' == v' then redReturn $ refl u else
return $ NoReduction $ map notReduced ts
getReflArgInfo :: ConHead -> TCM (Maybe ArgInfo)
getReflArgInfo rf = do
def <- getConInfo rf
TelV reflTel _ <- telView $ defType def
return $ fmap getArgInfo $ headMaybe $ drop (conPars $ theDef def) $ telToList reflTel
genPrimForce :: TCM Type -> (Term -> Arg Term -> Term) -> TCM PrimitiveImpl
genPrimForce b ret = do
let varEl s a = El (varSort s) <$> a
varT s a = varEl s (varM a)
varS s = pure $ sort $ varSort s
t <- hPi "a" (el primLevel) $
hPi "b" (el primLevel) $
hPi "A" (varS 1) $
hPi "B" (varT 2 0 --> varS 1) b
return $ PrimImpl t $ PrimFun __IMPOSSIBLE__ 6 $ \ ts ->
case ts of
[a, b, s, t, u, f] -> do
u <- reduceB' u
let isWHNF Blocked{} = return False
isWHNF (NotBlocked _ u) =
case unArg u of
Lit{} -> return True
Con{} -> return True
Lam{} -> return True
Pi{} -> return True
Sort{} -> return True
Level{} -> return True
DontCare{} -> return True
Def q _ -> do
def <- theDef <$> getConstInfo q
return $ case def of
Datatype{} -> True
Record{} -> True
_ -> False
MetaV{} -> return False
Var{} -> return False
ifM (isWHNF u)
(redReturn $ ret (unArg f) (ignoreBlocking u))
(return $ NoReduction $ map notReduced [a, b, s, t] ++ [reduced u, notReduced f])
_ -> __IMPOSSIBLE__
primForce :: TCM PrimitiveImpl
primForce = do
let varEl s a = El (varSort s) <$> a
varT s a = varEl s (varM a)
varS s = pure $ sort $ varSort s
genPrimForce (nPi "x" (varT 3 1) $
(nPi "y" (varT 4 2) $ varEl 4 $ varM 2 <@> varM 0) -->
varEl 3 (varM 1 <@> varM 0)) $
\ f u -> apply f [u]
primForceLemma :: TCM PrimitiveImpl
primForceLemma = do
let varEl s a = El (varSort s) <$> a
varT s a = varEl s (varM a)
varS s = pure $ sort $ varSort s
refl <- primRefl
force <- primFunName <$> getPrimitive "primForce"
genPrimForce (nPi "x" (varT 3 1) $
nPi "f" (nPi "y" (varT 4 2) $ varEl 4 $ varM 2 <@> varM 0) $
varEl 4 $ primEquality <#> varM 4 <#> (varM 2 <@> varM 1)
<@> (pure (Def force []) <#> varM 5 <#> varM 4 <#> varM 3 <#> varM 2 <@> varM 1 <@> varM 0)
<@> (varM 0 <@> varM 1)
) $ \ _ _ -> refl
mkPrimLevelZero :: TCM PrimitiveImpl
mkPrimLevelZero = do
t <- primType (undefined :: Lvl)
return $ PrimImpl t $ PrimFun __IMPOSSIBLE__ 0 $ \_ -> redReturn $ Level $ Max []
mkPrimLevelSuc :: TCM PrimitiveImpl
mkPrimLevelSuc = do
t <- primType (id :: Lvl -> Lvl)
return $ PrimImpl t $ PrimFun __IMPOSSIBLE__ 1 $ \ ~[a] -> do
l <- levelView' $ unArg a
redReturn $ Level $ levelSuc l
mkPrimLevelMax :: TCM PrimitiveImpl
mkPrimLevelMax = do
t <- primType (max :: Op Lvl)
return $ PrimImpl t $ PrimFun __IMPOSSIBLE__ 2 $ \ ~[a, b] -> do
Max as <- levelView' $ unArg a
Max bs <- levelView' $ unArg b
redReturn $ Level $ levelMax $ as ++ bs
mkPrimFun1TCM :: (FromTerm a, ToTerm b, TermLike b) =>
TCM Type -> (a -> ReduceM b) -> TCM PrimitiveImpl
mkPrimFun1TCM mt f = do
toA <- fromTerm
fromB <- toTermR
t <- mt
return $ PrimImpl t $ PrimFun __IMPOSSIBLE__ 1 $ \ts ->
case ts of
[v] ->
redBind (toA v) (\v' -> [v']) $ \x -> do
b <- f x
case allMetas b of
(m:_) -> return $ NoReduction [reduced (Blocked m v)]
[] -> redReturn =<< fromB b
_ -> __IMPOSSIBLE__
mkPrimFun1 :: (PrimType a, FromTerm a, PrimType b, ToTerm b) =>
(a -> b) -> TCM PrimitiveImpl
mkPrimFun1 f = do
toA <- fromTerm
fromB <- toTerm
t <- primType f
return $ PrimImpl t $ PrimFun __IMPOSSIBLE__ 1 $ \ts ->
case ts of
[v] ->
redBind (toA v)
(\v' -> [v']) $ \x ->
redReturn $ fromB $ f x
_ -> __IMPOSSIBLE__
mkPrimFun2 :: ( PrimType a, FromTerm a, ToTerm a
, PrimType b, FromTerm b
, PrimType c, ToTerm c ) =>
(a -> b -> c) -> TCM PrimitiveImpl
mkPrimFun2 f = do
toA <- fromTerm
fromA <- toTerm
toB <- fromTerm
fromC <- toTerm
t <- primType f
return $ PrimImpl t $ PrimFun __IMPOSSIBLE__ 2 $ \ts ->
case ts of
[v,w] ->
redBind (toA v)
(\v' -> [v', notReduced w]) $ \x ->
redBind (toB w)
(\w' -> [ reduced $ notBlocked $ Arg (argInfo v) (fromA x)
, w']) $ \y ->
redReturn $ fromC $ f x y
_ -> __IMPOSSIBLE__
mkPrimFun4 :: ( PrimType a, FromTerm a, ToTerm a
, PrimType b, FromTerm b, ToTerm b
, PrimType c, FromTerm c, ToTerm c
, PrimType d, FromTerm d
, PrimType e, ToTerm e ) =>
(a -> b -> c -> d -> e) -> TCM PrimitiveImpl
mkPrimFun4 f = do
(toA, fromA) <- (,) <$> fromTerm <*> toTerm
(toB, fromB) <- (,) <$> fromTerm <*> toTerm
(toC, fromC) <- (,) <$> fromTerm <*> toTerm
toD <- fromTerm
fromE <- toTerm
t <- primType f
return $ PrimImpl t $ PrimFun __IMPOSSIBLE__ 4 $ \ts ->
let argFrom fromX a x =
reduced $ notBlocked $ Arg (argInfo a) (fromX x)
in case ts of
[a,b,c,d] ->
redBind (toA a)
(\a' -> a' : map notReduced [b,c,d]) $ \x ->
redBind (toB b)
(\b' -> [argFrom fromA a x, b', notReduced c, notReduced d]) $ \y ->
redBind (toC c)
(\c' -> [ argFrom fromA a x
, argFrom fromB b y
, c', notReduced d]) $ \z ->
redBind (toD d)
(\d' -> [ argFrom fromA a x
, argFrom fromB b y
, argFrom fromC c z
, d']) $ \w ->
redReturn $ fromE $ f x y z w
_ -> __IMPOSSIBLE__
infixr 4 -->
infixr 4 .-->
infixr 4 ..-->
(-->), (.-->), (..-->) :: TCM Type -> TCM Type -> TCM Type
a --> b = garr id a b
a .--> b = garr (const $ Irrelevant) a b
a ..--> b = garr (const $ NonStrict) a b
garr :: (Relevance -> Relevance) -> TCM Type -> TCM Type -> TCM Type
garr f a b = do
a' <- a
b' <- b
return $ El (funSort (getSort a') (getSort b')) $
Pi (Dom (mapRelevance f defaultArgInfo) a') (NoAbs "_" b')
gpi :: ArgInfo -> String -> TCM Type -> TCM Type -> TCM Type
gpi info name a b = do
a <- a
b <- addContext (name, Dom info a) b
let y = stringToArgName name
return $ El (piSort (getSort a) (Abs y (getSort b)))
(Pi (Dom info a) (Abs y b))
hPi, nPi :: String -> TCM Type -> TCM Type -> TCM Type
hPi = gpi $ setHiding Hidden defaultArgInfo
nPi = gpi defaultArgInfo
varM :: Int -> TCM Term
varM = return . var
infixl 9 <@>, <#>
gApply :: Hiding -> TCM Term -> TCM Term -> TCM Term
gApply h a b = do
x <- a
y <- b
return $ x `apply` [Arg (setHiding h defaultArgInfo) y]
(<@>),(<#>) :: TCM Term -> TCM Term -> TCM Term
(<@>) = gApply NotHidden
(<#>) = gApply Hidden
list :: TCM Term -> TCM Term
list t = primList <@> t
io :: TCM Term -> TCM Term
io t = primIO <@> t
el :: TCM Term -> TCM Type
el t = El (mkType 0) <$> t
tset :: TCM Type
tset = return $ sort (mkType 0)
sSizeUniv :: Sort
sSizeUniv = mkType 0
tSizeUniv :: TCM Type
tSizeUniv = tset
argN :: e -> Arg e
argN = Arg defaultArgInfo
domN :: e -> Dom e
domN = Dom defaultArgInfo
argH :: e -> Arg e
argH = Arg $ setHiding Hidden defaultArgInfo
domH :: e -> Dom e
domH = Dom $ setHiding Hidden defaultArgInfo
type Op a = a -> a -> a
type Fun a = a -> a
type Rel a = a -> a -> Bool
type Pred a = a -> Bool
primitiveFunctions :: Map String (TCM PrimitiveImpl)
primitiveFunctions = Map.fromList
[ "primShowInteger" |-> mkPrimFun1 (Str . show :: Integer -> Str)
, "primNatPlus" |-> mkPrimFun2 ((+) :: Op Nat)
, "primNatMinus" |-> mkPrimFun2 ((\x y -> max 0 (x - y)) :: Op Nat)
, "primNatTimes" |-> mkPrimFun2 ((*) :: Op Nat)
, "primNatDivSucAux" |-> mkPrimFun4 ((\k m n j -> k + div (max 0 $ n + m - j) (m + 1)) :: Nat -> Nat -> Nat -> Nat -> Nat)
, "primNatModSucAux" |->
let aux :: Nat -> Nat -> Nat -> Nat -> Nat
aux k m n j | n > j = mod (n - j - 1) (m + 1)
| otherwise = k + n
in mkPrimFun4 aux
, "primNatEquality" |-> mkPrimFun2 ((==) :: Rel Nat)
, "primNatLess" |-> mkPrimFun2 ((<) :: Rel Nat)
, "primWord64ToNat" |-> mkPrimFun1 (fromIntegral :: Word64 -> Nat)
, "primWord64FromNat" |-> mkPrimFun1 (fromIntegral :: Nat -> Word64)
, "primLevelZero" |-> mkPrimLevelZero
, "primLevelSuc" |-> mkPrimLevelSuc
, "primLevelMax" |-> mkPrimLevelMax
, "primNatToFloat" |-> mkPrimFun1 (fromIntegral :: Nat -> Double)
, "primFloatPlus" |-> mkPrimFun2 ((+) :: Op Double)
, "primFloatMinus" |-> mkPrimFun2 ((-) :: Op Double)
, "primFloatTimes" |-> mkPrimFun2 ((*) :: Op Double)
, "primFloatNegate" |-> mkPrimFun1 (negate :: Fun Double)
, "primFloatDiv" |-> mkPrimFun2 ((/) :: Op Double)
, "primFloatEquality" |-> mkPrimFun2 (floatEq :: Rel Double)
, "primFloatLess" |-> mkPrimFun2 (floatLt :: Rel Double)
, "primFloatNumericalEquality" |-> mkPrimFun2 ((==) :: Rel Double)
, "primFloatNumericalLess" |-> mkPrimFun2 ((<) :: Rel Double)
, "primFloatSqrt" |-> mkPrimFun1 (sqrt :: Double -> Double)
, "primRound" |-> mkPrimFun1 (round :: Double -> Integer)
, "primFloor" |-> mkPrimFun1 (floor :: Double -> Integer)
, "primCeiling" |-> mkPrimFun1 (ceiling :: Double -> Integer)
, "primExp" |-> mkPrimFun1 (exp :: Fun Double)
, "primLog" |-> mkPrimFun1 (log :: Fun Double)
, "primSin" |-> mkPrimFun1 (sin :: Fun Double)
, "primCos" |-> mkPrimFun1 (cos :: Fun Double)
, "primTan" |-> mkPrimFun1 (tan :: Fun Double)
, "primASin" |-> mkPrimFun1 (asin :: Fun Double)
, "primACos" |-> mkPrimFun1 (acos :: Fun Double)
, "primATan" |-> mkPrimFun1 (atan :: Fun Double)
, "primATan2" |-> mkPrimFun2 (atan2 :: Double -> Double -> Double)
, "primShowFloat" |-> mkPrimFun1 (Str . show :: Double -> Str)
, "primCharEquality" |-> mkPrimFun2 ((==) :: Rel Char)
, "primIsLower" |-> mkPrimFun1 isLower
, "primIsDigit" |-> mkPrimFun1 isDigit
, "primIsAlpha" |-> mkPrimFun1 isAlpha
, "primIsSpace" |-> mkPrimFun1 isSpace
, "primIsAscii" |-> mkPrimFun1 isAscii
, "primIsLatin1" |-> mkPrimFun1 isLatin1
, "primIsPrint" |-> mkPrimFun1 isPrint
, "primIsHexDigit" |-> mkPrimFun1 isHexDigit
, "primToUpper" |-> mkPrimFun1 toUpper
, "primToLower" |-> mkPrimFun1 toLower
, "primCharToNat" |-> mkPrimFun1 (fromIntegral . fromEnum :: Char -> Nat)
, "primNatToChar" |-> mkPrimFun1 (toEnum . fromIntegral . (`mod` 0x110000) :: Nat -> Char)
, "primShowChar" |-> mkPrimFun1 (Str . show . pretty . LitChar noRange)
, "primStringToList" |-> mkPrimFun1 unStr
, "primStringFromList" |-> mkPrimFun1 Str
, "primStringAppend" |-> mkPrimFun2 (\s1 s2 -> Str $ unStr s1 ++ unStr s2)
, "primStringEquality" |-> mkPrimFun2 ((==) :: Rel Str)
, "primShowString" |-> mkPrimFun1 (Str . show . pretty . LitString noRange . unStr)
, "primTrustMe" |-> primTrustMe
, "primForce" |-> primForce
, "primForceLemma" |-> primForceLemma
, "primQNameEquality" |-> mkPrimFun2 ((==) :: Rel QName)
, "primQNameLess" |-> mkPrimFun2 ((<) :: Rel QName)
, "primShowQName" |-> mkPrimFun1 (Str . show :: QName -> Str)
, "primQNameFixity" |-> mkPrimFun1 (nameFixity . qnameName)
, "primMetaEquality" |-> mkPrimFun2 ((==) :: Rel MetaId)
, "primMetaLess" |-> mkPrimFun2 ((<) :: Rel MetaId)
, "primShowMeta" |-> mkPrimFun1 (Str . show . pretty :: MetaId -> Str)
]
where
(|->) = (,)
lookupPrimitiveFunction :: String -> TCM PrimitiveImpl
lookupPrimitiveFunction x =
fromMaybe (typeError $ NoSuchPrimitiveFunction x)
(Map.lookup x primitiveFunctions)
lookupPrimitiveFunctionQ :: QName -> TCM (String, PrimitiveImpl)
lookupPrimitiveFunctionQ q = do
let s = case qnameName q of
Name _ x _ _ -> prettyShow x
PrimImpl t pf <- lookupPrimitiveFunction s
return (s, PrimImpl t $ pf { primFunName = q })
getBuiltinName :: String -> TCM (Maybe QName)
getBuiltinName b = do
caseMaybeM (getBuiltin' b) (return Nothing) (Just <.> getName)
where
getName v = do
v <- reduce v
case unSpine $ v of
Def x _ -> return x
Con x _ _ -> return $ conName x
Lam _ b -> getName $ unAbs b
_ -> __IMPOSSIBLE__
isBuiltin :: QName -> String -> TCM Bool
isBuiltin q b = (Just q ==) <$> getBuiltinName b