/* * Copyright (c) 2024 The GoPlus Authors (goplus.org). All rights reserved. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package runtime import ( "unsafe" c "github.com/goplus/llgo/runtime/internal/clite" "github.com/goplus/llgo/runtime/internal/clite/debug" "github.com/goplus/llgo/runtime/internal/clite/pthread" "github.com/goplus/llgo/runtime/internal/clite/setjmp" ) // ----------------------------------------------------------------------------- // Defer presents defer statements in a function. type Defer struct { Addr unsafe.Pointer // sigjmpbuf Bits uintptr Link *Defer Reth unsafe.Pointer // block address after Rethrow Rund unsafe.Pointer // block address after RunDefers Args unsafe.Pointer // defer func and args links } // Recover recovers a panic. func Recover() (ret any) { ptr := excepKey.Get() if ptr != nil { excepKey.Set(nil) ret = *(*any)(ptr) c.Free(ptr) } return } // Panic panics with a value. func Panic(v any) { ptr := c.Malloc(unsafe.Sizeof(v)) *(*any)(ptr) = v excepKey.Set(ptr) Rethrow((*Defer)(c.GoDeferData())) } // Rethrow rethrows a panic. func Rethrow(link *Defer) { if ptr := excepKey.Get(); ptr != nil { if link == nil { TracePanic(*(*any)(ptr)) debug.StackTrace(0, func(fr *debug.Frame) bool { var info debug.Info debug.Addrinfo(unsafe.Pointer(fr.PC), &info) c.Fprintf(c.Stderr, c.Str("[0x%08X %s+0x%x, SP = 0x%x]\n"), fr.PC, fr.Name, fr.Offset, fr.SP) return true }) c.Free(ptr) c.Exit(2) } else { c.Siglongjmp(link.Addr, 1) } } else if link == nil && goexitKey.Get() != nil { if pthread.Equal(mainThread, pthread.Self()) != 0 { fatal("no goroutines (main called runtime.Goexit) - deadlock!") c.Exit(2) } pthread.Exit(nil) } } var ( excepKey pthread.Key goexitKey pthread.Key mainThread pthread.Thread ) func Goexit() { goexitKey.Set(unsafe.Pointer(&goexitKey)) Rethrow((*Defer)(c.GoDeferData())) } func init() { excepKey.Create(nil) goexitKey.Create(nil) mainThread = pthread.Self() } // ----------------------------------------------------------------------------- // TracePanic prints panic message. func TracePanic(v any) { print("panic: ") printany(v) println("\n") } /* func stringTracef(fp c.FilePtr, format *c.Char, s String) { cs := c.Alloca(uintptr(s.len) + 1) c.Fprintf(fp, format, CStrCopy(cs, s)) } */ // ----------------------------------------------------------------------------- // New allocates memory and initializes it to zero. func New(t *Type) unsafe.Pointer { return AllocZ(t.Size_) } // NewArray allocates memory for an array and initializes it to zero. func NewArray(t *Type, n int) unsafe.Pointer { return AllocZ(uintptr(n) * t.Size_) } // ----------------------------------------------------------------------------- // TODO(xsw): check this // must match declarations in runtime/map.go. const MaxZero = 1024 var ZeroVal [MaxZero]byte // func init() { // signal.Signal(c.Int(syscall.SIGSEGV), func(v c.Int) { // switch syscall.Signal(v) { // case syscall.SIGSEGV: // panic(errorString("invalid memory address or nil pointer dereference")) // default: // var buf [20]byte // panic(errorString("unexpected signal value: " + string(itoa(buf[:], uint64(v))))) // } // }) // } // ----------------------------------------------------------------------------- type SigjmpBuf struct { Unused [setjmp.SigjmpBufSize]byte } // -----------------------------------------------------------------------------