2024-12-12 19:44:53 +00:00
|
|
|
//go:build unix
|
2024-05-27 15:46:15 +00:00
|
|
|
|
|
|
|
package util
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"os"
|
|
|
|
"unsafe"
|
|
|
|
|
|
|
|
"github.com/tetratelabs/wazero/api"
|
|
|
|
"golang.org/x/sys/unix"
|
|
|
|
)
|
|
|
|
|
|
|
|
type mmapState struct {
|
|
|
|
regions []*MappedRegion
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *mmapState) new(ctx context.Context, mod api.Module, size int32) *MappedRegion {
|
|
|
|
// Find unused region.
|
|
|
|
for _, r := range s.regions {
|
|
|
|
if !r.used && r.size == size {
|
|
|
|
return r
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Allocate page aligned memmory.
|
|
|
|
alloc := mod.ExportedFunction("aligned_alloc")
|
2024-08-15 00:30:58 +00:00
|
|
|
stack := [...]uint64{
|
2024-05-27 15:46:15 +00:00
|
|
|
uint64(unix.Getpagesize()),
|
|
|
|
uint64(size),
|
|
|
|
}
|
|
|
|
if err := alloc.CallWithStack(ctx, stack[:]); err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
if stack[0] == 0 {
|
|
|
|
panic(OOMErr)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Save the newly allocated region.
|
|
|
|
ptr := uint32(stack[0])
|
|
|
|
buf := View(mod, ptr, uint64(size))
|
2024-12-20 21:37:19 +00:00
|
|
|
res := &MappedRegion{
|
2024-05-27 15:46:15 +00:00
|
|
|
Ptr: ptr,
|
|
|
|
size: size,
|
2024-12-20 21:37:19 +00:00
|
|
|
addr: unsafe.Pointer(&buf[0]),
|
|
|
|
}
|
|
|
|
s.regions = append(s.regions, res)
|
|
|
|
return res
|
2024-05-27 15:46:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type MappedRegion struct {
|
2024-07-05 10:06:03 +00:00
|
|
|
addr unsafe.Pointer
|
2024-05-27 15:46:15 +00:00
|
|
|
Ptr uint32
|
|
|
|
size int32
|
|
|
|
used bool
|
|
|
|
}
|
|
|
|
|
2024-11-07 00:16:28 +00:00
|
|
|
func MapRegion(ctx context.Context, mod api.Module, f *os.File, offset int64, size int32, readOnly bool) (*MappedRegion, error) {
|
2024-05-27 15:46:15 +00:00
|
|
|
s := ctx.Value(moduleKey{}).(*moduleState)
|
|
|
|
r := s.new(ctx, mod, size)
|
2024-11-07 00:16:28 +00:00
|
|
|
err := r.mmap(f, offset, readOnly)
|
2024-05-27 15:46:15 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return r, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *MappedRegion) Unmap() error {
|
|
|
|
// We can't munmap the region, otherwise it could be remaped.
|
|
|
|
// Instead, convert it to a protected, private, anonymous mapping.
|
|
|
|
// If successful, it can be reused for a subsequent mmap.
|
2024-07-05 10:06:03 +00:00
|
|
|
_, err := unix.MmapPtr(-1, 0, r.addr, uintptr(r.size),
|
|
|
|
unix.PROT_NONE, unix.MAP_PRIVATE|unix.MAP_FIXED|unix.MAP_ANON)
|
2024-05-27 15:46:15 +00:00
|
|
|
r.used = err != nil
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2024-11-07 00:16:28 +00:00
|
|
|
func (r *MappedRegion) mmap(f *os.File, offset int64, readOnly bool) error {
|
|
|
|
prot := unix.PROT_READ
|
|
|
|
if !readOnly {
|
|
|
|
prot |= unix.PROT_WRITE
|
|
|
|
}
|
2024-07-05 10:06:03 +00:00
|
|
|
_, err := unix.MmapPtr(int(f.Fd()), offset, r.addr, uintptr(r.size),
|
|
|
|
prot, unix.MAP_SHARED|unix.MAP_FIXED)
|
2024-05-27 15:46:15 +00:00
|
|
|
r.used = err == nil
|
|
|
|
return err
|
|
|
|
}
|