import Generics.gCons
+import iTasks.UI.Editor.Common
+import iTasks.UI.Editor
+
import GenEq, StdMisc, StdArray, GenBimap
import GenPrint
import StdEnum
import Data.Functor
import StdList
from Data.Func import $
-from Text import class Text(concat,toUpperCase), instance Text String
+from Text import class Text(subString,lpad,concat,toUpperCase), instance Text String
import qualified Text
import Text.JSON
import qualified Data.List as DL
import Text.Encodings.Base64
+import Tasks.Examples
+
encode :: MTaskMSGSend -> String
encode (MTTask to data) = "t" +++ tob +++ to16bit (size data) +++ data +++ "\n"
where
encode (MTSds i v) = "s" +++ to16bit i +++ v +++ "\n"
encode (MTUpd i v) = "u" +++ to16bit i +++ v +++ "\n"
+import StdDebug
decode :: String -> MTaskMSGRecv
decode x
+| not (trace_tn ("decoding: " +++ toString (toJSON x))) = undef
| size x == 0 = MTEmpty
= case x.[0] of
- 't' = MTTaskAck (from16bit (x % (1,3)))
- 'd' = MTTaskDelAck (from16bit (x % (1,3)))
+ 't' = MTTaskAck $ fromByteCode x
+ 'd' = MTTaskDelAck $ fromByteCode x
'm' = MTMessage x
- 's' = MTSDSAck (from16bit (x % (1,3)))
- 'a' = MTSDSDelAck (from16bit (x % (1,3)))
- 'p' = MTPub (from16bit (x % (1,3))) (x % (3,5))
+ 's' = MTSDSAck $ fromByteCode x
+ 'a' = MTSDSDelAck $ fromByteCode x
+ 'p' = MTPub (fromByteCode x) $ fromByteCode $ x % (3,size x)
+ 'c' = MTDevSpec $ fromByteCode (x % (1, size x))
'\0' = MTEmpty
'\n' = MTEmpty
_ = MTMessage x//abort ("Didn't understand message: " +++ join " " [toString (toInt c)\\c<-: x] +++ "\n")
toString (MTSDSDelAck i) = "SDS deleted with id: " +++ toString i
toString (MTPub i v) = "Publish id: " +++ toString i
+++ " value " +++ safePrint v
+ toString (MTDevSpec mt) = "Specification: " +++ printToString mt
toString (MTMessage m) = m
toString MTEmpty = "Empty message"
toByteVal :: BC -> String
toByteVal b = {toChar $ consIndex{|*|} b} +++
case b of
- (BCPush i) = i
+ (BCPush (BCValue i)) = toByteCode i
(BCLab i) = {toChar i}
(BCSdsStore i) = to16bit i
(BCSdsFetch i) = to16bit i
(BCSdsPublish i) = to16bit i
- (BCAnalogRead i) = {toChar i}
- (BCAnalogWrite i) = {toChar i}
- (BCDigitalRead i) = {toChar i}
- (BCDigitalWrite i) = {toChar i}
+ (BCAnalogRead i) = {toChar $ consIndex{|*|} i}
+ (BCAnalogWrite i) = {toChar $ consIndex{|*|} i}
+ (BCDigitalRead i) = {toChar $ consIndex{|*|} i}
+ (BCDigitalWrite i) = {toChar $ consIndex{|*|} i}
(BCJmp i) = {toChar i}
(BCJmpT i) = {toChar i}
(BCJmpF i) = {toChar i}
_ = ""
-//(>>) infixl 2 :: (ByteCode a p) (ByteCode b q) -> ByteCode b q
-//(>>) m n = BC \s->(let (_, s1) = runBC m s in
-// let (a, s2) = runBC n s1
-// in (a, {s2 & bytecode=(s2.BCState.bytecode ++ s1.BCState.bytecode)}))
-//
-//(<+->) infixr 1
-//(<+->) m n :== m >> tell n
-//
-//runBC (BC m) = m
-//
-//tell :: [BC] -> ByteCode a p | mTaskType a
-//tell b = BC \s->(zero, {s & bytecode=b++s.bytecode})
-//
-//fmp :: (a -> BC) (ByteCode a p) -> ByteCode a q
-//fmp f b = BC \s->let (a, s`) = runBC b s in (a, {s` & code=[f a:s`.code]})
-
-instance toByteCode Bool where toByteCode b = if b "\x01" "\x00"
-instance toByteCode Int where toByteCode n = {toChar $ n/256,toChar $ n rem 256}
-instance toByteCode Long where toByteCode (L n) = toByteCode n
-instance toByteCode Char where toByteCode s = toString s
+parseBCValue :: Char String -> BCValue
+parseBCValue c s = case c of
+ 'b' = BCValue $ castfbc True s
+ 'i' = BCValue $ castfbc 0 s
+ 'l' = BCValue $ castfbc (L 0) s
+ 'c' = BCValue $ castfbc ('0') s
+ 'B' = BCValue $ castfbc (NoButton) s
+ 'L' = BCValue $ castfbc (LED1) s
+
+castfbc :: a -> (String -> a) | mTaskType a
+castfbc _ = fromByteCode
+
+instance toByteCode Bool where toByteCode b = {#'b',if b '\x01' '\0'}
+instance toByteCode Int where toByteCode n = {'i',toChar $ n/256,toChar $ n rem 256}
+instance toByteCode Long where toByteCode (L n) = {'l',toChar $ n/256,toChar $ n rem 256}
+instance toByteCode Char where toByteCode c = {'c',c}
instance toByteCode String where toByteCode s = abort $ "Undef on toBytecode String" +++ s
-instance toByteCode Button where toByteCode s = {toChar $ consIndex{|*|} s}
-instance toByteCode UserLED where toByteCode s = {toChar $ consIndex{|*|} s}
+instance toByteCode Button where toByteCode s = {'B',toChar $ consIndex{|*|} s}
+instance toByteCode UserLED where toByteCode s = {'L',toChar $ consIndex{|*|} s}
+instance toByteCode BCValue where toByteCode (BCValue v) = toByteCode v
+
+instance fromByteCode Bool where fromByteCode s = s.[1] == '\x01'
+instance fromByteCode Int where fromByteCode s = (toInt s.[1])*256 + toInt s.[2]
+instance fromByteCode Long where fromByteCode s = L $ fromByteCode s
+instance fromByteCode Char where fromByteCode s = s.[1]
+instance fromByteCode String where fromByteCode s = abort $ "Undef on fromBytecode String" +++ s
+instance fromByteCode Button where fromByteCode s = conses{|*|} !! toInt s.[1]
+instance fromByteCode UserLED where fromByteCode s = conses{|*|} !! toInt s.[1]
+instance fromByteCode BCValue where fromByteCode s = parseBCValue s.[0] s
+
instance toByteCode MTaskInterval where
toByteCode OneShot = toByteCode 0
//Intervals have the first bit 0 and the rest is a 15 bit unsigned int
toByteCode (OnInterval i) = {toChar $ i/256 bitand 127, toChar $ i rem 256}
//Intervals have the first bit 1 and the rest is a 15 bit unsigned int
toByteCode (OnInterrupt i) = {toChar $ i/256 bitor 127, toChar $ i rem 256}
-
-instance fromByteCode Bool where fromByteCode s = s == "\x01"
-instance fromByteCode Int where fromByteCode s = (toInt s.[0])*256 + toInt s.[1]
-instance fromByteCode Long where fromByteCode s = L $ fromByteCode s
-instance fromByteCode Char where fromByteCode s = toChar s.[0]
-instance fromByteCode String where fromByteCode s = abort $ "Undef on fromBytecode String" +++ s
-instance fromByteCode Button where fromByteCode s = conses{|*|} !! toInt s.[0]
-instance fromByteCode UserLED where fromByteCode s = conses{|*|} !! toInt s.[0]
instance fromByteCode MTaskInterval
where
fromByteCode s
0 = OneShot
i = OnInterval i
= OnInterrupt $ fromByteCode s bitand 127
+instance fromByteCode MTaskDeviceSpec where
+ fromByteCode s = let c = toInt s.[0] in
+ {MTaskDeviceSpec
+ |haveLed=c bitand 1 > 0
+ ,haveAio=c bitand 2 > 0
+ ,haveDio=c bitand 4 > 0
+ ,maxTask=from16bit $ s % (1,3)
+ ,maxSDS=from16bit $ s % (3,5)
+ }
+
+derive gPrint Long, UserLED, Button, AnalogPin, DigitalPin, PinMode, Pin, BC, MTaskDeviceSpec
+derive class gCons BC
-instance toChar Pin where
- toChar (Digital p) = toChar $ consIndex{|*|} p
- toChar (Analog p) = toChar $ consIndex{|*|} p
+consIndex{|BCValue|} _ = 0
+consName{|BCValue|} _ = "BCValue"
+conses{|BCValue|} = [BCValue 0]
+consNum{|BCValue|} _ = 1
+gPrint{|BCValue|} v ps = gPrint{|*|} (readable $ BCPush v) ps
-derive gPrint BC
-derive class gCons BC
+gEditor{|BCValue|} = {Editor|genUI=genUI`,onEdit=onEdit`,onRefresh=onRefresh`}
+ where
+ genUI` dp (BCValue a) vst = (castEditor a).Editor.genUI dp a vst
+ onEdit` dp jsn (BCValue a) em vst = appSnd3 BCValue $ (castEditor a).Editor.onEdit dp jsn a em vst
+ onRefresh` dp (BCValue a) (BCValue a`) em vst = appSnd3 BCValue $ (castEditor a).Editor.onRefresh dp a (fromByteCode $ toByteCode a`) em vst
+
+ castEditor :: a -> (Editor a) | mTaskType a
+ castEditor _ = gEditor{|*|}
+
+gText{|BCValue|} fm Nothing = []
+gText{|BCValue|} fm (Just (BCValue e)) = gText{|*|} fm (Just e)
+JSONEncode{|BCValue|} b (BCValue e) = JSONEncode{|*|} b (toByteCode e)
+JSONDecode{|BCValue|} b n = appFst (fmap fromByteCode) $ JSS b n
+ where
+ JSS :: (Bool [JSONNode] -> (Maybe String, [JSONNode]))
+ JSS = JSONDecode{|*|}
+gDefault{|BCValue|} = BCValue 0
+gEq{|BCValue|} (BCValue e) (BCValue f) = toByteCode e == toByteCode f
+
+derive class gCons Long, UserLED, Button, AnalogPin, DigitalPin, PinMode, Pin
+derive class iTask UserLED, Long, Pin, Button, AnalogPin, DigitalPin, PinMode, MTaskDeviceSpec
op2 :: (ByteCode a p1) (ByteCode a p2) BC -> ByteCode b Expr
op2 (BC x) (BC y) bc = BC $ x >>| y >>| tell [bc]
tell` x = BC $ tell x
-instance zero Bool where zero = False
-
instance arith ByteCode where
- lit x = tell` [BCPush $ toByteCode x]
+ lit x = tell` [BCPush $ BCValue x]
(+.) x y = op2 x y BCAdd
(-.) x y = op2 x y BCSub
(*.) x y = op2 x y BCMul
>>= \(v In bdy)->modify (addSDS sds v) >>| unBC (unMain bdy)}
// >>= \(v In bdy)->modify (addSDS sds (toByteCode v)) >>| unBC (unMain bdy)}
where
- addSDS i v s = {s & sdss=[{sdsi=i,sdspub=False,sdsval=(dynamic v),sdsbc=toByteCode v}:s.sdss]}
+ addSDS i v s = {s & sdss=[{sdsi=i,sdspub=False,sdsval=(BCValue v)}:s.sdss]}
con f = undef
pub (BC x) = BC $ censor (\[BCSdsFetch s]->[BCSdsPublish s])
ledOn (BC l) = BC $ l >>| tell [BCLedOn]
ledOff (BC l) = BC $ l >>| tell [BCLedOff]
-func :: (a -> BC) [BC] -> [BC] | mTaskType a
-func f b = abort ('Text'.join "\n" (map printToString b))
-
instance zero BCState where
zero = {freshl=[1..], freshs=[1..], sdss=[]}
implGotos map (BCJmpF t) = BCJmpF $ fromJust ('DM'.get t map)
implGotos _ i = i
+bclength :: BC -> Int
+bclength (BCPush s) = 1 + size (toByteCode s)
+bclength (BCSdsStore _) = 3
+bclength (BCSdsFetch _) = 3
+bclength (BCSdsPublish _) = 3
+bclength x = 1 + consNum{|*|} x
+
computeGotos :: [BC] Int -> ([BC], 'DM'.Map Int Int)
computeGotos [] _ = ([], 'DM'.newMap)
computeGotos [BCLab l:xs] i = appSnd ('DM'.put l i) (computeGotos xs i)
-computeGotos [x:xs] i = appFst (\bc->[x:bc])
- (computeGotos xs $ i + 1 + consNum{|*|} x)
+computeGotos [x:xs] i = appFst (\bc->[x:bc]) (computeGotos xs $ i + bclength x)
readable :: BC -> String
-readable (BCPush d) = "BCPush " +++ concat [safe c\\c<-:d]
+readable (BCPush d) = "BCPush " +++ concat [safe c\\c<-:toByteCode d]
where
safe c = if (isControl c) ("\\d" +++ toString (toInt c)) (toString c)
readable b = printToString b
toReadableByteCode x s
# (s, bc) = runBC x s
# (bc, gtmap) = computeGotos bc 0
-= ('Text'.join "\n" $ map readable (map (implGotos gtmap) bc), s)
+= ('Text'.join "\n" $ lineNumbers numbers (map (implGotos gtmap) bc), s)
+ where
+ numbers = map (\n->lpad (toString n) 3 ' ' +++ ". ") [0..]
+ lineNumbers ls [] = []
+ lineNumbers [l:ls] [b:bc] = [l +++ readable b:ex ++ lineNumbers newls bc]
+ where
+ (ex, newls) = splitAt (bclength b - 1) ls
toMessages :: MTaskInterval (String, BCState) -> ([MTaskMSGSend], BCState)
toMessages interval (bytes, st=:{sdss}) = (
- [MTSds s.sdsi s.sdsbc\\s<-sdss] ++
+ [MTSds sdsi $ toByteCode e\\{sdsi,sdsval=(BCValue e)}<-sdss] ++
[MTTask interval bytes], st)
toSDSUpdate :: Int Int -> [MTaskMSGSend]
toSDSUpdate i v = [MTUpd i (to16bit v)]
//Start = toMessages (OnInterval 500) $ toRealByteCode (unMain bc) zero
-Start = let (bcs, st) = toReadableByteCode (unMain bc) zero
- in (bcs, st.sdss)
+Start = fst $ toReadableByteCode (unMain $ countAndLed) zero
+//Start = fst $ toReadableByteCode (unMain $ blink LED1) zero
+//Start = let (bcs, st) = toReadableByteCode (unMain bc) zero
+// in (bcs, st.sdss)
where
// bc = {main = ledOn (lit LED1)}
bc = sds \x=5 In
from16bit :: String -> Int
from16bit s = toInt s.[0] * 256 + toInt s.[1]
+
+//derive class gCons UserLED, Long, Pin, Button, UserLED, AnalogPin, DigitalPin, PinMode