call.go 2.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129
  1. package eval
  2. import (
  3. "encoding/json"
  4. "fmt"
  5. "fw/cp"
  6. "fw/cp/constant"
  7. "fw/cp/node"
  8. "fw/cp/object"
  9. "fw/rt2"
  10. "fw/rt2/frame"
  11. rtm "fw/rt2/module"
  12. "fw/rt2/scope"
  13. "log"
  14. "math"
  15. "reflect"
  16. "ypk/assert"
  17. "ypk/halt"
  18. "ypk/mathe"
  19. )
  20. var sys map[string]func(IN, node.Node) OUT
  21. type Msg struct {
  22. Type string
  23. Command string
  24. Data string
  25. }
  26. func callHandler(f frame.Frame, obj object.Object, data interface{}) {
  27. //n := rt2.Utils.NodeOf(f)
  28. //fmt.Println("call handler", obj)
  29. if obj == nil {
  30. return
  31. }
  32. m := rtm.DomainModule(f.Domain())
  33. cn := node.New(constant.CALL, cp.Some())
  34. ol := m.NodeByObject(obj)
  35. assert.For(len(ol) <= 1, 40)
  36. cn.SetLeft(ol[0])
  37. cc := node.New(constant.CONSTANT, cp.Some()).(node.ConstantNode)
  38. cc.SetData(data)
  39. cc.SetType(object.SHORTSTRING)
  40. cn.SetRight(cc)
  41. rt2.Push(rt2.New(cn), f)
  42. }
  43. func go_process(in IN, par node.Node) OUT {
  44. assert.For(par != nil, 20)
  45. sm := rt2.CallScope(in.Frame)
  46. do := func(val string) {
  47. if val != "" {
  48. msg := &Msg{}
  49. if err := json.Unmarshal([]byte(val), msg); err == nil {
  50. switch msg.Type {
  51. case "log":
  52. fmt.Print(msg.Data)
  53. callHandler(in.Frame, scope.FindObjByName(sm, "go_handler"), `{"type":"log"}`)
  54. case "core":
  55. switch msg.Command {
  56. case "load":
  57. LoadMod(in.Frame, msg.Data)
  58. default:
  59. halt.As(100, msg.Command)
  60. }
  61. default:
  62. panic(40)
  63. }
  64. } else {
  65. log.Println(val, "not a json")
  66. }
  67. }
  68. }
  69. const left = "sys:left"
  70. return GetExpression(in, left, par, func(IN) OUT {
  71. val := rt2.ValueOf(in.Frame)[KeyOf(in, left)]
  72. assert.For(val != nil, 20)
  73. do(scope.GoTypeFrom(val).(string))
  74. return Later(Tail(STOP))
  75. })
  76. }
  77. func go_math(in IN, par node.Node) OUT {
  78. const (
  79. LN = 1.0
  80. MANT = 2.0
  81. EXP = 3.0
  82. )
  83. assert.For(par != nil, 20)
  84. res := math.NaN()
  85. switch p := par.(type) {
  86. case node.VariableNode:
  87. sm := rt2.ScopeFor(in.Frame, p.Object().Adr())
  88. sm.Select(p.Object().Adr(), func(val scope.Value) {
  89. rv, ok := scope.GoTypeFrom(val).([]float64)
  90. assert.For(ok && (len(rv) > 1), 100, rv)
  91. switch rv[0] {
  92. case LN:
  93. res = math.Log(rv[1])
  94. case MANT:
  95. res, _ = mathe.Me(rv[1])
  96. case EXP:
  97. _, res = mathe.Me(rv[1])
  98. default:
  99. halt.As(100, rv[0])
  100. }
  101. })
  102. default:
  103. halt.As(100, reflect.TypeOf(p))
  104. }
  105. id := cp.ID(cp.Some())
  106. rt2.RegOf(in.Parent)[in.Key] = id
  107. rt2.ValueOf(in.Parent)[id] = scope.TypeFromGo(res)
  108. return End()
  109. }
  110. func init() {
  111. sys = make(map[string]func(IN, node.Node) OUT)
  112. sys["go_process"] = go_process
  113. sys["go_math"] = go_math
  114. }
  115. var LoadMod func(frame.Frame, string)
  116. func syscall(in IN) OUT {
  117. n := in.IR
  118. name := n.Left().Object().Name()
  119. return sys[name](in, n.Right())
  120. }