module Data.LLVM.BitCode.IR.Constants where
import Data.LLVM.BitCode.Bitstream
import Data.LLVM.BitCode.IR.Values
import Data.LLVM.BitCode.Match
import Data.LLVM.BitCode.Parse
import Data.LLVM.BitCode.Record
import Text.LLVM.AST
import Control.Monad (mplus,mzero,foldM,(<=<))
import Control.Monad.ST (runST,ST)
import Data.Array.ST (newArray,readArray,MArray,STUArray)
import Data.Bits (shiftL,shiftR,testBit)
import Data.Char (chr)
import Data.Maybe (fromMaybe)
import Data.Word (Word32,Word64)
import qualified Data.Map as Map
#if __GLASGOW_HASKELL__ >= 704
import Data.Array.Unsafe (castSTUArray)
#else
import Data.Array.ST (castSTUArray)
#endif
binop :: Match Field (Maybe Int -> Typed PValue -> PValue -> PInstr)
binop = choose <=< numeric
where
constant = return . const
nuw x = testBit x 0
nsw x = testBit x 1
wrapFlags i k = return $ \ mb x y ->
case mb of
Nothing -> i (k False False) x y
Just w -> i (k (nuw w) (nsw w)) x y
exact x = testBit x 0
exactFlag i k = return $ \ mb x y ->
case mb of
Nothing -> i (k False) x y
Just w -> i (k (exact w)) x y
choose :: Match Int (Maybe Int -> Typed PValue -> PValue -> PInstr)
choose 0 = wrapFlags Arith Add
choose 1 = wrapFlags Arith Sub
choose 2 = wrapFlags Arith Mul
choose 3 = exactFlag Arith UDiv
choose 4 = exactFlag Arith SDiv
choose 5 = constant (Arith URem)
choose 6 = constant (Arith SRem)
choose 7 = wrapFlags Bit Shl
choose 8 = exactFlag Bit Lshr
choose 9 = exactFlag Bit Ashr
choose 10 = constant (Bit And)
choose 11 = constant (Bit Or)
choose 12 = constant (Bit Xor)
choose _ = mzero
fcmpOp :: Match Field (Typed PValue -> PValue -> PInstr)
fcmpOp = choose <=< numeric
where
op = return . FCmp
choose :: Match Int (Typed PValue -> PValue -> PInstr)
choose 0 = op Ffalse
choose 1 = op Foeq
choose 2 = op Fogt
choose 3 = op Foge
choose 4 = op Folt
choose 5 = op Fole
choose 6 = op Fone
choose 7 = op Ford
choose 8 = op Funo
choose 9 = op Fueq
choose 10 = op Fugt
choose 11 = op Fuge
choose 12 = op Fult
choose 13 = op Fule
choose 14 = op Fune
choose 15 = op Ftrue
choose _ = mzero
icmpOp :: Match Field (Typed PValue -> PValue -> PInstr)
icmpOp = choose <=< numeric
where
op = return . ICmp
choose :: Match Int (Typed PValue -> PValue -> PInstr)
choose 32 = op Ieq
choose 33 = op Ine
choose 34 = op Iugt
choose 35 = op Iuge
choose 36 = op Iult
choose 37 = op Iule
choose 38 = op Isgt
choose 39 = op Isge
choose 40 = op Islt
choose 41 = op Isle
choose _ = mzero
castOp :: Match Field (Typed PValue -> Type -> PInstr)
castOp = choose <=< numeric
where
op = return . Conv
choose :: Match Int (Typed PValue -> Type -> PInstr)
choose 0 = op Trunc
choose 1 = op ZExt
choose 2 = op SExt
choose 3 = op FpToUi
choose 4 = op FpToSi
choose 5 = op UiToFp
choose 6 = op SiToFp
choose 7 = op FpTrunc
choose 8 = op FpExt
choose 9 = op PtrToInt
choose 10 = op IntToPtr
choose 11 = op BitCast
choose _ = mzero
castOpCE :: Match Field (Typed PValue -> Type -> PValue)
castOpCE = choose <=< numeric
where
op c = return (\ tv t -> ValConstExpr (ConstConv c tv t))
choose :: Match Int (Typed PValue -> Type -> PValue)
choose 0 = op Trunc
choose 1 = op ZExt
choose 2 = op SExt
choose 3 = op FpToUi
choose 4 = op FpToSi
choose 5 = op UiToFp
choose 6 = op SiToFp
choose 7 = op FpTrunc
choose 8 = op FpExt
choose 9 = op PtrToInt
choose 10 = op IntToPtr
choose 11 = op BitCast
choose _ = mzero
type ConstantTable = Map.Map Int (Typed Value)
cstGep :: Match Entry Record
cstGep = hasRecordCode 12 <=< fromEntry
cstInboundsGep :: Match Entry Record
cstInboundsGep = hasRecordCode 20 <=< fromEntry
setCurType :: Int -> Parse Type
setCurType = getType'
parseConstantsBlock :: [Entry] -> Parse ()
parseConstantsBlock es = fixValueTable_ $ \ vs' -> do
let curTy = fail "no current type id set"
(_,vs) <- foldM (parseConstantEntry vs') (curTy,[]) es
return vs
parseConstantEntry :: ValueTable -> (Parse Type,[Typed PValue]) -> Entry
-> Parse (Parse Type, [Typed PValue])
parseConstantEntry t (getTy,cs) (fromEntry -> Just r) =
label "CONSTANTS_BLOCK" $ case recordCode r of
1 -> label "CST_CODE_SETTYPE" $ do
let field = parseField r
i <- field 0 numeric
return (setCurType i, cs)
2 -> label "CST_CODE_NULL" $ do
ty <- getTy
val <- resolveNull ty
return (getTy, Typed ty val:cs)
3 -> label "CST_CODE_UNDEF" $ do
ty <- getTy
return (getTy, Typed ty ValUndef:cs)
4 -> label "CST_CODE_INTEGER" $ do
let field = parseField r
ty <- getTy
n <- field 0 signed
let val = fromMaybe (ValInteger n) $ do
Integer 0 <- elimPrimType ty
return (ValBool (n /= 0))
return (getTy, Typed ty val:cs)
5 -> label "CST_CODE_WIDE_INTEGER" $ do
ty <- getTy
n <- parseWideInteger r
return (getTy, Typed ty (ValInteger n):cs)
6 -> label "CST_CODE_FLOAT" $ do
let field = parseField r
ty <- getTy
ft <- (elimFloatType =<< elimPrimType ty)
`mplus` fail "expecting a float type"
let build k = do
w <- field 0 numeric
return (getTy, (Typed ty $! k w):cs)
case ft of
Float -> build (ValFloat . castFloat)
_ -> build (ValDouble . castDouble)
7 -> label "CST_CODE_AGGREGATE" $ do
ty <- getTy
elems <- parseField r 0 (fieldArray numeric)
`mplus` parseFields r 0 numeric
cxt <- getContext
let vals = [forwardRef cxt ix t | ix <- elems ]
case ty of
Struct _fs ->
return (getTy, Typed ty (ValStruct vals):cs)
PackedStruct _fs ->
return (getTy, Typed ty (ValPackedStruct vals):cs)
Array _n fty ->
return (getTy, Typed ty (ValArray fty (map typedValue vals)):cs)
Vector _n ety -> do
return (getTy, Typed ty (ValVector ety (map typedValue vals)):cs)
_ -> return (getTy, Typed ty ValUndef:cs)
8 -> label "CST_CODE_STRING" $ do
let field = parseField r
ty <- getTy
values <- field 0 string
return (getTy, Typed ty (ValString values):cs)
9 -> label "CST_CODE_CSTRING" $ do
ty <- getTy
values <- parseField r 0 cstring
`mplus` parseFields r 0 (fieldChar6 ||| char)
return (getTy, Typed ty (ValString (values ++ [chr 0])):cs)
10 -> label "CST_CODE_CE_BINOP" $ do
fail "not implemented"
11 -> label "CST_CODE_CE_CAST" $ do
let field = parseField r
ty <- getTy
cast' <- field 0 castOpCE
opval <- field 2 numeric
cxt <- getContext
return (getTy,Typed ty (cast' (forwardRef cxt opval t) ty):cs)
12 -> label "CST_CODE_CE_GEP" $ do
ty <- getTy
args <- parseCeGep t r
return (getTy,Typed ty (ValConstExpr (ConstGEP False args)):cs)
13 -> label "CST_CODE_CE_SELECT" $ do
let field = parseField r
ty <- getTy
ix1 <- field 0 numeric
ix2 <- field 1 numeric
ix3 <- field 2 numeric
cxt <- getContext
let ref ix = forwardRef cxt ix t
ce = ConstSelect (ref ix1) (ref ix2) (ref ix3)
return (getTy, Typed ty (ValConstExpr ce):cs)
14 -> label "CST_CODE_CE_EXTRACTELT" $ do
fail "not implemented"
15 -> label "CST_CODE_CE_INSERTELT" $ do
fail "not implemented"
16 -> label "CST_CODE_CE_SHUFFLEVEC" $ do
fail "not implemented"
17 -> label "CST_CODE_CE_CMP" $ do
fail "not implemented"
18 -> label "CST_CODE_INLINEASM" $ do
let field = parseField r
ty <- getTy
flags <- field 0 numeric
let sideEffect = testBit (flags :: Int) 0
alignStack = (flags `shiftR` 1) == 1
alen <- field 1 numeric
asm <- parseSlice r 2 alen char
clen <- field (2+alen) numeric
cst <- parseSlice r (3+alen) clen char
return (getTy, Typed ty (ValAsm sideEffect alignStack asm cst):cs)
19 -> label "CST_CODE_CE_SHUFFLEVEC_EX" $ do
fail "not implemented"
20 -> label "CST_CODE_CE_INBOUNDS_GEP" $ do
ty <- getTy
args <- parseCeGep t r
return (getTy,Typed ty (ValConstExpr (ConstGEP True args)):cs)
21 -> label "CST_CODE_BLOCKADDRESS" $ do
let field = parseField r
ty <- getTy
val <- getValue ty =<< field 1 numeric
bid <- field 2 numeric
sym <- elimValSymbol (typedValue val)
`mplus` fail "invalid function symbol in BLOCKADDRESS record"
let ce = ConstBlockAddr sym bid
return (getTy, Typed ty (ValConstExpr ce):cs)
22 -> label "CST_CODE_DATA" $ do
ty <- getTy
elemTy <- (elimPrimType =<< elimSequentialType ty)
`mplus` fail "invalid container type for CST_CODE_DATA"
let build mk = do
ns <- parseFields r 0 numeric
let elems = map mk ns
val | isArray ty = ValArray (PrimType elemTy) elems
| otherwise = ValVector (PrimType elemTy) elems
return (getTy, Typed ty val : cs)
case elemTy of
Integer 8 -> build ValInteger
Integer 16 -> build ValInteger
Integer 32 -> build ValInteger
Integer 64 -> build ValInteger
FloatType Float -> build ValFloat
FloatType Double -> build ValDouble
_ -> fail "unknown element type in CE_DATA"
code -> fail ("unknown constant record code: " ++ show code)
parseConstantEntry _ st (abbrevDef -> Just _) =
return st
parseConstantEntry _ _ e =
fail ("constant block: unexpected: " ++ show e)
parseCeGep :: ValueTable -> Record -> Parse [Typed PValue]
parseCeGep t r = loop 0
where
field = parseField r
loop n = do
ty <- getType =<< field n numeric
elt <- field (n+1) numeric
rest <- loop (n+2) `mplus` return []
cxt <- getContext
return (Typed ty (typedValue (forwardRef cxt elt t)) : rest)
parseWideInteger :: Record -> Parse Integer
parseWideInteger r = do
limbs <- parseFields r 0 signed
return (foldr (\l acc -> acc `shiftL` 64 + l) 0 limbs)
resolveNull :: Type -> Parse PValue
resolveNull ty = case typeNull ty of
HasNull nv -> return nv
ResolveNull i -> resolveNull =<< getType' =<< getTypeId i
castFloat :: Word32 -> Float
castFloat w = runST (cast w)
castDouble :: Word64 -> Double
castDouble w = runST (cast w)
cast :: (MArray (STUArray s) a (ST s), MArray (STUArray s) b (ST s))
=> a -> ST s b
cast x = do
arr <- newArray (0 :: Int, 0) x
res <- castSTUArray arr
readArray res 0