{-|
Module      : IRTS.JavaScript.PrimOp
Description : The JavaScript primitive operations.
Copyright   :
License     : BSD3
Maintainer  : The Idris Community.
-}

{-# LANGUAGE OverloadedStrings, PatternGuards, StandaloneDeriving #-}

module IRTS.JavaScript.PrimOp
  ( PrimF
  , PrimDec
  , JsPrimTy(..)
  , primDB
  , jsPrimCoerce
  ) where

import Data.Char
import Data.List
import qualified Data.Map.Strict as Map
import Data.Text (Text)
import qualified Data.Text as T
import Idris.Core.TT
import IRTS.JavaScript.AST
import IRTS.Lang

data JsPrimTy = PTBool | PTAny deriving (Eq, Ord)

type PrimF = [JsExpr] -> JsExpr
type PrimDec = (Bool, JsPrimTy, PrimF) -- the bool indicates if bigint library is used or not

deriving instance Ord PrimFn

primDB :: Map.Map PrimFn PrimDec
primDB =
  Map.fromList [
    item (LPlus ATFloat) False PTAny $ binop "+"
  , item (LPlus (ATInt ITChar)) False PTAny $ JsForeign "String.fromCharCode(%0.charCodeAt(0) + %1.charCodeAt(0))"
  , item (LPlus (ATInt ITNative)) False PTAny $ binop "+"
  , item (LPlus (ATInt (ITFixed IT8))) False PTAny $ JsForeign "%0 + %1 & 0xFF"
  , item (LPlus (ATInt (ITFixed IT16))) False PTAny $ JsForeign "%0 + %1 & 0xFFFF"
  , item (LPlus (ATInt (ITFixed IT32))) False PTAny $ JsForeign "%0+%1|0"
  , item (LPlus (ATInt ITBig)) True PTAny $ method "add"
  , item (LPlus (ATInt (ITFixed IT64))) True PTAny $
    \[l, r] -> JsForeign "%0.add(%1).and(new $JSRTS.jsbn.BigInteger(%2))" [l,r, JsStr $ show 0xFFFFFFFFFFFFFFFF]
  , item (LMinus ATFloat) False PTAny $ binop "-"
  , item (LMinus (ATInt ITChar)) False PTAny $ JsForeign "String.fromCharCode(%0.charCodeAt(0) - %1.charCodeAt(0))"
  , item (LMinus (ATInt ITNative)) False PTAny $ binop "-"
  , item (LMinus (ATInt (ITFixed IT8))) False PTAny $ JsForeign "%0 - %1 & 0xFF"
  , item (LMinus (ATInt (ITFixed IT16))) False PTAny $ JsForeign "%0 - %1 & 0xFFFF"
  , item (LMinus (ATInt (ITFixed IT32))) False PTAny $ JsForeign "%0-%1|0"
  , item (LMinus (ATInt ITBig)) True PTAny $ method "subtract"
  , item (LMinus (ATInt (ITFixed IT64))) True PTAny $
    \[l, r] -> JsForeign "%0.subtract(%1).and(new $JSRTS.jsbn.BigInteger(%2))" [l,r, JsStr $ show 0xFFFFFFFFFFFFFFFF]
  , item (LTimes ATFloat) False PTAny $ binop "*"
  , item (LTimes (ATInt ITChar)) False PTAny $ JsForeign "String.fromCharCode(%0.charCodeAt(0) * %1.charCodeAt(0))"
  , item (LTimes (ATInt ITNative)) False PTAny $ binop "*"
  , item (LTimes (ATInt (ITFixed IT8))) False PTAny $ JsForeign "%0 * %1 & 0xFF"
  , item (LTimes (ATInt (ITFixed IT16))) False PTAny $ JsForeign "%0 * %1 & 0xFFFF"
  , item (LTimes (ATInt (ITFixed IT32))) False PTAny $ JsForeign "%0*%1|0"
  , item (LTimes (ATInt ITBig)) True PTAny $ method "multiply"
  , item (LTimes (ATInt (ITFixed IT64))) True PTAny $
    \[l, r] -> JsForeign "%0.multiply(%1).and(new $JSRTS.jsbn.BigInteger(%2))" [l,r, JsStr $ show 0xFFFFFFFFFFFFFFFF]
  , item (LUDiv (ITFixed IT8)) False PTAny $ JsForeign "%0 / %1"
  , item (LUDiv (ITFixed IT16)) False PTAny $ JsForeign "%0 / %1"
  , item (LUDiv (ITFixed IT32)) False PTAny $ JsForeign "(%0>>>0)  / (%1>>>0) |0"
  , item (LUDiv (ITFixed IT64)) True PTAny $ JsForeign "%0.divide(%1)"
  , item (LSDiv ATFloat) False PTAny $ binop "/"
  , item (LSDiv (ATInt (ITFixed IT8))) False PTAny $ JsForeign "%0 / %1"
  , item (LSDiv (ATInt (ITFixed IT16))) False PTAny $ JsForeign "%0 / %1"
  , item (LSDiv (ATInt (ITFixed IT32))) False PTAny $ JsForeign "%0  / %1 |0"
  , item (LSDiv (ATInt (ITFixed IT64))) True PTAny $ JsForeign "%0.divide(%1)"
  , item (LSDiv (ATInt ITNative)) False PTAny $ JsForeign "%0/%1|0" -- we need "|0" in this
  , item (LSDiv (ATInt ITBig)) True PTAny $ method "divide"
  , item (LURem (ITFixed IT8)) False PTAny $ JsForeign "%0 % %1"
  , item (LURem (ITFixed IT16)) False PTAny $ JsForeign "%0 % %1"
  , item (LURem (ITFixed IT32)) False PTAny $ JsForeign "(%0>>>0) % (%1>>>0) |0"
  , item (LURem (ITFixed IT64)) True PTAny $ method "remainder"
  , item (LSRem ATFloat) False PTAny $ binop "%"
  , item (LSRem (ATInt ITNative)) False PTAny $ binop "%"
  , item (LSRem (ATInt (ITFixed IT8))) False PTAny $ JsForeign "%0 % %1"
  , item (LSRem (ATInt (ITFixed IT16))) False PTAny $ JsForeign "%0 % %1"
  , item (LSRem (ATInt (ITFixed IT32))) False PTAny $ JsForeign "%0 % %1 |0"
  , item (LSRem (ATInt (ITFixed IT64))) True PTAny $ method "remainder"
  , item (LSRem (ATInt ITBig)) True PTAny $ method "remainder"
  , item (LAnd ITNative) False PTAny $ JsForeign "%0 & %1"
  , item (LAnd (ITFixed IT8)) False PTAny $ JsForeign "%0 & %1"
  , item (LAnd (ITFixed IT16)) False PTAny $ JsForeign "%0 & %1"
  , item (LAnd (ITFixed IT32)) False PTAny $ JsForeign "%0  & %1"
  , item (LAnd (ITFixed IT64)) True PTAny $ method "and"
  , item (LAnd ITBig) True PTAny $ method "and"
  , item (LOr ITNative) False PTAny $ JsForeign "%0 | %1"
  , item (LOr (ITFixed IT8)) False PTAny $ JsForeign "%0 | %1"
  , item (LOr (ITFixed IT16)) False PTAny $ JsForeign "%0 | %1"
  , item (LOr (ITFixed IT32)) False PTAny $ JsForeign "%0 | %1"
  , item (LOr (ITFixed IT64)) True PTAny $ method "or"
  , item (LOr ITBig) True PTAny $ method "or"
  , item (LXOr ITNative) False PTAny $ JsForeign "%0 ^ %1"
  , item (LXOr (ITFixed IT8)) False PTAny $ JsForeign "%0 ^ %1"
  , item (LXOr (ITFixed IT16)) False PTAny $ JsForeign "%0 ^ %1"
  , item (LXOr (ITFixed IT32)) False PTAny $ JsForeign "%0 ^ %1"
  , item (LXOr (ITFixed IT64)) True PTAny $ method "xor"
  , item (LXOr ITBig) True PTAny $ method "xor"
  , item (LSHL ITNative) False PTAny $ JsForeign "%0 << %1 |0"
  , item (LSHL (ITFixed IT8)) False PTAny $ JsForeign "%0 << %1 & 0xFF"
  , item (LSHL (ITFixed IT16)) False PTAny $ JsForeign "%0 << %1 & 0xFFFF"
  , item (LSHL (ITFixed IT32)) False PTAny $ JsForeign "%0 << %1 | 0"
  , item (LSHL (ITFixed IT64)) True PTAny $
    \[l, r] -> JsForeign "%0.shiftLeft(%1).and(new $JSRTS.jsbn.BigInteger(%2))" [l,r, JsStr $ show 0xFFFFFFFFFFFFFFFF]
  , item (LSHL ITBig) True PTAny $ method "shiftLeft"
  , item (LLSHR ITNative) False PTAny $ JsForeign "%0 >> %1 |0"
  , item (LLSHR (ITFixed IT8)) False PTAny $ JsForeign "%0 >> %1"
  , item (LLSHR (ITFixed IT16)) False PTAny $ JsForeign "%0 >> %1"
  , item (LLSHR (ITFixed IT32)) False PTAny $ JsForeign "%0 >> %1|0"
  , item (LLSHR (ITFixed IT64)) True PTAny $ JsForeign "%0.shiftRight(%1)"
  , item (LASHR ITNative) False PTAny $ JsForeign "%0 >> %1 |0"
  , item (LASHR (ITFixed IT8)) False PTAny $ JsForeign "%0 >> %1"
  , item (LASHR (ITFixed IT16)) False PTAny $ JsForeign "%0 >> %1"
  , item (LASHR (ITFixed IT32)) False PTAny $ JsForeign "%0 >> %1|0"
  , item (LASHR (ITFixed IT64)) True PTAny $ JsForeign "%0.shiftRight(%1)"
  , item (LEq ATFloat) False PTBool $ binop "==="
  , item (LEq (ATInt ITNative)) False PTBool $ binop "==="
  , item (LEq (ATInt ITChar)) False PTBool $ binop "==="
  , item (LEq (ATInt ITBig)) True PTBool $ method "equals"
  , item (LEq (ATInt (ITFixed IT8))) False PTBool $ binop "==="
  , item (LEq (ATInt (ITFixed IT16))) False PTBool $ binop "==="
  , item (LEq (ATInt (ITFixed IT32))) False PTBool $ binop "==="
  , item (LEq (ATInt (ITFixed IT64))) True PTBool $ method "equals"
  , item (LLt (ITFixed IT8)) False PTBool $ JsForeign "%0 < %1"
  , item (LLt (ITFixed IT16)) False PTBool $ JsForeign "%0 < %1"
  , item (LLt (ITFixed IT32)) False PTBool $ JsForeign "(%0>>>0) < (%1>>>0)"
  , item (LLt (ITFixed IT64)) True PTBool $ JsForeign "%0.compareTo(%1) < 0"
  , item (LLe (ITFixed IT8)) False PTBool $ JsForeign "%0 <= %1"
  , item (LLe (ITFixed IT16)) False PTBool $ JsForeign "%0 <= %1"
  , item (LLe (ITFixed IT32)) False PTBool $ JsForeign "(%0>>>0) <= (%1>>>0)"
  , item (LLe (ITFixed IT64)) True PTBool $ JsForeign "%0.compareTo(%1) <= 0"
  , item (LGt (ITFixed IT8)) False PTBool $ JsForeign "%0 > %1"
  , item (LGt (ITFixed IT16)) False PTBool $ JsForeign "%0 > %1"
  , item (LGt (ITFixed IT32)) False PTBool $ JsForeign "(%0>>>0) > (%1>>>0)"
  , item (LGt (ITFixed IT64)) True PTBool $ JsForeign "%0.compareTo(%1) > 0"
  , item (LGe (ITFixed IT8)) False PTBool $ JsForeign "%0 >= %1"
  , item (LGe (ITFixed IT16)) False PTBool $ JsForeign "%0 >= %1"
  , item (LGe (ITFixed IT32)) False PTBool $ JsForeign "(%0>>>0) >= (%1>>>0)"
  , item (LGe (ITFixed IT64)) True PTBool $ JsForeign "%0.compareTo(%1) >= 0"
  , item (LSLt ATFloat) False PTBool $ binop "<"
  , item (LSLt (ATInt ITChar)) False PTBool $ binop "<"
  , item (LSLt (ATInt ITNative)) False PTBool $ binop "<"
  , item (LSLt (ATInt ITBig)) True PTBool $ JsForeign "%0.compareTo(%1) < 0"
  , item (LSLt (ATInt (ITFixed IT8))) False PTBool $ binop "<"
  , item (LSLt (ATInt (ITFixed IT16))) False PTBool $ binop "<"
  , item (LSLt (ATInt (ITFixed IT32))) False PTBool $ binop "<"
  , item (LSLt (ATInt (ITFixed IT64))) True PTBool $ JsForeign "%0.compareTo(%1) < 0"
  , item (LSLe ATFloat) False PTBool $ binop "<="
  , item (LSLe (ATInt ITNative)) False PTBool $ binop "<="
  , item (LSLe (ATInt ITBig)) True PTBool $ JsForeign "%0.compareTo(%1) <= 0"
  , item (LSLe (ATInt (ITFixed IT8))) False PTBool $ binop "<="
  , item (LSLe (ATInt (ITFixed IT16))) False PTBool $ binop "<="
  , item (LSLe (ATInt (ITFixed IT32))) False PTBool $ binop "<="
  , item (LSLe (ATInt (ITFixed IT64))) True PTBool $ JsForeign "%0.compareTo(%1) <= 0"
  , item (LSGt ATFloat) False PTBool $ binop ">"
  , item (LSGt (ATInt ITNative)) False PTBool $ binop ">"
  , item (LSGt (ATInt ITBig)) True PTBool $ JsForeign "%0.compareTo(%1) > 0"
  , item (LSGt (ATInt (ITFixed IT8))) False PTBool $ binop ">"
  , item (LSGt (ATInt (ITFixed IT16))) False PTBool $ binop ">"
  , item (LSGt (ATInt (ITFixed IT32))) False PTBool $ binop ">"
  , item (LSGt (ATInt (ITFixed IT64))) True PTBool $ JsForeign "%0.compareTo(%1) > 0"
  , item (LSGe ATFloat) False PTBool $ binop ">="
  , item (LSGe (ATInt ITNative)) False PTBool $ binop ">="
  , item (LSGe (ATInt ITBig)) True PTBool $ JsForeign "%0.compareTo(%1) >= 0"
  , item (LSGe (ATInt (ITFixed IT8))) False PTBool $ binop ">="
  , item (LSGe (ATInt (ITFixed IT16))) False PTBool $ binop ">="
  , item (LSGe (ATInt (ITFixed IT32))) False PTBool $ binop ">="
  , item (LSGe (ATInt (ITFixed IT64))) True PTBool $ JsForeign "%0.compareTo(%1) >= 0"
  , item (LSExt ITNative ITBig) True PTAny $ JsForeign "new $JSRTS.jsbn.BigInteger(''+%0)"
  , item (LZExt (ITFixed IT8) ITNative) False PTAny $ head
  , item (LZExt (ITFixed IT16) ITNative) False PTAny $ head
  , item (LZExt (ITFixed IT32) ITNative) False PTAny $ head
  , item (LZExt ITNative ITBig) True PTAny $ JsForeign "new $JSRTS.jsbn.BigInteger(''+%0)"
  , item (LZExt (ITFixed IT8) ITBig) True PTAny $ JsForeign "new $JSRTS.jsbn.BigInteger(''+%0)"
  , item (LZExt (ITFixed IT16) ITBig) True PTAny $ JsForeign "new $JSRTS.jsbn.BigInteger(''+%0)"
  , item (LZExt (ITFixed IT32) ITBig) True PTAny $ JsForeign "new $JSRTS.jsbn.BigInteger(''+%0)"
  , item (LZExt (ITFixed IT64) ITBig) True PTAny $ head
  , item (LTrunc ITBig ITNative) True PTAny $ JsForeign "%0.intValue()|0"
  , item (LTrunc ITBig (ITFixed IT8)) True PTAny $ JsForeign "%0.intValue() & 0xFF"
  , item (LTrunc ITBig (ITFixed IT16)) True PTAny $ JsForeign "%0.intValue() & 0xFFFF"
  , item (LTrunc ITBig (ITFixed IT32)) True PTAny $ JsForeign "%0.intValue() & 0xFFFFFFFF"
  , item (LTrunc ITBig (ITFixed IT64)) True PTAny $
    \[x] -> JsForeign "%0.and(new $JSRTS.jsbn.BigInteger(%1))" [x, JsStr $ show 0xFFFFFFFFFFFFFFFF]
  , item (LTrunc (ITFixed IT16) (ITFixed IT8)) False PTAny $ JsForeign "%0 & 0xFF"
  , item (LTrunc (ITFixed IT32) (ITFixed IT8)) False PTAny $ JsForeign "%0 & 0xFF"
  , item (LTrunc (ITFixed IT64) (ITFixed IT8)) True PTAny $ JsForeign "%0.intValue() & 0xFF"
  , item (LTrunc (ITFixed IT32) (ITFixed IT16)) False PTAny $ JsForeign "%0 & 0xFFFF"
  , item (LTrunc (ITFixed IT64) (ITFixed IT16)) True PTAny $ JsForeign "%0.intValue() & 0xFFFF"
  , item (LTrunc (ITFixed IT64) (ITFixed IT32)) True PTAny $ JsForeign "%0.intValue() & 0xFFFFFFFF"
  , item LStrConcat False PTAny $ binop "+"
  , item LStrLt False PTBool $ binop "<"
  , item LStrEq False PTBool $ binop "=="
  , item LStrLen False PTAny $ JsForeign "%0.length"
  , item (LIntFloat ITNative) False PTAny $ head
  , item (LIntFloat ITBig) True PTAny $ JsForeign "%0.intValue()"
  , item (LFloatInt ITNative) False PTAny $ JsForeign "%0|0"
  , item (LFloatInt ITBig) True PTAny $ JsForeign "new $JSRTS.jsbn.BigInteger(Math.trunc(%0)+ '')"
  , item (LIntStr ITNative) False PTAny $ JsForeign "''+%0"
  , item (LIntStr ITBig) True PTAny $ JsForeign "%0.toString()"
  , item (LStrInt ITNative) False PTAny $ JsForeign "parseInt(%0)|0"
  , item (LStrInt ITBig) True PTAny $ JsForeign "new $JSRTS.jsbn.BigInteger(%0)"
  , item (LFloatStr) False PTAny $ JsForeign "''+%0"
  , item (LStrFloat) False PTAny $ jsAppN "parseFloat"
  , item (LChInt ITNative) False PTAny $ JsForeign "%0.charCodeAt(0)|0"
  , item (LIntCh ITNative) False PTAny $ jsAppN "String.fromCharCode"
  , item LFExp False PTAny $ jsAppN "Math.exp"
  , item LFLog False PTAny $ jsAppN "Math.log"
  , item LFSin False PTAny $ jsAppN "Math.sin"
  , item LFCos False PTAny $ jsAppN "Math.cos"
  , item LFTan False PTAny $ jsAppN "Math.tan"
  , item LFASin False PTAny $ jsAppN "Math.asin"
  , item LFACos False PTAny $ jsAppN "Math.acos"
  , item LFATan False PTAny $ jsAppN "Math.atan"
  , item LFSqrt False PTAny $ jsAppN "Math.sqrt"
  , item LFFloor False PTAny $ jsAppN "Math.floor"
  , item LFCeil False PTAny $ jsAppN "Math.ceil"
  , item LFNegate False PTAny $ JsForeign "-%0"
  , item LStrHead False PTAny $ \[x] -> JsArrayProj (JsInt 0) x
  , item LStrTail False PTAny $ \[x] -> JsMethod x "slice" [JsInt 1]
  , item LStrCons False PTAny $ JsForeign "%0+%1"
  , item LStrIndex False PTAny $ \[x, y] -> JsArrayProj y x
  , item LStrRev False PTAny $ JsForeign "%0.split('').reverse().join('')"
  , item LStrSubstr False PTAny $ JsForeign "$JSRTS.prim_strSubstr(%0, %1, %2)"
  , item LSystemInfo False PTAny $ JsApp (JsProp (JsVar "$JSRTS") "prim_systemInfo")
  , item LCrash False PTAny $ \[l] -> JsErrorExp l
  , item LReadStr False PTAny $ \[_] -> JsApp (JsProp (JsVar "$JSRTS") "prim_readStr") []
  , item LWriteStr False PTAny $ \[_, str] -> JsApp (JsProp (JsVar "$JSRTS") "prim_writeStr") [str]
  , item LNoOp False PTAny $ head
  ]
  where
    item :: PrimFn -> Bool -> JsPrimTy -> PrimF -> (PrimFn, PrimDec)
    item fn ubi pty c = (fn, (ubi, pty, c))
    binop op [l, r] = JsBinOp op l r
    method op (l:r) = JsMethod l op r

jsB2I :: JsExpr -> JsExpr
jsB2I x = JsForeign "%0 ? 1|0 : 0|0" [x]

jsPrimCoerce :: JsPrimTy -> JsPrimTy -> JsExpr -> JsExpr
jsPrimCoerce PTBool PTAny x = jsB2I x
jsPrimCoerce _ _ x = x