runtime: tracePanic

This commit is contained in:
visualfc
2024-06-19 22:36:55 +08:00
parent 870dde232a
commit aecde91d33
6 changed files with 99 additions and 10 deletions

View File

@@ -171,7 +171,6 @@ Common Go syntax is already supported. Except for the following, which needs to
* map (Very limited support) * map (Very limited support)
* chan (Not supported yet) * chan (Not supported yet)
* generics (Not supported yet)
Here are some examples related to Go syntax: Here are some examples related to Go syntax:
@@ -212,7 +211,7 @@ Here are the Go packages that can be imported correctly:
* [strconv](https://pkg.go.dev/strconv) * [strconv](https://pkg.go.dev/strconv)
* [syscall](https://pkg.go.dev/syscall) (partially) * [syscall](https://pkg.go.dev/syscall) (partially)
* [sync](https://pkg.go.dev/sync) (partially) * [sync](https://pkg.go.dev/sync) (partially)
* [sync/atomic](https://pkg.go.dev/sync/atomic) (partially) * [sync/atomic](https://pkg.go.dev/sync/atomic)
## Dependencies ## Dependencies

17
c/bitcast/_cast/cast.c Normal file
View File

@@ -0,0 +1,17 @@
typedef union {
double d;
float f;
long v;
} castUnion;
double llgoToFloat64(long v) {
castUnion k;
k.v = v;
return k.d;
}
float llgoToFloat32(long v) {
castUnion k;
k.v = v;
return k.f;
}

30
c/bitcast/bitcast.go Normal file
View File

@@ -0,0 +1,30 @@
/*
* 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 bitcast
import _ "unsafe"
const (
LLGoFiles = "_cast/cast.c"
LLGoPackage = "link"
)
//go:linkname ToFloat64 C.llgoToFloat64
func ToFloat64(v uintptr) float64
//go:linkname ToFloat32 C.llgoToFloat32
func ToFloat32(v uintptr) float32

BIN
c/bitcast/llgo_autogen.lla Normal file

Binary file not shown.

View File

@@ -16,7 +16,6 @@
package atomic package atomic
// llgo:skipall
import ( import (
"unsafe" "unsafe"
) )

View File

@@ -20,6 +20,7 @@ import (
"unsafe" "unsafe"
"github.com/goplus/llgo/c" "github.com/goplus/llgo/c"
"github.com/goplus/llgo/c/bitcast"
"github.com/goplus/llgo/c/pthread" "github.com/goplus/llgo/c/pthread"
"github.com/goplus/llgo/internal/abi" "github.com/goplus/llgo/internal/abi"
) )
@@ -58,7 +59,7 @@ func Panic(v any) {
func Rethrow(link *Defer) { func Rethrow(link *Defer) {
if ptr := excepKey.Get(); ptr != nil { if ptr := excepKey.Get(); ptr != nil {
if link == nil { if link == nil {
TracePanic(*(*Eface)(ptr)) TracePanic(*(*any)(ptr))
c.Free(ptr) c.Free(ptr)
c.Exit(2) c.Exit(2)
} else { } else {
@@ -77,14 +78,57 @@ func init() {
// ----------------------------------------------------------------------------- // -----------------------------------------------------------------------------
func unpackEface(i any) *eface {
return (*eface)(unsafe.Pointer(&i))
}
// TracePanic prints panic message. // TracePanic prints panic message.
func TracePanic(v Eface) { func TracePanic(v any) {
kind := v._type.Kind() print("panic: ")
switch { switch e := v.(type) {
case kind == abi.String: case nil:
stringTracef(c.Stderr, c.Str("panic: %s\n"), *(*String)(v.data)) println("nil")
return
case (interface{ Error() string }):
println(e.Error())
return
case (interface{ String() string }):
println(e.String())
return
}
e := unpackEface(v)
switch e.Kind() {
case abi.Int, abi.Int8, abi.Int16, abi.Int32, abi.Int64:
if isDirectIface(e._type) {
println(int64(uintptr(e.data)))
} else {
println(*(*int64)(e.data))
}
case abi.Uint, abi.Uint8, abi.Uint16, abi.Uint32, abi.Uint64, abi.Uintptr:
if isDirectIface(e._type) {
println(uint64(uintptr(e.data)))
} else {
println(*(*uint64)(e.data))
}
case abi.Float32:
if isDirectIface(e._type) {
println(bitcast.ToFloat32((uintptr(e.data))))
} else {
println(*(*float32)(e.data))
}
case abi.Float64:
if isDirectIface(e._type) {
println(bitcast.ToFloat64(uintptr(e.data)))
} else {
println(*(*float64)(e.data))
}
case abi.String:
println(*(*string)(e.data))
default:
// TODO kind to e._type.Str_
print("(", e.Kind(), ") ")
println(e.data)
} }
// TODO(xsw): other message type
} }
func stringTracef(fp c.FilePtr, format *c.Char, s String) { func stringTracef(fp c.FilePtr, format *c.Char, s String) {