Files
trivy/pkg/module/wasm/sdk.go
2022-06-15 15:23:00 +03:00

180 lines
4.1 KiB
Go

//go:build tinygo.wasm
package wasm
// This package is designed to be imported by WASM modules.
// TinyGo can build this package, but Go cannot.
import (
"fmt"
"reflect"
"unsafe"
"github.com/mailru/easyjson"
"github.com/aquasecurity/trivy/pkg/module/api"
"github.com/aquasecurity/trivy/pkg/module/serialize"
)
func Debug(message string) {
message = fmt.Sprintf("Module %s: %s", module.Name(), message)
ptr, size := stringToPtr(message)
_debug(ptr, size)
}
func Info(message string) {
message = fmt.Sprintf("Module %s: %s", module.Name(), message)
ptr, size := stringToPtr(message)
_info(ptr, size)
}
func Warn(message string) {
message = fmt.Sprintf("Module %s: %s", module.Name(), message)
ptr, size := stringToPtr(message)
_warn(ptr, size)
}
func Error(message string) {
message = fmt.Sprintf("Module %s: %s", module.Name(), message)
ptr, size := stringToPtr(message)
_error(ptr, size)
}
//go:wasm-module env
//export debug
func _debug(ptr uint32, size uint32)
//go:wasm-module env
//export info
func _info(ptr uint32, size uint32)
//go:wasm-module env
//export warn
func _warn(ptr uint32, size uint32)
//go:wasm-module env
//export error
func _error(ptr uint32, size uint32)
var module api.Module
func RegisterModule(p api.Module) {
module = p
}
//export name
func _name() uint64 {
name := module.Name()
ptr, size := stringToPtr(name)
return (uint64(ptr) << uint64(32)) | uint64(size)
}
//export api_version
func _apiVersion() uint32 {
return api.Version
}
//export version
func _version() uint32 {
return uint32(module.Version())
}
//export is_analyzer
func _isAnalyzer() uint64 {
if _, ok := module.(api.Analyzer); !ok {
return 0
}
return 1
}
//export required
func _required() uint64 {
files := module.(api.Analyzer).RequiredFiles()
ss := serialize.StringSlice(files)
return marshal(ss)
}
//export analyze
func _analyze(ptr, size uint32) uint64 {
filePath := ptrToString(ptr, size)
custom, err := module.(api.Analyzer).Analyze(filePath)
if err != nil {
Error(fmt.Sprintf("analyze error: %s", err))
return 0
}
return marshal(custom)
}
//export is_post_scanner
func _isPostScanner() uint64 {
if _, ok := module.(api.PostScanner); !ok {
return 0
}
return 1
}
//export post_scan_spec
func _post_scan_spec() uint64 {
return marshal(module.(api.PostScanner).PostScanSpec())
}
//export post_scan
func _post_scan(ptr, size uint32) uint64 {
var results serialize.Results
if err := unmarshal(ptr, size, &results); err != nil {
Error(fmt.Sprintf("post scan error: %s", err))
return 0
}
results, err := module.(api.PostScanner).PostScan(results)
if err != nil {
Error(fmt.Sprintf("post scan error: %s", err))
return 0
}
return marshal(results)
}
func marshal(v easyjson.Marshaler) uint64 {
b, err := easyjson.Marshal(v)
if err != nil {
Error(fmt.Sprintf("marshal error: %s", err))
return 0
}
p := uintptr(unsafe.Pointer(&b[0]))
return (uint64(p) << uint64(32)) | uint64(len(b))
}
func unmarshal(ptr, size uint32, v easyjson.Unmarshaler) error {
var b []byte
s := (*reflect.SliceHeader)(unsafe.Pointer(&b))
s.Len = uintptr(size)
s.Cap = uintptr(size)
s.Data = uintptr(ptr)
if err := easyjson.Unmarshal(b, v); err != nil {
return fmt.Errorf("unmarshal error: %s", err)
}
return nil
}
// ptrToString returns a string from WebAssembly compatible numeric types representing its pointer and length.
func ptrToString(ptr uint32, size uint32) string {
// Get a slice view of the underlying bytes in the stream. We use SliceHeader, not StringHeader
// as it allows us to fix the capacity to what was allocated.
return *(*string)(unsafe.Pointer(&reflect.SliceHeader{
Data: uintptr(ptr),
Len: uintptr(size), // Tinygo requires these as uintptrs even if they are int fields.
Cap: uintptr(size), // ^^ See https://github.com/tinygo-org/tinygo/issues/1284
}))
}
// stringToPtr returns a pointer and size pair for the given string in a way compatible with WebAssembly numeric types.
func stringToPtr(s string) (uint32, uint32) {
buf := []byte(s)
ptr := &buf[0]
unsafePtr := uintptr(unsafe.Pointer(ptr))
return uint32(unsafePtr), uint32(len(buf))
}