This repository was archived by the owner on Nov 23, 2024. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmemory.go
142 lines (107 loc) · 3.2 KB
/
memory.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
package wasmer
import (
"encoding/binary"
"encoding/json"
"github.com/temphia/temphia/code/backend/libx/xutils/kosher"
"github.com/wasmerio/wasmer-go/wasmer"
)
type Memory struct {
inner []byte
instance *wasmer.Instance
executor *Executor
_allocFunc func(...interface{}) (interface{}, error)
}
func (e *Executor) getMemory() Memory {
allocfunc, err := e.instance.Exports.GetFunction("allocate_bytes")
if err != nil {
panic(err)
}
return Memory{
inner: importMemory(e.instance),
instance: e.instance,
executor: e,
_allocFunc: allocfunc,
}
}
func (m Memory) allocate(size int32) int32 {
out, err := m._allocFunc(size)
if err != nil {
panic(err)
}
// fixme => do we need to reimport memory []byte (inner)
// cz mem could have expanded
return out.(int32)
}
// read
func (m Memory) getString(offset wasmer.Value, len wasmer.Value) string {
_start := offset.I32()
_end := _start + len.I32()
return string(m.inner[_start:_end])
}
func (m Memory) getBytes(offset wasmer.Value, len wasmer.Value) []byte {
_start := offset.I32()
_end := _start + len.I32()
return (m.inner[_start:_end])
}
func (m Memory) getJSON(offset wasmer.Value, len wasmer.Value, target any) error {
out := m.getBytes(offset, len)
return json.Unmarshal(out, target)
}
// write
func (m Memory) rwrite(respPtr wasmer.Value, respLen wasmer.Value, err error) ([]wasmer.Value, error) {
if err != nil {
m.writeError(respPtr, respLen, err)
return ErrAtom, nil
}
return OkAtom, nil
}
func (m Memory) rwriteErr(respPtr wasmer.Value, respLen wasmer.Value, err error) ([]wasmer.Value, error) {
m.writeError(respPtr, respLen, err)
return ErrAtom, nil
}
func (m Memory) rwriteBytes1(respPtr wasmer.Value, respLen wasmer.Value, data []byte, err error) ([]wasmer.Value, error) {
if err != nil {
m.writeError(respPtr, respLen, err)
return ErrAtom, nil
}
return m.rwriteBytes2(respPtr, respLen, data)
}
func (m Memory) rwriteBytes2(respPtr wasmer.Value, respLen wasmer.Value, data []byte) ([]wasmer.Value, error) {
m.writeBytes(data, respPtr, respLen)
return OkAtom, nil
}
func (m Memory) rwriteJson(respPtr wasmer.Value, respLen wasmer.Value, resp any, err error) ([]wasmer.Value, error) {
if err != nil {
m.writeError(respPtr, respLen, err)
return ErrAtom, nil
}
m.writeJson(respPtr, respLen, resp)
return OkAtom, nil
}
func (m Memory) writeJson(respPtr wasmer.Value, respLen wasmer.Value, resp any) error {
out, err := json.Marshal(resp)
if err != nil {
return err
}
m.writeBytes(out, respPtr, respLen)
return nil
}
func (m Memory) writeError(respPtr wasmer.Value, respLen wasmer.Value, err error) {
m.writeBytes(kosher.Byte(err.Error()), respPtr, respLen)
}
func (m Memory) writeBytes(data []byte, respPtr wasmer.Value, respLen wasmer.Value) {
size := int32(len(data))
ptr := m.allocate(size)
chunk := m.inner[ptr : ptr+size]
copy(chunk, data)
m.writeUint32(respPtr.I32(), ptr)
m.writeUint32(respLen.I32(), size)
}
func (m Memory) writeUint32(offset int32, value int32) {
b := m.inner[offset : offset+4]
binary.LittleEndian.PutUint32(b, uint32(value))
}
func (m Memory) writeUint64(offset int32, value int64) {
b := m.inner[offset : offset+8]
binary.LittleEndian.PutUint32(b, uint32(value))
}