123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178 |
- package rules
- import (
- "encoding/json"
- "fmt"
- "fw/cp"
- "fw/cp/constant"
- "fw/cp/node"
- "fw/cp/object"
- "fw/rt2"
- "fw/rt2/context"
- "fw/rt2/frame"
- rt_mod "fw/rt2/module"
- "fw/rt2/scope"
- "fw/utils"
- "reflect"
- "ypk/assert"
- "ypk/halt"
- )
- /**
- Для CallNode
- .Left() указывает на процедуру или на переменную процедурного типа
- .Left().Object() указывает на список внутренних объектов, в т.ч. переменных
- .Object() указывает первый элемент из списка входных параметров/переменных,
- то же что и.Left().Object().Link(), далее .Link() указывает на следующие за ним входные параметры
- .Right() указывает на узлы, которые передаются в параметры
- */
- var sys map[string]func(f frame.Frame, par node.Node) (frame.Sequence, frame.WAIT)
- type Msg struct {
- Type string
- Data string
- }
- func callHandler(f frame.Frame, obj object.Object, data interface{}) {
- //n := rt2.Utils.NodeOf(f)
- //fmt.Println("call handler", obj)
- if obj == nil {
- return
- }
- m := rt_mod.DomainModule(f.Domain())
- cn := node.New(constant.CALL, int(cp.SomeAdr()))
- ol := m.NodeByObject(obj)
- assert.For(len(ol) <= 1, 40)
- cn.SetLeft(ol[0])
- cc := node.New(constant.CONSTANT, int(cp.SomeAdr())).(node.ConstantNode)
- cc.SetData(data)
- cc.SetType(object.SHORTSTRING)
- cn.SetRight(cc)
- rt2.Push(rt2.New(cn), f)
- }
- func process(f frame.Frame, par node.Node) (frame.Sequence, frame.WAIT) {
- assert.For(par != nil, 20)
- sm := f.Domain().Discover(context.SCOPE).(scope.Manager)
- do := func(val string) {
- if val != "" {
- msg := &Msg{}
- if err := json.Unmarshal([]byte(val), msg); err == nil {
- switch msg.Type {
- case "log":
- fmt.Println(msg.Data)
- callHandler(f, scope.FindObjByName(sm, "go_handler"), `{"type":"log"}`)
- default:
- panic(40)
- }
- } else {
- fmt.Println(val, "not a json")
- }
- }
- }
- var val string
- switch p := par.(type) {
- case node.ConstantNode:
- val = par.(node.ConstantNode).Data().(string)
- do(val)
- return frame.Tail(frame.STOP), frame.LATER
- case node.VariableNode, node.ParameterNode:
- val = scope.GoTypeFrom(sm.Select(p.Object().Adr())).(string)
- do(val)
- return frame.Tail(frame.STOP), frame.LATER
- case node.DerefNode:
- rt2.Push(rt2.New(p), f)
- return This(expectExpr(f, p, func(...IN) (out OUT) {
- v := rt2.ValueOf(f)[p.Adr()]
- assert.For(v != nil, 60)
- val = scope.GoTypeFrom(v).(string)
- do(val)
- out.do = Tail(STOP)
- out.next = LATER
- return out
- }))
- default:
- halt.As(100, "unsupported param", reflect.TypeOf(p))
- }
- panic(0)
- }
- func init() {
- sys = make(map[string]func(f frame.Frame, par node.Node) (frame.Sequence, frame.WAIT))
- sys["go_process"] = process
- }
- func syscall(f frame.Frame) (frame.Sequence, frame.WAIT) {
- n := rt2.NodeOf(f)
- name := n.Left().Object().Name()
- return sys[name](f, n.Right())
- }
- func callSeq(f frame.Frame) (seq frame.Sequence, ret frame.WAIT) {
- n := rt2.NodeOf(f)
- call := func(proc node.Node, d context.Domain) {
- nf := rt2.New(proc)
- rt2.Push(nf, f)
- if d != nil {
- rt2.ReplaceDomain(nf, d)
- }
- //передаем ссылку на цепочку значений параметров в данные фрейма входа в процедуру
- if (n.Right() != nil) && (proc.Object() != nil) {
- rt2.RegOf(nf)[proc.Object()] = n.Right()
- } else {
- //fmt.Println("no data for call")
- }
- seq = func(f frame.Frame) (seq frame.Sequence, ret frame.WAIT) {
- // rt2.DataOf(f.Parent())[n] = rt2.DataOf(f)[n.Left().Object()]
- rt2.ValueOf(f.Parent())[n.Adr()] = rt2.ValueOf(f)[n.Left().Object().Adr()]
- return frame.End()
- }
- ret = frame.LATER
- }
- switch p := n.Left().(type) {
- case node.ProcedureNode:
- m := rt_mod.DomainModule(f.Domain())
- ml := f.Domain().Discover(context.UNIVERSE).(context.Domain).Discover(context.MOD).(rt_mod.List)
- if p.Super() {
- fmt.Println("supercall, stop for now")
- seq = Propose(Tail(STOP))
- ret = frame.NOW
- } else {
- if imp := m.ImportOf(n.Left().Object()); imp == "" {
- proc := m.NodeByObject(n.Left().Object())
- fmt.Println(len(proc), len(n.Left().Object().Ref()))
- call(proc[0], nil)
- } else {
- m := ml.Loaded(imp)
- proc := m.ObjectByName(m.Enter, n.Left().Object().Name())
- nl := m.NodeByObject(proc)
- utils.PrintFrame("foreign call", len(nl))
- call(nl[0], f.Domain().Discover(context.UNIVERSE).(context.Domain).Discover(imp).(context.Domain))
- }
- }
- case node.VariableNode:
- m := rt_mod.DomainModule(f.Domain())
- sc := f.Domain().Discover(context.SCOPE).(scope.Manager)
- obj := scope.GoTypeFrom(sc.Select(n.Left().Object().Adr()))
- if obj, ok := obj.(object.Object); ok {
- proc := m.NodeByObject(obj)
- call(proc[0], nil)
- } else {
- name := n.Left().Object().Name()
- switch {
- case name == "go_process":
- return syscall(f)
- default:
- panic(fmt.Sprintln("unknown sysproc variable", name))
- }
- }
- default:
- panic("unknown call left")
- }
- return seq, ret
- }
|