mirror of
https://github.com/kubenetworks/kubevpn.git
synced 2025-10-13 11:04:03 +08:00
618 lines
11 KiB
Go
618 lines
11 KiB
Go
package vm
|
|
|
|
//go:generate sh -c "go run ./func_types > ./func_types[generated].go"
|
|
|
|
import (
|
|
"fmt"
|
|
"reflect"
|
|
"regexp"
|
|
"sort"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/expr-lang/expr/builtin"
|
|
"github.com/expr-lang/expr/conf"
|
|
"github.com/expr-lang/expr/file"
|
|
"github.com/expr-lang/expr/internal/deref"
|
|
"github.com/expr-lang/expr/vm/runtime"
|
|
)
|
|
|
|
func Run(program *Program, env any) (any, error) {
|
|
if program == nil {
|
|
return nil, fmt.Errorf("program is nil")
|
|
}
|
|
vm := VM{}
|
|
return vm.Run(program, env)
|
|
}
|
|
|
|
func Debug() *VM {
|
|
vm := &VM{
|
|
debug: true,
|
|
step: make(chan struct{}, 0),
|
|
curr: make(chan int, 0),
|
|
}
|
|
return vm
|
|
}
|
|
|
|
type VM struct {
|
|
Stack []any
|
|
Scopes []*Scope
|
|
Variables []any
|
|
MemoryBudget uint
|
|
ip int
|
|
memory uint
|
|
debug bool
|
|
step chan struct{}
|
|
curr chan int
|
|
}
|
|
|
|
func (vm *VM) Run(program *Program, env any) (_ any, err error) {
|
|
defer func() {
|
|
if r := recover(); r != nil {
|
|
var location file.Location
|
|
if vm.ip-1 < len(program.locations) {
|
|
location = program.locations[vm.ip-1]
|
|
}
|
|
f := &file.Error{
|
|
Location: location,
|
|
Message: fmt.Sprintf("%v", r),
|
|
}
|
|
if err, ok := r.(error); ok {
|
|
f.Wrap(err)
|
|
}
|
|
err = f.Bind(program.source)
|
|
}
|
|
}()
|
|
|
|
if vm.Stack == nil {
|
|
vm.Stack = make([]any, 0, 2)
|
|
} else {
|
|
vm.Stack = vm.Stack[0:0]
|
|
}
|
|
if vm.Scopes != nil {
|
|
vm.Scopes = vm.Scopes[0:0]
|
|
}
|
|
if len(vm.Variables) < program.variables {
|
|
vm.Variables = make([]any, program.variables)
|
|
}
|
|
|
|
if vm.MemoryBudget == 0 {
|
|
vm.MemoryBudget = conf.DefaultMemoryBudget
|
|
}
|
|
vm.memory = 0
|
|
vm.ip = 0
|
|
|
|
for vm.ip < len(program.Bytecode) {
|
|
if debug && vm.debug {
|
|
<-vm.step
|
|
}
|
|
|
|
op := program.Bytecode[vm.ip]
|
|
arg := program.Arguments[vm.ip]
|
|
vm.ip += 1
|
|
|
|
switch op {
|
|
|
|
case OpInvalid:
|
|
panic("invalid opcode")
|
|
|
|
case OpPush:
|
|
vm.push(program.Constants[arg])
|
|
|
|
case OpInt:
|
|
vm.push(arg)
|
|
|
|
case OpPop:
|
|
vm.pop()
|
|
|
|
case OpStore:
|
|
vm.Variables[arg] = vm.pop()
|
|
|
|
case OpLoadVar:
|
|
vm.push(vm.Variables[arg])
|
|
|
|
case OpLoadConst:
|
|
vm.push(runtime.Fetch(env, program.Constants[arg]))
|
|
|
|
case OpLoadField:
|
|
vm.push(runtime.FetchField(env, program.Constants[arg].(*runtime.Field)))
|
|
|
|
case OpLoadFast:
|
|
vm.push(env.(map[string]any)[program.Constants[arg].(string)])
|
|
|
|
case OpLoadMethod:
|
|
vm.push(runtime.FetchMethod(env, program.Constants[arg].(*runtime.Method)))
|
|
|
|
case OpLoadFunc:
|
|
vm.push(program.functions[arg])
|
|
|
|
case OpFetch:
|
|
b := vm.pop()
|
|
a := vm.pop()
|
|
vm.push(runtime.Fetch(a, b))
|
|
|
|
case OpFetchField:
|
|
a := vm.pop()
|
|
vm.push(runtime.FetchField(a, program.Constants[arg].(*runtime.Field)))
|
|
|
|
case OpLoadEnv:
|
|
vm.push(env)
|
|
|
|
case OpMethod:
|
|
a := vm.pop()
|
|
vm.push(runtime.FetchMethod(a, program.Constants[arg].(*runtime.Method)))
|
|
|
|
case OpTrue:
|
|
vm.push(true)
|
|
|
|
case OpFalse:
|
|
vm.push(false)
|
|
|
|
case OpNil:
|
|
vm.push(nil)
|
|
|
|
case OpNegate:
|
|
v := runtime.Negate(vm.pop())
|
|
vm.push(v)
|
|
|
|
case OpNot:
|
|
v := vm.pop().(bool)
|
|
vm.push(!v)
|
|
|
|
case OpEqual:
|
|
b := vm.pop()
|
|
a := vm.pop()
|
|
vm.push(runtime.Equal(a, b))
|
|
|
|
case OpEqualInt:
|
|
b := vm.pop()
|
|
a := vm.pop()
|
|
vm.push(a.(int) == b.(int))
|
|
|
|
case OpEqualString:
|
|
b := vm.pop()
|
|
a := vm.pop()
|
|
vm.push(a.(string) == b.(string))
|
|
|
|
case OpJump:
|
|
vm.ip += arg
|
|
|
|
case OpJumpIfTrue:
|
|
if vm.current().(bool) {
|
|
vm.ip += arg
|
|
}
|
|
|
|
case OpJumpIfFalse:
|
|
if !vm.current().(bool) {
|
|
vm.ip += arg
|
|
}
|
|
|
|
case OpJumpIfNil:
|
|
if runtime.IsNil(vm.current()) {
|
|
vm.ip += arg
|
|
}
|
|
|
|
case OpJumpIfNotNil:
|
|
if !runtime.IsNil(vm.current()) {
|
|
vm.ip += arg
|
|
}
|
|
|
|
case OpJumpIfEnd:
|
|
scope := vm.scope()
|
|
if scope.Index >= scope.Len {
|
|
vm.ip += arg
|
|
}
|
|
|
|
case OpJumpBackward:
|
|
vm.ip -= arg
|
|
|
|
case OpIn:
|
|
b := vm.pop()
|
|
a := vm.pop()
|
|
vm.push(runtime.In(a, b))
|
|
|
|
case OpLess:
|
|
b := vm.pop()
|
|
a := vm.pop()
|
|
vm.push(runtime.Less(a, b))
|
|
|
|
case OpMore:
|
|
b := vm.pop()
|
|
a := vm.pop()
|
|
vm.push(runtime.More(a, b))
|
|
|
|
case OpLessOrEqual:
|
|
b := vm.pop()
|
|
a := vm.pop()
|
|
vm.push(runtime.LessOrEqual(a, b))
|
|
|
|
case OpMoreOrEqual:
|
|
b := vm.pop()
|
|
a := vm.pop()
|
|
vm.push(runtime.MoreOrEqual(a, b))
|
|
|
|
case OpAdd:
|
|
b := vm.pop()
|
|
a := vm.pop()
|
|
vm.push(runtime.Add(a, b))
|
|
|
|
case OpSubtract:
|
|
b := vm.pop()
|
|
a := vm.pop()
|
|
vm.push(runtime.Subtract(a, b))
|
|
|
|
case OpMultiply:
|
|
b := vm.pop()
|
|
a := vm.pop()
|
|
vm.push(runtime.Multiply(a, b))
|
|
|
|
case OpDivide:
|
|
b := vm.pop()
|
|
a := vm.pop()
|
|
vm.push(runtime.Divide(a, b))
|
|
|
|
case OpModulo:
|
|
b := vm.pop()
|
|
a := vm.pop()
|
|
vm.push(runtime.Modulo(a, b))
|
|
|
|
case OpExponent:
|
|
b := vm.pop()
|
|
a := vm.pop()
|
|
vm.push(runtime.Exponent(a, b))
|
|
|
|
case OpRange:
|
|
b := vm.pop()
|
|
a := vm.pop()
|
|
min := runtime.ToInt(a)
|
|
max := runtime.ToInt(b)
|
|
size := max - min + 1
|
|
if size <= 0 {
|
|
size = 0
|
|
}
|
|
vm.memGrow(uint(size))
|
|
vm.push(runtime.MakeRange(min, max))
|
|
|
|
case OpMatches:
|
|
b := vm.pop()
|
|
a := vm.pop()
|
|
if runtime.IsNil(a) || runtime.IsNil(b) {
|
|
vm.push(false)
|
|
break
|
|
}
|
|
match, err := regexp.MatchString(b.(string), a.(string))
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
vm.push(match)
|
|
|
|
case OpMatchesConst:
|
|
a := vm.pop()
|
|
if runtime.IsNil(a) {
|
|
vm.push(false)
|
|
break
|
|
}
|
|
r := program.Constants[arg].(*regexp.Regexp)
|
|
vm.push(r.MatchString(a.(string)))
|
|
|
|
case OpContains:
|
|
b := vm.pop()
|
|
a := vm.pop()
|
|
if runtime.IsNil(a) || runtime.IsNil(b) {
|
|
vm.push(false)
|
|
break
|
|
}
|
|
vm.push(strings.Contains(a.(string), b.(string)))
|
|
|
|
case OpStartsWith:
|
|
b := vm.pop()
|
|
a := vm.pop()
|
|
if runtime.IsNil(a) || runtime.IsNil(b) {
|
|
vm.push(false)
|
|
break
|
|
}
|
|
vm.push(strings.HasPrefix(a.(string), b.(string)))
|
|
|
|
case OpEndsWith:
|
|
b := vm.pop()
|
|
a := vm.pop()
|
|
if runtime.IsNil(a) || runtime.IsNil(b) {
|
|
vm.push(false)
|
|
break
|
|
}
|
|
vm.push(strings.HasSuffix(a.(string), b.(string)))
|
|
|
|
case OpSlice:
|
|
from := vm.pop()
|
|
to := vm.pop()
|
|
node := vm.pop()
|
|
vm.push(runtime.Slice(node, from, to))
|
|
|
|
case OpCall:
|
|
fn := reflect.ValueOf(vm.pop())
|
|
size := arg
|
|
in := make([]reflect.Value, size)
|
|
for i := int(size) - 1; i >= 0; i-- {
|
|
param := vm.pop()
|
|
if param == nil {
|
|
in[i] = reflect.Zero(fn.Type().In(i))
|
|
} else {
|
|
in[i] = reflect.ValueOf(param)
|
|
}
|
|
}
|
|
out := fn.Call(in)
|
|
if len(out) == 2 && out[1].Type() == errorType && !out[1].IsNil() {
|
|
panic(out[1].Interface().(error))
|
|
}
|
|
vm.push(out[0].Interface())
|
|
|
|
case OpCall0:
|
|
out, err := program.functions[arg]()
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
vm.push(out)
|
|
|
|
case OpCall1:
|
|
a := vm.pop()
|
|
out, err := program.functions[arg](a)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
vm.push(out)
|
|
|
|
case OpCall2:
|
|
b := vm.pop()
|
|
a := vm.pop()
|
|
out, err := program.functions[arg](a, b)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
vm.push(out)
|
|
|
|
case OpCall3:
|
|
c := vm.pop()
|
|
b := vm.pop()
|
|
a := vm.pop()
|
|
out, err := program.functions[arg](a, b, c)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
vm.push(out)
|
|
|
|
case OpCallN:
|
|
fn := vm.pop().(Function)
|
|
size := arg
|
|
in := make([]any, size)
|
|
for i := int(size) - 1; i >= 0; i-- {
|
|
in[i] = vm.pop()
|
|
}
|
|
out, err := fn(in...)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
vm.push(out)
|
|
|
|
case OpCallFast:
|
|
fn := vm.pop().(func(...any) any)
|
|
size := arg
|
|
in := make([]any, size)
|
|
for i := int(size) - 1; i >= 0; i-- {
|
|
in[i] = vm.pop()
|
|
}
|
|
vm.push(fn(in...))
|
|
|
|
case OpCallSafe:
|
|
fn := vm.pop().(SafeFunction)
|
|
size := arg
|
|
in := make([]any, size)
|
|
for i := int(size) - 1; i >= 0; i-- {
|
|
in[i] = vm.pop()
|
|
}
|
|
out, mem, err := fn(in...)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
vm.memGrow(mem)
|
|
vm.push(out)
|
|
|
|
case OpCallTyped:
|
|
vm.push(vm.call(vm.pop(), arg))
|
|
|
|
case OpCallBuiltin1:
|
|
vm.push(builtin.Builtins[arg].Fast(vm.pop()))
|
|
|
|
case OpArray:
|
|
size := vm.pop().(int)
|
|
vm.memGrow(uint(size))
|
|
array := make([]any, size)
|
|
for i := size - 1; i >= 0; i-- {
|
|
array[i] = vm.pop()
|
|
}
|
|
vm.push(array)
|
|
|
|
case OpMap:
|
|
size := vm.pop().(int)
|
|
vm.memGrow(uint(size))
|
|
m := make(map[string]any)
|
|
for i := size - 1; i >= 0; i-- {
|
|
value := vm.pop()
|
|
key := vm.pop()
|
|
m[key.(string)] = value
|
|
}
|
|
vm.push(m)
|
|
|
|
case OpLen:
|
|
vm.push(runtime.Len(vm.current()))
|
|
|
|
case OpCast:
|
|
switch arg {
|
|
case 0:
|
|
vm.push(runtime.ToInt(vm.pop()))
|
|
case 1:
|
|
vm.push(runtime.ToInt64(vm.pop()))
|
|
case 2:
|
|
vm.push(runtime.ToFloat64(vm.pop()))
|
|
}
|
|
|
|
case OpDeref:
|
|
a := vm.pop()
|
|
vm.push(deref.Interface(a))
|
|
|
|
case OpIncrementIndex:
|
|
vm.scope().Index++
|
|
|
|
case OpDecrementIndex:
|
|
scope := vm.scope()
|
|
scope.Index--
|
|
|
|
case OpIncrementCount:
|
|
scope := vm.scope()
|
|
scope.Count++
|
|
|
|
case OpGetIndex:
|
|
vm.push(vm.scope().Index)
|
|
|
|
case OpGetCount:
|
|
scope := vm.scope()
|
|
vm.push(scope.Count)
|
|
|
|
case OpGetLen:
|
|
scope := vm.scope()
|
|
vm.push(scope.Len)
|
|
|
|
case OpGetAcc:
|
|
vm.push(vm.scope().Acc)
|
|
|
|
case OpSetAcc:
|
|
vm.scope().Acc = vm.pop()
|
|
|
|
case OpSetIndex:
|
|
scope := vm.scope()
|
|
scope.Index = vm.pop().(int)
|
|
|
|
case OpPointer:
|
|
scope := vm.scope()
|
|
vm.push(scope.Array.Index(scope.Index).Interface())
|
|
|
|
case OpThrow:
|
|
panic(vm.pop().(error))
|
|
|
|
case OpCreate:
|
|
switch arg {
|
|
case 1:
|
|
vm.push(make(groupBy))
|
|
case 2:
|
|
scope := vm.scope()
|
|
var desc bool
|
|
switch vm.pop().(string) {
|
|
case "asc":
|
|
desc = false
|
|
case "desc":
|
|
desc = true
|
|
default:
|
|
panic("unknown order, use asc or desc")
|
|
}
|
|
vm.push(&runtime.SortBy{
|
|
Desc: desc,
|
|
Array: make([]any, 0, scope.Len),
|
|
Values: make([]any, 0, scope.Len),
|
|
})
|
|
default:
|
|
panic(fmt.Sprintf("unknown OpCreate argument %v", arg))
|
|
}
|
|
|
|
case OpGroupBy:
|
|
scope := vm.scope()
|
|
key := vm.pop()
|
|
item := scope.Array.Index(scope.Index).Interface()
|
|
scope.Acc.(groupBy)[key] = append(scope.Acc.(groupBy)[key], item)
|
|
|
|
case OpSortBy:
|
|
scope := vm.scope()
|
|
value := vm.pop()
|
|
item := scope.Array.Index(scope.Index).Interface()
|
|
sortable := scope.Acc.(*runtime.SortBy)
|
|
sortable.Array = append(sortable.Array, item)
|
|
sortable.Values = append(sortable.Values, value)
|
|
|
|
case OpSort:
|
|
scope := vm.scope()
|
|
sortable := scope.Acc.(*runtime.SortBy)
|
|
sort.Sort(sortable)
|
|
vm.memGrow(uint(scope.Len))
|
|
vm.push(sortable.Array)
|
|
|
|
case OpProfileStart:
|
|
span := program.Constants[arg].(*Span)
|
|
span.start = time.Now()
|
|
|
|
case OpProfileEnd:
|
|
span := program.Constants[arg].(*Span)
|
|
span.Duration += time.Since(span.start).Nanoseconds()
|
|
|
|
case OpBegin:
|
|
a := vm.pop()
|
|
array := reflect.ValueOf(a)
|
|
vm.Scopes = append(vm.Scopes, &Scope{
|
|
Array: array,
|
|
Len: array.Len(),
|
|
})
|
|
|
|
case OpEnd:
|
|
vm.Scopes = vm.Scopes[:len(vm.Scopes)-1]
|
|
|
|
default:
|
|
panic(fmt.Sprintf("unknown bytecode %#x", op))
|
|
}
|
|
|
|
if debug && vm.debug {
|
|
vm.curr <- vm.ip
|
|
}
|
|
}
|
|
|
|
if debug && vm.debug {
|
|
close(vm.curr)
|
|
close(vm.step)
|
|
}
|
|
|
|
if len(vm.Stack) > 0 {
|
|
return vm.pop(), nil
|
|
}
|
|
|
|
return nil, nil
|
|
}
|
|
|
|
func (vm *VM) push(value any) {
|
|
vm.Stack = append(vm.Stack, value)
|
|
}
|
|
|
|
func (vm *VM) current() any {
|
|
return vm.Stack[len(vm.Stack)-1]
|
|
}
|
|
|
|
func (vm *VM) pop() any {
|
|
value := vm.Stack[len(vm.Stack)-1]
|
|
vm.Stack = vm.Stack[:len(vm.Stack)-1]
|
|
return value
|
|
}
|
|
|
|
func (vm *VM) memGrow(size uint) {
|
|
vm.memory += size
|
|
if vm.memory >= vm.MemoryBudget {
|
|
panic("memory budget exceeded")
|
|
}
|
|
}
|
|
|
|
func (vm *VM) scope() *Scope {
|
|
return vm.Scopes[len(vm.Scopes)-1]
|
|
}
|
|
|
|
func (vm *VM) Step() {
|
|
vm.step <- struct{}{}
|
|
}
|
|
|
|
func (vm *VM) Position() chan int {
|
|
return vm.curr
|
|
}
|