Compare commits
55 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
210c483635 | ||
|
|
8ca4212650 | ||
|
|
c91dba5ed6 | ||
|
|
c8de05f101 | ||
|
|
0ac7cde498 | ||
|
|
108829ad9c | ||
|
|
c5b96f4e9c | ||
|
|
4c2099d33e | ||
|
|
fe5de95008 | ||
|
|
4b0cfc0751 | ||
|
|
c2bf05942e | ||
|
|
df37f80c8e | ||
|
|
2c19d7218d | ||
|
|
34899e8d36 | ||
|
|
bf8c10ed25 | ||
|
|
93c33e08c2 | ||
|
|
3992dd1dd0 | ||
|
|
cab29c2be7 | ||
|
|
f582657ffd | ||
|
|
2823ac1aee | ||
|
|
289caa7cc2 | ||
|
|
0a8bad46b5 | ||
|
|
aa4f518262 | ||
|
|
f76fa879fc | ||
|
|
8d70aba1f5 | ||
|
|
a44bb35aec | ||
|
|
4fda2b656f | ||
|
|
e626d00fdf | ||
|
|
bf09e3c3ae | ||
|
|
753dcd3301 | ||
|
|
8b5dee510e | ||
|
|
9cb73fbf78 | ||
|
|
e6b4deb5c4 | ||
|
|
8848222728 | ||
|
|
3cd62994c7 | ||
|
|
dd93a97790 | ||
|
|
e40e2d2d14 | ||
|
|
26f8ce7b5a | ||
|
|
9a61e374b5 | ||
|
|
9b12e9819c | ||
|
|
5d0a91239c | ||
|
|
c848278690 | ||
|
|
2ebb929e2c | ||
|
|
b34334ba93 | ||
|
|
05a01cd803 | ||
|
|
9ac0c06f26 | ||
|
|
52af22b0e8 | ||
|
|
81cfc73b48 | ||
|
|
f892bfccdf | ||
|
|
dbed8fefac | ||
|
|
ca14637909 | ||
|
|
7db618fba5 | ||
|
|
29c74c09ce | ||
|
|
a2b5b9f97e | ||
|
|
6a05aa4e53 |
@@ -312,7 +312,9 @@ Here are the Go packages that can be imported correctly:
|
|||||||
* [crypto/sha1](https://pkg.go.dev/crypto/sha1)
|
* [crypto/sha1](https://pkg.go.dev/crypto/sha1)
|
||||||
* [crypto/sha256](https://pkg.go.dev/crypto/sha256)
|
* [crypto/sha256](https://pkg.go.dev/crypto/sha256)
|
||||||
* [crypto/sha512](https://pkg.go.dev/crypto/sha512) (partially)
|
* [crypto/sha512](https://pkg.go.dev/crypto/sha512) (partially)
|
||||||
|
* [crypto/hmac](https://pkg.go.dev/crypto/hmac) (partially)
|
||||||
* [crypto/rand](https://pkg.go.dev/crypto/rand) (partially)
|
* [crypto/rand](https://pkg.go.dev/crypto/rand) (partially)
|
||||||
|
* [crypto/subtle](https://pkg.go.dev/crypto/subtle) (partially)
|
||||||
* [regexp](https://pkg.go.dev/regexp)
|
* [regexp](https://pkg.go.dev/regexp)
|
||||||
* [regexp/syntax](https://pkg.go.dev/regexp/syntax)
|
* [regexp/syntax](https://pkg.go.dev/regexp/syntax)
|
||||||
* [go/token](https://pkg.go.dev/go/token)
|
* [go/token](https://pkg.go.dev/go/token)
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ import (
|
|||||||
"math/big"
|
"math/big"
|
||||||
)
|
)
|
||||||
|
|
||||||
func main() {
|
func fib() {
|
||||||
// Initialize two big ints with the first two numbers in the sequence.
|
// Initialize two big ints with the first two numbers in the sequence.
|
||||||
a := big.NewInt(0)
|
a := big.NewInt(0)
|
||||||
b := big.NewInt(1)
|
b := big.NewInt(1)
|
||||||
@@ -23,3 +23,41 @@ func main() {
|
|||||||
}
|
}
|
||||||
fmt.Println(a) // 100-digit Fibonacci number
|
fmt.Println(a) // 100-digit Fibonacci number
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func abs() {
|
||||||
|
a := big.NewInt(64)
|
||||||
|
b := big.NewInt(-52)
|
||||||
|
a.Set(b)
|
||||||
|
a.Abs(a)
|
||||||
|
a.Set(big.NewInt(-164))
|
||||||
|
a.Abs(a)
|
||||||
|
fmt.Println("value: ", a.String())
|
||||||
|
}
|
||||||
|
|
||||||
|
func neg() {
|
||||||
|
fmt.Println("value: ", big.NewInt(-64).Neg(big.NewInt(-64)))
|
||||||
|
fmt.Println("value: ", big.NewInt(64).Neg(big.NewInt(64)))
|
||||||
|
fmt.Println("value: ", big.NewInt(0).Neg(big.NewInt(0)))
|
||||||
|
}
|
||||||
|
|
||||||
|
func calc() {
|
||||||
|
a := big.NewInt(64)
|
||||||
|
b := big.NewInt(-52)
|
||||||
|
c := big.NewInt(54)
|
||||||
|
fmt.Println("value:", a.Add(a, b))
|
||||||
|
fmt.Println("value:", a.Sub(b, c))
|
||||||
|
d := big.NewInt(10)
|
||||||
|
e := big.NewInt(4)
|
||||||
|
fmt.Println("value:", d.Mul(d, e))
|
||||||
|
}
|
||||||
|
|
||||||
|
func bitop() {
|
||||||
|
a := big.NewInt(4)
|
||||||
|
fmt.Println("value:", a.Lsh(a, 1))
|
||||||
|
b := big.NewInt(16)
|
||||||
|
fmt.Println("value:", b.Rsh(b, 2))
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
bitop()
|
||||||
|
}
|
||||||
|
|||||||
15
_cmptest/hmacdemo/hmac.go
Normal file
15
_cmptest/hmacdemo/hmac.go
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"crypto/hmac"
|
||||||
|
"crypto/sha1"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
h := hmac.New(sha1.New, []byte("<key>"))
|
||||||
|
io.WriteString(h, "The fog is getting thicker!")
|
||||||
|
io.WriteString(h, "And Leon's getting laaarger!")
|
||||||
|
fmt.Printf("%x", h.Sum(nil))
|
||||||
|
}
|
||||||
3
c/c.go
3
c/c.go
@@ -79,6 +79,9 @@ func AllocaNew[T any]() *T { return nil }
|
|||||||
//go:linkname Malloc C.malloc
|
//go:linkname Malloc C.malloc
|
||||||
func Malloc(size uintptr) Pointer
|
func Malloc(size uintptr) Pointer
|
||||||
|
|
||||||
|
//go:linkname Calloc C.calloc
|
||||||
|
func Calloc(num uintptr, size uintptr) Pointer
|
||||||
|
|
||||||
//go:linkname Free C.free
|
//go:linkname Free C.free
|
||||||
func Free(ptr Pointer)
|
func Free(ptr Pointer)
|
||||||
|
|
||||||
|
|||||||
@@ -1,61 +0,0 @@
|
|||||||
package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"os"
|
|
||||||
|
|
||||||
"github.com/goplus/llgo/c"
|
|
||||||
"github.com/goplus/llgo/c/clang"
|
|
||||||
)
|
|
||||||
|
|
||||||
func visit(cursor, parent clang.Cursor, clientData c.Pointer) clang.ChildVisitResult {
|
|
||||||
depth := *(*c.Uint)(clientData)
|
|
||||||
printAST(cursor, depth+1)
|
|
||||||
return clang.ChildVisit_Continue
|
|
||||||
}
|
|
||||||
|
|
||||||
func printAST(cursor clang.Cursor, depth c.Uint) {
|
|
||||||
cursorKind := cursor.Kind.String()
|
|
||||||
|
|
||||||
cursorSpelling := cursor.String()
|
|
||||||
|
|
||||||
for i := c.Uint(0); i < depth; i++ {
|
|
||||||
c.Fputs(c.Str(" "), c.Stdout)
|
|
||||||
}
|
|
||||||
|
|
||||||
c.Printf(c.Str("%s: %s\n"), cursorKind.CStr(), cursorSpelling.CStr())
|
|
||||||
|
|
||||||
cursorKind.Dispose()
|
|
||||||
cursorSpelling.Dispose()
|
|
||||||
|
|
||||||
clang.VisitChildren(cursor, visit, c.Pointer(&depth))
|
|
||||||
}
|
|
||||||
|
|
||||||
func main() {
|
|
||||||
if c.Argc != 2 {
|
|
||||||
fmt.Fprintln(os.Stderr, "Usage: castdump <headerFile>")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
sourceFile := *c.Advance(c.Argv, 1)
|
|
||||||
|
|
||||||
index := clang.CreateIndex(0, 0)
|
|
||||||
|
|
||||||
unit := index.ParseTranslationUnit(
|
|
||||||
sourceFile,
|
|
||||||
nil, 0,
|
|
||||||
nil, 0,
|
|
||||||
clang.TranslationUnit_None,
|
|
||||||
)
|
|
||||||
|
|
||||||
if unit == nil {
|
|
||||||
println("Unable to parse translation unit. Quitting.")
|
|
||||||
c.Exit(1)
|
|
||||||
}
|
|
||||||
|
|
||||||
cursor := unit.Cursor()
|
|
||||||
|
|
||||||
printAST(cursor, 0)
|
|
||||||
|
|
||||||
unit.Dispose()
|
|
||||||
index.Dispose()
|
|
||||||
}
|
|
||||||
@@ -80,7 +80,7 @@ func printFuncInfo(cursor clang.Cursor) {
|
|||||||
}
|
}
|
||||||
c.Printf(c.Str("%s\n"), cursorStr.CStr())
|
c.Printf(c.Str("%s\n"), cursorStr.CStr())
|
||||||
|
|
||||||
if cursor.Kind == clang.CXXMethod || cursor.Kind == clang.FunctionDecl {
|
if cursor.Kind == clang.CursorCXXMethod || cursor.Kind == clang.CursorFunctionDecl {
|
||||||
c.Printf(c.Str("symbol:%s\n"), symbol.CStr())
|
c.Printf(c.Str("symbol:%s\n"), symbol.CStr())
|
||||||
|
|
||||||
typeStr := cursor.ResultType().String()
|
typeStr := cursor.ResultType().String()
|
||||||
@@ -107,19 +107,19 @@ func printFuncInfo(cursor clang.Cursor) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func visit(cursor, parent clang.Cursor, clientData c.Pointer) clang.ChildVisitResult {
|
func visit(cursor, parent clang.Cursor, clientData c.Pointer) clang.ChildVisitResult {
|
||||||
if cursor.Kind == clang.MacroDefinition {
|
if cursor.Kind == clang.CursorMacroDefinition {
|
||||||
printMarcoInfo(cursor)
|
printMarcoInfo(cursor)
|
||||||
} else if cursor.Kind == clang.Namespace {
|
} else if cursor.Kind == clang.CursorNamespace {
|
||||||
nameStr := cursor.String()
|
nameStr := cursor.String()
|
||||||
context.setNamespaceName(c.GoString(nameStr.CStr()))
|
context.setNamespaceName(c.GoString(nameStr.CStr()))
|
||||||
clang.VisitChildren(cursor, visit, nil)
|
clang.VisitChildren(cursor, visit, nil)
|
||||||
context.setNamespaceName("")
|
context.setNamespaceName("")
|
||||||
} else if cursor.Kind == clang.ClassDecl {
|
} else if cursor.Kind == clang.CursorClassDecl {
|
||||||
nameStr := cursor.String()
|
nameStr := cursor.String()
|
||||||
context.setClassName(c.GoString(nameStr.CStr()))
|
context.setClassName(c.GoString(nameStr.CStr()))
|
||||||
clang.VisitChildren(cursor, visit, nil)
|
clang.VisitChildren(cursor, visit, nil)
|
||||||
context.setClassName("")
|
context.setClassName("")
|
||||||
} else if cursor.Kind == clang.CXXMethod || cursor.Kind == clang.FunctionDecl {
|
} else if cursor.Kind == clang.CursorCXXMethod || cursor.Kind == clang.CursorFunctionDecl {
|
||||||
printFuncInfo(cursor)
|
printFuncInfo(cursor)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -33,6 +33,24 @@ void wrap_clang_getCursorType(CXCursor *cur, CXType *typ) { *typ = clang_getCurs
|
|||||||
|
|
||||||
void wrap_clang_getCursorResultType(CXCursor *cur, CXType *typ) { *typ = clang_getCursorResultType(*cur); }
|
void wrap_clang_getCursorResultType(CXCursor *cur, CXType *typ) { *typ = clang_getCursorResultType(*cur); }
|
||||||
|
|
||||||
|
void wrap_clang_getResultType(CXType *typ, CXType *resultTyp) { *resultTyp = clang_getResultType(*typ); }
|
||||||
|
|
||||||
|
int wrap_clang_getNumArgTypes(CXType *typ) { return clang_getNumArgTypes(*typ); }
|
||||||
|
|
||||||
|
void wrap_clang_getArgType(CXType *typ, unsigned i, CXType *argTyp) { *argTyp = clang_getArgType(*typ, i); }
|
||||||
|
|
||||||
|
long long wrap_clang_getEnumConstantDeclValue(CXCursor *cur) { return clang_getEnumConstantDeclValue(*cur); }
|
||||||
|
|
||||||
|
void wrap_clang_getPointeeType(CXType *pointerTyp, CXType *pointeeTyp) {
|
||||||
|
*pointeeTyp = clang_getPointeeType(*pointerTyp);
|
||||||
|
}
|
||||||
|
|
||||||
|
void wrap_clang_getArrayElementType(CXType *arrayTyp, CXType *elemTyp) {
|
||||||
|
*elemTyp = clang_getArrayElementType(*arrayTyp);
|
||||||
|
}
|
||||||
|
|
||||||
|
void wrap_clang_getCanonicalType(CXType *typ, CXType *canonicalType) { *canonicalType = clang_getCanonicalType(*typ); }
|
||||||
|
|
||||||
CXString wrap_clang_getTypeSpelling(CXType *typ) { return clang_getTypeSpelling(*typ); }
|
CXString wrap_clang_getTypeSpelling(CXType *typ) { return clang_getTypeSpelling(*typ); }
|
||||||
|
|
||||||
CXString wrap_clang_getTokenSpelling(CXTranslationUnit unit, CXToken *token) {
|
CXString wrap_clang_getTokenSpelling(CXTranslationUnit unit, CXToken *token) {
|
||||||
@@ -46,6 +64,10 @@ void wrap_clang_getSpellingLocation(CXSourceLocation *loc, CXFile *file, unsigne
|
|||||||
clang_getSpellingLocation(*loc, file, line, column, offset);
|
clang_getSpellingLocation(*loc, file, line, column, offset);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
enum CX_CXXAccessSpecifier wrap_clang_getCXXAccessSpecifier(CXCursor *cursor) {
|
||||||
|
return clang_getCXXAccessSpecifier(*cursor);
|
||||||
|
}
|
||||||
|
|
||||||
void wrap_clang_getCursorExtent(CXCursor *cur, CXSourceRange *range) { *range = clang_getCursorExtent(*cur); }
|
void wrap_clang_getCursorExtent(CXCursor *cur, CXSourceRange *range) { *range = clang_getCursorExtent(*cur); }
|
||||||
|
|
||||||
void wrap_clang_tokenize(CXTranslationUnit unit, CXSourceRange *Range, CXToken **Tokens, unsigned *NumTokens) {
|
void wrap_clang_tokenize(CXTranslationUnit unit, CXSourceRange *Range, CXToken **Tokens, unsigned *NumTokens) {
|
||||||
|
|||||||
871
c/clang/clang.go
871
c/clang/clang.go
File diff suppressed because it is too large
Load Diff
41
c/libuv/_demo/async/async.go
Normal file
41
c/libuv/_demo/async/async.go
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/goplus/llgo/c"
|
||||||
|
"github.com/goplus/llgo/c/libuv"
|
||||||
|
)
|
||||||
|
|
||||||
|
func ensure(b bool, msg string) {
|
||||||
|
if !b {
|
||||||
|
panic(msg)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
loop := libuv.LoopNew()
|
||||||
|
defer loop.Close()
|
||||||
|
|
||||||
|
a := &libuv.Async{}
|
||||||
|
r := loop.Async(a, func(a *libuv.Async) {
|
||||||
|
println("async callback")
|
||||||
|
a.Close(nil) // or loop.Stop()
|
||||||
|
})
|
||||||
|
ensure(r == 0, "Async failed")
|
||||||
|
|
||||||
|
go func() {
|
||||||
|
println("begin async task")
|
||||||
|
c.Usleep(100 * 1000)
|
||||||
|
println("send async event")
|
||||||
|
ensure(a.Send() == 0, "Send failed")
|
||||||
|
}()
|
||||||
|
|
||||||
|
loop.Run(libuv.RUN_DEFAULT)
|
||||||
|
println("done")
|
||||||
|
}
|
||||||
|
|
||||||
|
/*Expected Output:
|
||||||
|
begin async task
|
||||||
|
send async event
|
||||||
|
async callback
|
||||||
|
done
|
||||||
|
*/
|
||||||
@@ -31,7 +31,7 @@ func main() {
|
|||||||
libuv.FsOpen(loop, &openReq, c.Str("example.txt"), os.O_RDONLY, 0, onOpen)
|
libuv.FsOpen(loop, &openReq, c.Str("example.txt"), os.O_RDONLY, 0, onOpen)
|
||||||
|
|
||||||
// Run the loop
|
// Run the loop
|
||||||
result := libuv.Run(loop, libuv.RUN_DEFAULT)
|
result := loop.Run(libuv.RUN_DEFAULT)
|
||||||
|
|
||||||
if result != 0 {
|
if result != 0 {
|
||||||
c.Fprintf(c.Stderr, c.Str("Error in Run: %s\n"), libuv.Strerror(libuv.Errno(result)))
|
c.Fprintf(c.Stderr, c.Str("Error in Run: %s\n"), libuv.Strerror(libuv.Errno(result)))
|
||||||
@@ -43,14 +43,14 @@ func main() {
|
|||||||
|
|
||||||
func onOpen(req *libuv.Fs) {
|
func onOpen(req *libuv.Fs) {
|
||||||
// Check for errors
|
// Check for errors
|
||||||
if libuv.FsGetResult(req) < 0 {
|
if req.GetResult() < 0 {
|
||||||
c.Fprintf(c.Stderr, c.Str("Error opening file: %s\n"), libuv.Strerror(libuv.Errno(libuv.FsGetResult(req))))
|
c.Fprintf(c.Stderr, c.Str("Error opening file: %s\n"), libuv.Strerror(libuv.Errno(req.GetResult())))
|
||||||
libuv.LoopClose(loop)
|
loop.Close()
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// Store the file descriptor
|
// Store the file descriptor
|
||||||
file = libuv.File(libuv.FsGetResult(req))
|
file = libuv.File(req.GetResult())
|
||||||
|
|
||||||
// Init buffer
|
// Init buffer
|
||||||
iov = libuv.InitBuf((*c.Char)(unsafe.Pointer(&buffer[0])), c.Uint(unsafe.Sizeof(buffer)))
|
iov = libuv.InitBuf((*c.Char)(unsafe.Pointer(&buffer[0])), c.Uint(unsafe.Sizeof(buffer)))
|
||||||
@@ -68,28 +68,28 @@ func readFile() {
|
|||||||
readRes := libuv.FsRead(loop, &readReq, file, &iov, 1, -1, onRead)
|
readRes := libuv.FsRead(loop, &readReq, file, &iov, 1, -1, onRead)
|
||||||
if readRes != 0 {
|
if readRes != 0 {
|
||||||
c.Printf(c.Str("Error in FsRead: %s (code: %d)\n"), libuv.Strerror(libuv.Errno(readRes)), readRes)
|
c.Printf(c.Str("Error in FsRead: %s (code: %d)\n"), libuv.Strerror(libuv.Errno(readRes)), readRes)
|
||||||
libuv.FsReqCleanup(&readReq)
|
readReq.ReqCleanup()
|
||||||
libuv.LoopClose(loop)
|
loop.Close()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func onRead(req *libuv.Fs) {
|
func onRead(req *libuv.Fs) {
|
||||||
// Cleanup the request
|
// Cleanup the request
|
||||||
defer libuv.FsReqCleanup(req)
|
defer req.ReqCleanup()
|
||||||
// Check for errors
|
// Check for errors
|
||||||
if libuv.FsGetResult(req) < 0 {
|
if req.GetResult() < 0 {
|
||||||
c.Fprintf(c.Stderr, c.Str("Read error: %s\n"), libuv.Strerror(libuv.Errno(libuv.FsGetResult(req))))
|
c.Fprintf(c.Stderr, c.Str("Read error: %s\n"), libuv.Strerror(libuv.Errno(req.GetResult())))
|
||||||
} else if libuv.FsGetResult(req) == 0 {
|
} else if req.GetResult() == 0 {
|
||||||
// Close the file
|
// Close the file
|
||||||
closeRes := libuv.FsClose(loop, &closeReq, libuv.File(libuv.FsGetResult(&openReq)), onClose)
|
closeRes := libuv.FsClose(loop, &closeReq, libuv.File(openReq.GetResult()), onClose)
|
||||||
if closeRes != 0 {
|
if closeRes != 0 {
|
||||||
c.Printf(c.Str("Error in FsClose: %s (code: %d)\n"), libuv.Strerror(libuv.Errno(closeRes)), closeRes)
|
c.Printf(c.Str("Error in FsClose: %s (code: %d)\n"), libuv.Strerror(libuv.Errno(closeRes)), closeRes)
|
||||||
libuv.LoopClose(loop)
|
loop.Close()
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
c.Printf(c.Str("Read %d bytes\n"), libuv.FsGetResult(req))
|
c.Printf(c.Str("Read %d bytes\n"), req.GetResult())
|
||||||
c.Printf(c.Str("Read content: %.*s\n"), libuv.FsGetResult(req), (*c.Char)(unsafe.Pointer(&buffer[0])))
|
c.Printf(c.Str("Read content: %.*s\n"), req.GetResult(), (*c.Char)(unsafe.Pointer(&buffer[0])))
|
||||||
// Read the file again
|
// Read the file again
|
||||||
readFile()
|
readFile()
|
||||||
}
|
}
|
||||||
@@ -97,8 +97,8 @@ func onRead(req *libuv.Fs) {
|
|||||||
|
|
||||||
func onClose(req *libuv.Fs) {
|
func onClose(req *libuv.Fs) {
|
||||||
// Check for errors
|
// Check for errors
|
||||||
if libuv.FsGetResult(req) < 0 {
|
if req.GetResult() < 0 {
|
||||||
c.Fprintf(c.Stderr, c.Str("Error closing file: %s\n"), libuv.Strerror(libuv.Errno(libuv.FsGetResult(req))))
|
c.Fprintf(c.Stderr, c.Str("Error closing file: %s\n"), libuv.Strerror(libuv.Errno(req.GetResult())))
|
||||||
} else {
|
} else {
|
||||||
c.Printf(c.Str("\nFile closed successfully.\n"))
|
c.Printf(c.Str("\nFile closed successfully.\n"))
|
||||||
}
|
}
|
||||||
@@ -106,10 +106,10 @@ func onClose(req *libuv.Fs) {
|
|||||||
|
|
||||||
func cleanup() {
|
func cleanup() {
|
||||||
// Cleanup the requests
|
// Cleanup the requests
|
||||||
libuv.FsReqCleanup(&openReq)
|
openReq.ReqCleanup()
|
||||||
libuv.FsReqCleanup(&closeReq)
|
closeReq.ReqCleanup()
|
||||||
// Close the loop
|
// Close the loop
|
||||||
result := libuv.LoopClose(loop)
|
result := loop.Close()
|
||||||
if result != 0 {
|
if result != 0 {
|
||||||
c.Fprintf(c.Stderr, c.Str("Error in LoopClose: %s\n"), libuv.Strerror(libuv.Errno(result)))
|
c.Fprintf(c.Stderr, c.Str("Error in LoopClose: %s\n"), libuv.Strerror(libuv.Errno(result)))
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,8 +1,6 @@
|
|||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"unsafe"
|
|
||||||
|
|
||||||
"github.com/goplus/llgo/c"
|
"github.com/goplus/llgo/c"
|
||||||
"github.com/goplus/llgo/c/libuv"
|
"github.com/goplus/llgo/c/libuv"
|
||||||
"github.com/goplus/llgo/c/net"
|
"github.com/goplus/llgo/c/net"
|
||||||
@@ -11,18 +9,18 @@ import (
|
|||||||
var DEFAULT_PORT c.Int = 8080
|
var DEFAULT_PORT c.Int = 8080
|
||||||
var DEFAULT_BACKLOG c.Int = 128
|
var DEFAULT_BACKLOG c.Int = 128
|
||||||
|
|
||||||
type WriteReq struct {
|
var (
|
||||||
Req libuv.Write
|
Req *libuv.Write
|
||||||
Buf libuv.Buf
|
Buf libuv.Buf
|
||||||
}
|
)
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
// Initialize the default event loop
|
// Initialize the default event loop
|
||||||
var loop = libuv.DefaultLoop()
|
var loop = libuv.DefaultLoop()
|
||||||
|
|
||||||
// Initialize a TCP server
|
// Initialize a TCP server
|
||||||
var server libuv.Tcp
|
server := &libuv.Tcp{}
|
||||||
libuv.InitTcp(loop, &server)
|
libuv.InitTcp(loop, server)
|
||||||
|
|
||||||
// Set up the address to bind the server to
|
// Set up the address to bind the server to
|
||||||
var addr net.SockaddrIn
|
var addr net.SockaddrIn
|
||||||
@@ -30,22 +28,20 @@ func main() {
|
|||||||
c.Printf(c.Str("Listening on %s:%d\n"), c.Str("0.0.0.0"), DEFAULT_PORT)
|
c.Printf(c.Str("Listening on %s:%d\n"), c.Str("0.0.0.0"), DEFAULT_PORT)
|
||||||
|
|
||||||
// Bind the server to the specified address and port
|
// Bind the server to the specified address and port
|
||||||
(&server).Bind((*net.SockAddr)(c.Pointer(&addr)), 0)
|
server.Bind((*net.SockAddr)(c.Pointer(&addr)), 0)
|
||||||
res := (*libuv.Stream)(&server).Listen(DEFAULT_BACKLOG, OnNewConnection)
|
res := (*libuv.Stream)(server).Listen(DEFAULT_BACKLOG, OnNewConnection)
|
||||||
if res != 0 {
|
if res != 0 {
|
||||||
c.Fprintf(c.Stderr, c.Str("Listen error: %s\n"), libuv.Strerror((libuv.Errno(res))))
|
c.Fprintf(c.Stderr, c.Str("Listen error: %s\n"), libuv.Strerror(libuv.Errno(res)))
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// Start listening for incoming connections
|
// Start listening for incoming connections
|
||||||
libuv.Run(loop, libuv.RUN_DEFAULT)
|
loop.Run(libuv.RUN_DEFAULT)
|
||||||
}
|
}
|
||||||
|
|
||||||
func FreeWriteReq(req *libuv.Write) {
|
func FreeWriteReq() {
|
||||||
wr := (*WriteReq)(c.Pointer(req))
|
// Free the buffer base.
|
||||||
// Free the buffer base and the WriteReq itself.
|
c.Free(c.Pointer(Buf.Base))
|
||||||
c.Free(c.Pointer(wr.Buf.Base))
|
|
||||||
c.Free(c.Pointer(wr))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func AllocBuffer(handle *libuv.Handle, suggestedSize uintptr, buf *libuv.Buf) {
|
func AllocBuffer(handle *libuv.Handle, suggestedSize uintptr, buf *libuv.Buf) {
|
||||||
@@ -56,29 +52,24 @@ func AllocBuffer(handle *libuv.Handle, suggestedSize uintptr, buf *libuv.Buf) {
|
|||||||
|
|
||||||
func EchoWrite(req *libuv.Write, status c.Int) {
|
func EchoWrite(req *libuv.Write, status c.Int) {
|
||||||
if status != 0 {
|
if status != 0 {
|
||||||
c.Fprintf(c.Stderr, c.Str("Write error: %s\n"), libuv.Strerror((libuv.Errno(status))))
|
c.Fprintf(c.Stderr, c.Str("Write error: %s\n"), libuv.Strerror(libuv.Errno(status)))
|
||||||
}
|
}
|
||||||
FreeWriteReq(req)
|
FreeWriteReq()
|
||||||
}
|
}
|
||||||
|
|
||||||
func EchoRead(client *libuv.Stream, nread c.Long, buf *libuv.Buf) {
|
func EchoRead(client *libuv.Stream, nread c.Long, buf *libuv.Buf) {
|
||||||
if nread > 0 {
|
if nread > 0 {
|
||||||
req := (*WriteReq)(c.Malloc(unsafe.Sizeof(WriteReq{})))
|
|
||||||
if req == nil {
|
|
||||||
c.Fprintf(c.Stderr, c.Str("Failed to allocate memory for write request\n"))
|
|
||||||
c.Free(c.Pointer(buf.Base))
|
|
||||||
return
|
|
||||||
}
|
|
||||||
// Initialize the buffer with the data read.
|
// Initialize the buffer with the data read.
|
||||||
req.Buf = libuv.InitBuf(buf.Base, c.Uint(nread))
|
Buf = libuv.InitBuf(buf.Base, c.Uint(nread))
|
||||||
// Write the data back to the client.
|
// Write the data back to the client.
|
||||||
(&req.Req).Write(client, &req.Buf, 1, EchoWrite)
|
Req = &libuv.Write{}
|
||||||
|
Req.Write(client, &Buf, 1, EchoWrite)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if nread < 0 {
|
if nread < 0 {
|
||||||
// Handle read errors and EOF.
|
// Handle read errors and EOF.
|
||||||
if (libuv.Errno)(nread) != libuv.EOF {
|
if (libuv.Errno)(nread) != libuv.EOF {
|
||||||
c.Fprintf(c.Stderr, c.Str("Read error: %s\n"), libuv.Strerror((libuv.Errno)(nread)))
|
c.Fprintf(c.Stderr, c.Str("Read error: %s\n"), libuv.Strerror(libuv.Errno(nread)))
|
||||||
}
|
}
|
||||||
(*libuv.Handle)(c.Pointer(client)).Close(nil)
|
(*libuv.Handle)(c.Pointer(client)).Close(nil)
|
||||||
}
|
}
|
||||||
@@ -95,7 +86,7 @@ func OnNewConnection(server *libuv.Stream, status c.Int) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Allocate memory for a new client.
|
// Allocate memory for a new client.
|
||||||
client := (*libuv.Tcp)(c.Malloc(unsafe.Sizeof(libuv.Tcp{})))
|
client := &libuv.Tcp{}
|
||||||
|
|
||||||
if client == nil {
|
if client == nil {
|
||||||
c.Fprintf(c.Stderr, c.Str("Failed to allocate memory for client\n"))
|
c.Fprintf(c.Stderr, c.Str("Failed to allocate memory for client\n"))
|
||||||
|
|||||||
5
c/libuv/_wrap/libuv.c
Normal file
5
c/libuv/_wrap/libuv.c
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
#include <uv.h>
|
||||||
|
|
||||||
|
int uv_tcp_get_io_watcher_fd (uv_tcp_t* handle) {
|
||||||
|
return handle->io_watcher.fd;
|
||||||
|
}
|
||||||
50
c/libuv/async.go
Normal file
50
c/libuv/async.go
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
/*
|
||||||
|
* 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 libuv
|
||||||
|
|
||||||
|
import (
|
||||||
|
_ "unsafe"
|
||||||
|
|
||||||
|
"github.com/goplus/llgo/c"
|
||||||
|
)
|
||||||
|
|
||||||
|
// struct uv_async_t
|
||||||
|
type Async struct {
|
||||||
|
Handle
|
||||||
|
// On macOS arm64, sizeof uv_async_t is 128 bytes.
|
||||||
|
// Handle is 92 bytes, so we need 36 bytes to fill the gap.
|
||||||
|
// Maybe reserve more for future use.
|
||||||
|
Unused [36]byte
|
||||||
|
}
|
||||||
|
|
||||||
|
// typedef void (*uv_async_cb)(uv_async_t* handle);
|
||||||
|
// llgo:type C
|
||||||
|
type AsyncCb func(*Async)
|
||||||
|
|
||||||
|
// int uv_async_init(uv_loop_t*, uv_async_t* async, uv_async_cb async_cb);
|
||||||
|
//
|
||||||
|
// llgo:link (*Loop).Async C.uv_async_init
|
||||||
|
func (loop *Loop) Async(a *Async, cb AsyncCb) c.Int {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
// int uv_async_send(uv_async_t* async);
|
||||||
|
//
|
||||||
|
// llgo:link (*Async).Send C.uv_async_send
|
||||||
|
func (a *Async) Send() c.Int {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
101
c/libuv/fs.go
101
c/libuv/fs.go
@@ -106,29 +106,40 @@ type FsPollCb func(handle *FsPoll, status c.Int, events c.Int)
|
|||||||
|
|
||||||
/* Fs related function and method */
|
/* Fs related function and method */
|
||||||
|
|
||||||
//go:linkname FsGetType C.uv_fs_get_type
|
// llgo:link (*Fs).GetType C.uv_fs_get_type
|
||||||
func FsGetType(req *Fs) FsType
|
func (req *Fs) GetType() FsType {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
//go:linkname FsGetPath C.uv_fs_get_path
|
// llgo:link (*Fs).GetPath C.uv_fs_get_path
|
||||||
func FsGetPath(req *Fs) *c.Char
|
func (req *Fs) GetPath() *c.Char {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
//go:linkname FsGetResult C.uv_fs_get_result
|
// llgo:link (*Fs).GetResult C.uv_fs_get_result
|
||||||
func FsGetResult(req *Fs) c.Int
|
func (req *Fs) GetResult() c.Int {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
//go:linkname FsGetPtr C.uv_fs_get_ptr
|
// llgo:link (*Fs).GetPtr C.uv_fs_get_ptr
|
||||||
func FsGetPtr(req *Fs) c.Pointer
|
func (req *Fs) GetPtr() c.Pointer {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
//go:linkname FsGetSystemError C.uv_fs_get_system_error
|
// llgo:link (*Fs).GetSystemError C.uv_fs_get_system_error
|
||||||
func FsGetSystemError(req *Fs) c.Int
|
func (req *Fs) GetSystemError() c.Int {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
//go:linkname FsGetStatBuf C.uv_fs_get_statbuf
|
// llgo:link (*Fs).GetStatBuf C.uv_fs_get_statbuf
|
||||||
func FsGetStatBuf(req *Fs) *Stat
|
func (req *Fs) GetStatBuf() *Stat {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
//go:linkname FsReqCleanup C.uv_fs_req_cleanup
|
// llgo:link (*Fs).ReqCleanup C.uv_fs_req_cleanup
|
||||||
func FsReqCleanup(req *Fs)
|
func (req *Fs) ReqCleanup() {
|
||||||
|
// No return value needed for this method
|
||||||
//go:linkname DefaultLoop C.uv_default_loop
|
}
|
||||||
func DefaultLoop() *Loop
|
|
||||||
|
|
||||||
//go:linkname FsOpen C.uv_fs_open
|
//go:linkname FsOpen C.uv_fs_open
|
||||||
func FsOpen(loop *Loop, req *Fs, path *c.Char, flags c.Int, mode c.Int, cb FsCb) c.Int
|
func FsOpen(loop *Loop, req *Fs, path *c.Char, flags c.Int, mode c.Int, cb FsCb) c.Int
|
||||||
@@ -241,32 +252,56 @@ func FsLchown(loop *Loop, req *Fs, path *c.Char, uid c.Int, gid c.Int, cb FsCb)
|
|||||||
//go:linkname FsLstat C.uv_fs_lstat
|
//go:linkname FsLstat C.uv_fs_lstat
|
||||||
func FsLstat(loop *Loop, req *Fs, path *c.Char, cb FsCb) c.Int
|
func FsLstat(loop *Loop, req *Fs, path *c.Char, cb FsCb) c.Int
|
||||||
|
|
||||||
|
// ----------------------------------------------
|
||||||
|
|
||||||
|
/* FsEvent related function and method */
|
||||||
|
|
||||||
//go:linkname FsEventInit C.uv_fs_event_init
|
//go:linkname FsEventInit C.uv_fs_event_init
|
||||||
func FsEventInit(loop *Loop, handle *FsEvent) c.Int
|
func FsEventInit(loop *Loop, handle *FsEvent) c.Int
|
||||||
|
|
||||||
//go:linkname FsEventStart C.uv_fs_event_start
|
// llgo:link (*FsEvent).Start C.uv_fs_event_start
|
||||||
func FsEventStart(handle *FsEvent, cb FsEventCb, path *c.Char, flags c.Int) c.Int
|
func (handle *FsEvent) Start(cb FsEventCb, path *c.Char, flags c.Int) c.Int {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
//go:linkname FsEventStop C.uv_fs_event_stop
|
// llgo:link (*FsEvent).Stop C.uv_fs_event_stop
|
||||||
func FsEventStop(handle *FsEvent) c.Int
|
func (handle *FsEvent) Stop() c.Int {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
//go:linkname FsEventClose C.uv_fs_event_close
|
// llgo:link (*FsEvent).Close C.uv_fs_event_close
|
||||||
func FsEventClose(handle *FsEvent) c.Int
|
func (handle *FsEvent) Close() c.Int {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
//go:linkname FsEventGetpath C.uv_fs_event_getpath
|
// llgo:link (*FsEvent).Getpath C.uv_fs_event_getpath
|
||||||
func FsEventGetpath(handle *FsEvent) *c.Char
|
func (handle *FsEvent) Getpath() *c.Char {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ----------------------------------------------
|
||||||
|
|
||||||
|
/* FsPoll related function and method */
|
||||||
|
|
||||||
//go:linkname FsPollInit C.uv_fs_poll_init
|
//go:linkname FsPollInit C.uv_fs_poll_init
|
||||||
func FsPollInit(loop *Loop, handle *FsPoll) c.Int
|
func FsPollInit(loop *Loop, handle *FsPoll) c.Int
|
||||||
|
|
||||||
//go:linkname FsPollStart C.uv_fs_poll_start
|
// llgo:link (*FsPoll).Start C.uv_fs_poll_start
|
||||||
func FsPollStart(handle *FsPoll, cb FsPollCb, path *c.Char, interval uint) c.Int
|
func (handle *FsPoll) Start(cb FsPollCb, path *c.Char, interval uint) c.Int {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
//go:linkname FsPollStop C.uv_fs_poll_stop
|
// llgo:link (*FsPoll).Stop C.uv_fs_poll_stop
|
||||||
func FsPollStop(handle *FsPoll) c.Int
|
func (handle *FsPoll) Stop() c.Int {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
//go:linkname FsPollClose C.uv_fs_poll_close
|
// llgo:link (*FsPoll).Close C.uv_fs_poll_close
|
||||||
func FsPollClose(handle *FsPoll) c.Int
|
func (handle *FsPoll) Close() c.Int {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
//go:linkname FsPollGetPath C.uv_fs_poll_getpath
|
// llgo:link (*FsPoll).GetPath C.uv_fs_poll_getpath
|
||||||
func FsPollGetPath(handle *FsPoll) *c.Char
|
func (handle *FsPoll) GetPath() *c.Char {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|||||||
123
c/libuv/libuv.go
123
c/libuv/libuv.go
@@ -9,6 +9,7 @@ import (
|
|||||||
|
|
||||||
const (
|
const (
|
||||||
LLGoPackage = "link: $(pkg-config --libs libuv); -luv"
|
LLGoPackage = "link: $(pkg-config --libs libuv); -luv"
|
||||||
|
LLGoFiles = "$(pkg-config --cflags libuv): _wrap/libuv.c"
|
||||||
)
|
)
|
||||||
|
|
||||||
// ----------------------------------------------
|
// ----------------------------------------------
|
||||||
@@ -103,10 +104,6 @@ type Poll struct {
|
|||||||
|
|
||||||
/* Request types. */
|
/* Request types. */
|
||||||
|
|
||||||
type Shutdown struct {
|
|
||||||
Unused [0]byte
|
|
||||||
}
|
|
||||||
|
|
||||||
type Buf struct {
|
type Buf struct {
|
||||||
Base *c.Char
|
Base *c.Char
|
||||||
Len uintptr
|
Len uintptr
|
||||||
@@ -135,9 +132,6 @@ type GetaddrinfoCb func(req *GetAddrInfo, status c.Int, res *net.AddrInfo)
|
|||||||
// llgo:type C
|
// llgo:type C
|
||||||
type GetnameinfoCb func(req *GetNameInfo, status c.Int, hostname *c.Char, service *c.Char)
|
type GetnameinfoCb func(req *GetNameInfo, status c.Int, hostname *c.Char, service *c.Char)
|
||||||
|
|
||||||
// llgo:type C
|
|
||||||
type ShutdownCb func(req *Shutdown, status c.Int)
|
|
||||||
|
|
||||||
// llgo:type C
|
// llgo:type C
|
||||||
type WalkCb func(handle *Handle, arg c.Pointer)
|
type WalkCb func(handle *Handle, arg c.Pointer)
|
||||||
|
|
||||||
@@ -160,59 +154,88 @@ func ReplaceAllocator(mallocFunc MallocFunc, reallocFunc ReallocFunc, callocFunc
|
|||||||
|
|
||||||
// ----------------------------------------------
|
// ----------------------------------------------
|
||||||
|
|
||||||
// llgo:link (*Shutdown).Shutdown C.uv_shutdown
|
|
||||||
func (shutdown *Shutdown) Shutdown(stream *Stream, shutdownCb ShutdownCb) c.Int {
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
||||||
// ----------------------------------------------
|
|
||||||
|
|
||||||
/* Loop related functions and method. */
|
/* Loop related functions and method. */
|
||||||
|
|
||||||
|
//go:linkname DefaultLoop C.uv_default_loop
|
||||||
|
func DefaultLoop() *Loop
|
||||||
|
|
||||||
//go:linkname LoopSize C.uv_loop_size
|
//go:linkname LoopSize C.uv_loop_size
|
||||||
func LoopSize() uintptr
|
func LoopSize() uintptr
|
||||||
|
|
||||||
//go:linkname Run C.uv_run
|
// llgo:link (*Loop).Run C.uv_run
|
||||||
func Run(loop *Loop, mode RunMode) c.Int
|
func (loop *Loop) Run(mode RunMode) c.Int {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
//go:linkname LoopAlive C.uv_loop_alive
|
// llgo:link (*Loop).Alive C.uv_loop_alive
|
||||||
func LoopAlive(loop *Loop) c.Int
|
func (loop *Loop) Alive() c.Int {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
//go:linkname LoopClose C.uv_loop_close
|
// void uv_stop(uv_loop_t *loop)
|
||||||
func LoopClose(loop *Loop) c.Int
|
//
|
||||||
|
// llgo:link (*Loop).Stop C.uv_stop
|
||||||
|
func (loop *Loop) Stop() {}
|
||||||
|
|
||||||
//go:linkname LoopConfigure C.uv_loop_configure
|
// llgo:link (*Loop).Close C.uv_loop_close
|
||||||
func LoopConfigure(loop *Loop, option LoopOption, arg c.Int) c.Int
|
func (loop *Loop) Close() c.Int {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
//go:linkname LoopDefault C.uv_default_loop
|
// llgo:link (*Loop).Configure C.uv_loop_configure
|
||||||
func LoopDefault() *Loop
|
func (loop *Loop) Configure(option LoopOption, arg c.Int) c.Int {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
//go:linkname LoopDelete C.uv_loop_delete
|
// llgo:link LoopDefault C.uv_default_loop
|
||||||
func LoopDelete(loop *Loop) c.Int
|
func LoopDefault() *Loop {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
//go:linkname LoopFork C.uv_loop_fork
|
// llgo:link (*Loop).Delete C.uv_loop_delete
|
||||||
func LoopFork(loop *Loop) c.Int
|
func (loop *Loop) Delete() c.Int {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
//go:linkname LoopInit C.uv_loop_init
|
// llgo:link (*Loop).Fork C.uv_loop_fork
|
||||||
func LoopInit(loop *Loop) c.Int
|
func (loop *Loop) Fork() c.Int {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
//go:linkname LoopNew C.uv_loop_new
|
// llgo:link (*Loop).Init C.uv_loop_init
|
||||||
func LoopNew() *Loop
|
func (loop *Loop) Init() c.Int {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
//go:linkname LoopNow C.uv_now
|
// llgo:link LoopNew C.uv_loop_new
|
||||||
func LoopNow(loop *Loop) c.UlongLong
|
func LoopNew() *Loop {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
//go:linkname LoopUpdateTime C.uv_update_time
|
// llgo:link (*Loop).Now C.uv_now
|
||||||
func LoopUpdateTime(loop *Loop)
|
func (loop *Loop) Now() c.UlongLong {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
//go:linkname LoopBackendFd C.uv_backend_fd
|
// llgo:link (*Loop).UpdateTime C.uv_update_time
|
||||||
func LoopBackendFd(loop *Loop) c.Int
|
func (loop *Loop) UpdateTime() {
|
||||||
|
// No return value needed for this method
|
||||||
|
}
|
||||||
|
|
||||||
//go:linkname LoopBackendTimeout C.uv_backend_timeout
|
// llgo:link (*Loop).BackendFd C.uv_backend_fd
|
||||||
func LoopBackendTimeout(loop *Loop) c.Int
|
func (loop *Loop) BackendFd() c.Int {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
//go:linkname LoopWalk C.uv_walk
|
// llgo:link (*Loop).BackendTimeout C.uv_backend_timeout
|
||||||
func LoopWalk(loop *Loop, walkCb WalkCb, arg c.Pointer)
|
func (loop *Loop) BackendTimeout() c.Int {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
// llgo:link (*Loop).Walk C.uv_walk
|
||||||
|
func (loop *Loop) Walk(walkCb WalkCb, arg c.Pointer) {
|
||||||
|
// No return value needed for this method
|
||||||
|
}
|
||||||
|
|
||||||
// ----------------------------------------------
|
// ----------------------------------------------
|
||||||
|
|
||||||
@@ -228,11 +251,15 @@ func InitBuf(base *c.Char, len c.Uint) Buf
|
|||||||
//go:linkname PollInit C.uv_poll_init
|
//go:linkname PollInit C.uv_poll_init
|
||||||
func PollInit(loop *Loop, handle *Poll, fd OsFd) c.Int
|
func PollInit(loop *Loop, handle *Poll, fd OsFd) c.Int
|
||||||
|
|
||||||
//go:linkname PollStart C.uv_poll_start
|
|
||||||
func PollStart(handle *Poll, events c.Int, cb PollCb) c.Int
|
|
||||||
|
|
||||||
//go:linkname PollStop C.uv_poll_stop
|
|
||||||
func PollStop(handle *Poll) c.Int
|
|
||||||
|
|
||||||
//go:linkname PollInitSocket C.uv_poll_init_socket
|
//go:linkname PollInitSocket C.uv_poll_init_socket
|
||||||
func PollInitSocket(loop *Loop, handle *Poll, socket c.Int) c.Int
|
func PollInitSocket(loop *Loop, handle *Poll, socket c.Int) c.Int
|
||||||
|
|
||||||
|
// llgo:link (*Poll).Start C.uv_poll_start
|
||||||
|
func (handle *Poll) Start(events c.Int, cb PollCb) c.Int {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
// llgo:link (*Poll).Stop C.uv_poll_stop
|
||||||
|
func (handle *Poll) Stop() c.Int {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|||||||
@@ -117,6 +117,10 @@ type GetNameInfo struct {
|
|||||||
Unused [0]byte
|
Unused [0]byte
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type Shutdown struct {
|
||||||
|
Unused [0]byte
|
||||||
|
}
|
||||||
|
|
||||||
// ----------------------------------------------
|
// ----------------------------------------------
|
||||||
|
|
||||||
/* Function type */
|
/* Function type */
|
||||||
@@ -142,10 +146,19 @@ type WriteCb func(req *Write, status c.Int)
|
|||||||
// llgo:type C
|
// llgo:type C
|
||||||
type ConnectionCb func(server *Stream, status c.Int)
|
type ConnectionCb func(server *Stream, status c.Int)
|
||||||
|
|
||||||
|
// llgo:type C
|
||||||
|
type ShutdownCb func(req *Shutdown, status c.Int)
|
||||||
|
|
||||||
// ----------------------------------------------
|
// ----------------------------------------------
|
||||||
|
|
||||||
/* Handle related function and method */
|
/* Handle related function and method */
|
||||||
|
|
||||||
|
//go:linkname HandleSize C.uv_handle_size
|
||||||
|
func HandleSize(handleType HandleType) uintptr
|
||||||
|
|
||||||
|
//go:linkname HandleTypeName C.uv_handle_type_name
|
||||||
|
func HandleTypeName(handleType HandleType) *c.Char
|
||||||
|
|
||||||
// llgo:link (*Handle).Ref C.uv_ref
|
// llgo:link (*Handle).Ref C.uv_ref
|
||||||
func (handle *Handle) Ref() {}
|
func (handle *Handle) Ref() {}
|
||||||
|
|
||||||
@@ -157,17 +170,11 @@ func (handle *Handle) HasRef() c.Int {
|
|||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
|
|
||||||
//go:linkname HandleSize C.uv_handle_size
|
|
||||||
func HandleSize(handleType HandleType) uintptr
|
|
||||||
|
|
||||||
// llgo:link (*Handle).GetType C.uv_handle_get_type
|
// llgo:link (*Handle).GetType C.uv_handle_get_type
|
||||||
func (handle *Handle) GetType() HandleType {
|
func (handle *Handle) GetType() HandleType {
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
|
|
||||||
//go:linkname HandleTypeName C.uv_handle_type_name
|
|
||||||
func HandleTypeName(handleType HandleType) *c.Char
|
|
||||||
|
|
||||||
// llgo:link (*Handle).GetData C.uv_handle_get_data
|
// llgo:link (*Handle).GetData C.uv_handle_get_data
|
||||||
func (handle *Handle) GetData() c.Pointer {
|
func (handle *Handle) GetData() c.Pointer {
|
||||||
return nil
|
return nil
|
||||||
@@ -204,6 +211,21 @@ func (handle *Handle) Fileno(fd *OsFd) c.Int {
|
|||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// llgo:link (*Handle).IsClosing C.uv_is_closing
|
||||||
|
func (handle *Handle) IsClosing() c.Int {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
// llgo:link (*Handle).IsReadable C.uv_is_readable
|
||||||
|
func (handle *Handle) IsReadable() c.Int {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
// llgo:link (*Handle).IsWritable C.uv_is_writable
|
||||||
|
func (handle *Handle) IsWritable() c.Int {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
//go:linkname Pipe C.uv_pipe
|
//go:linkname Pipe C.uv_pipe
|
||||||
func Pipe(fds [2]File, readFlags c.Int, writeFlags c.Int) c.Int {
|
func Pipe(fds [2]File, readFlags c.Int, writeFlags c.Int) c.Int {
|
||||||
return 0
|
return 0
|
||||||
@@ -214,11 +236,6 @@ func Socketpair(_type c.Int, protocol c.Int, socketVector [2]OsSock, flag0 c.Int
|
|||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
|
|
||||||
// llgo:link (*Handle).IsClosing C.uv_is_closing
|
|
||||||
func (handle *Handle) IsClosing() c.Int {
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
||||||
// ----------------------------------------------
|
// ----------------------------------------------
|
||||||
|
|
||||||
/* Req related function and method */
|
/* Req related function and method */
|
||||||
@@ -226,12 +243,15 @@ func (handle *Handle) IsClosing() c.Int {
|
|||||||
//go:linkname ReqSize C.uv_req_size
|
//go:linkname ReqSize C.uv_req_size
|
||||||
func ReqSize(reqType ReqType) uintptr
|
func ReqSize(reqType ReqType) uintptr
|
||||||
|
|
||||||
|
//go:linkname TypeName C.uv_req_type_name
|
||||||
|
func TypeName(reqType ReqType) *c.Char
|
||||||
|
|
||||||
// llgo:link (*Req).GetData C.uv_req_get_data
|
// llgo:link (*Req).GetData C.uv_req_get_data
|
||||||
func (req *Req) GetData() c.Pointer {
|
func (req *Req) GetData() c.Pointer {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// llgo:link (*Req).SetData C.uv_handle_set_data
|
// llgo:link (*Req).SetData C.uv_req_set_data
|
||||||
func (req *Req) SetData(data c.Pointer) {}
|
func (req *Req) SetData(data c.Pointer) {}
|
||||||
|
|
||||||
// llgo:link (*Req).GetType C.uv_req_get_type
|
// llgo:link (*Req).GetType C.uv_req_get_type
|
||||||
@@ -239,9 +259,6 @@ func (req *Req) GetType() ReqType {
|
|||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
|
|
||||||
//go:linkname TypeName C.uv_req_type_name
|
|
||||||
func TypeName(reqType ReqType) *c.Char
|
|
||||||
|
|
||||||
// ----------------------------------------------
|
// ----------------------------------------------
|
||||||
|
|
||||||
/* Stream related function and method */
|
/* Stream related function and method */
|
||||||
@@ -306,6 +323,11 @@ func (stream *Stream) SetBlocking(blocking c.Int) c.Int {
|
|||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//go:linkname StreamShutdown C.uv_shutdown
|
||||||
|
func StreamShutdown(shutdown *Shutdown, stream *Stream, shutdownCb ShutdownCb) c.Int {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
// ----------------------------------------------
|
// ----------------------------------------------
|
||||||
|
|
||||||
/* Tcp related function and method */
|
/* Tcp related function and method */
|
||||||
@@ -356,6 +378,11 @@ func (tcp *Tcp) CloseReset(closeCb CloseCb) c.Int {
|
|||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// llgo:link (*Tcp).GetIoWatcherFd C.uv_tcp_get_io_watcher_fd
|
||||||
|
func (tcp *Tcp) GetIoWatcherFd() c.Int {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
//go:linkname TcpConnect C.uv_tcp_connect
|
//go:linkname TcpConnect C.uv_tcp_connect
|
||||||
func TcpConnect(req *Connect, tcp *Tcp, addr *net.SockAddr, connectCb ConnectCb) c.Int
|
func TcpConnect(req *Connect, tcp *Tcp, addr *net.SockAddr, connectCb ConnectCb) c.Int
|
||||||
|
|
||||||
@@ -429,9 +456,6 @@ func (udp *Udp) SetTTL(ttl c.Int) c.Int {
|
|||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
|
|
||||||
//go:linkname Send C.uv_udp_send
|
|
||||||
func Send(req *UdpSend, udp *Udp, bufs *Buf, nbufs c.Uint, addr *net.SockAddr, sendCb UdpSendCb) c.Int
|
|
||||||
|
|
||||||
// llgo:link (*Udp).TrySend C.uv_udp_try_send
|
// llgo:link (*Udp).TrySend C.uv_udp_try_send
|
||||||
func (udp *Udp) TrySend(bufs *Buf, nbufs c.Uint, addr *net.SockAddr) c.Int {
|
func (udp *Udp) TrySend(bufs *Buf, nbufs c.Uint, addr *net.SockAddr) c.Int {
|
||||||
return 0
|
return 0
|
||||||
@@ -462,8 +486,13 @@ func (udp *Udp) GetSendQueueCount() uintptr {
|
|||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//go:linkname Send C.uv_udp_send
|
||||||
|
func Send(req *UdpSend, udp *Udp, bufs *Buf, nbufs c.Uint, addr *net.SockAddr, sendCb UdpSendCb) c.Int
|
||||||
|
|
||||||
// ----------------------------------------------
|
// ----------------------------------------------
|
||||||
|
|
||||||
|
/* DNS related function and method */
|
||||||
|
|
||||||
//go:linkname Ip4Addr C.uv_ip4_addr
|
//go:linkname Ip4Addr C.uv_ip4_addr
|
||||||
func Ip4Addr(ip *c.Char, port c.Int, addr *net.SockaddrIn) c.Int
|
func Ip4Addr(ip *c.Char, port c.Int, addr *net.SockaddrIn) c.Int
|
||||||
|
|
||||||
|
|||||||
@@ -26,8 +26,17 @@ type SignalCb func(handle *Signal, sigNum c.Int)
|
|||||||
//go:linkname SignalInit C.uv_signal_init
|
//go:linkname SignalInit C.uv_signal_init
|
||||||
func SignalInit(loop *Loop, handle *Signal) c.Int
|
func SignalInit(loop *Loop, handle *Signal) c.Int
|
||||||
|
|
||||||
//go:linkname SignalStart C.uv_signal_start
|
// llgo:link (*Signal).Start C.uv_signal_start
|
||||||
func SignalStart(handle *Signal, cb SignalCb, signum c.Int) c.Int
|
func (handle *Signal) Start(cb SignalCb, signum c.Int) c.Int {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
//go:linkname SignalStartOneshot C.uv_signal_start_oneshot
|
// llgo:link (*Signal).StartOneshot C.uv_signal_start_oneshot
|
||||||
func SignalStartOneshot(handle *Signal, cb SignalCb, signum c.Int) c.Int
|
func (handle *Signal) StartOneshot(cb SignalCb, signum c.Int) c.Int {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
// llgo:link (*Signal).Stop C.uv_signal_stop
|
||||||
|
func (handle *Signal) Stop() c.Int {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|||||||
39
c/net/net.go
39
c/net/net.go
@@ -17,7 +17,7 @@
|
|||||||
package net
|
package net
|
||||||
|
|
||||||
import (
|
import (
|
||||||
_ "unsafe"
|
"unsafe"
|
||||||
|
|
||||||
"github.com/goplus/llgo/c"
|
"github.com/goplus/llgo/c"
|
||||||
)
|
)
|
||||||
@@ -97,6 +97,13 @@ const (
|
|||||||
EAI_OVERFLOW /* argument buffer overflow */
|
EAI_OVERFLOW /* argument buffer overflow */
|
||||||
)
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
ALIGNSIZE = unsafe.Sizeof(c.LongLong(0))
|
||||||
|
MAXSIZE = 128
|
||||||
|
PAD1_SIZE = ALIGNSIZE - unsafe.Sizeof(byte(0)) - unsafe.Sizeof(byte(0))
|
||||||
|
PAD2_SIZE = MAXSIZE - unsafe.Sizeof(byte(0)) - unsafe.Sizeof(byte(0)) - PAD1_SIZE - ALIGNSIZE
|
||||||
|
)
|
||||||
|
|
||||||
// (TODO) merge to inet
|
// (TODO) merge to inet
|
||||||
const INET_ADDRSTRLEN = 16
|
const INET_ADDRSTRLEN = 16
|
||||||
|
|
||||||
@@ -117,6 +124,14 @@ type SockaddrIn6 struct {
|
|||||||
ScopeId c.Uint
|
ScopeId c.Uint
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type SockaddrStorage struct {
|
||||||
|
Len uint8
|
||||||
|
Family uint8
|
||||||
|
pad1 [PAD1_SIZE]c.Char
|
||||||
|
align c.LongLong
|
||||||
|
pad2 [PAD2_SIZE]c.Char
|
||||||
|
}
|
||||||
|
|
||||||
type InAddr struct {
|
type InAddr struct {
|
||||||
Addr c.Uint
|
Addr c.Uint
|
||||||
}
|
}
|
||||||
@@ -171,6 +186,9 @@ func Send(c.Int, c.Pointer, uintptr, c.Int) c.Long
|
|||||||
//go:linkname Recv C.recv
|
//go:linkname Recv C.recv
|
||||||
func Recv(c.Int, c.Pointer, uintptr, c.Int) c.Long
|
func Recv(c.Int, c.Pointer, uintptr, c.Int) c.Long
|
||||||
|
|
||||||
|
//go:linkname SetSockOpt C.setsockopt
|
||||||
|
func SetSockOpt(socket c.Int, level c.Int, optionName c.Int, optionValue c.Pointer, sockLen c.Uint) c.Int
|
||||||
|
|
||||||
// -----------------------------------------------------------------------------
|
// -----------------------------------------------------------------------------
|
||||||
|
|
||||||
type AddrInfo struct {
|
type AddrInfo struct {
|
||||||
@@ -196,8 +214,27 @@ func swapInt16(data uint16) uint16 {
|
|||||||
return (data << 8) | (data >> 8)
|
return (data << 8) | (data >> 8)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func swapInt32(data c.Uint) c.Uint {
|
||||||
|
return ((data & 0xff) << 24) |
|
||||||
|
((data & 0xff00) << 8) |
|
||||||
|
((data & 0xff0000) >> 8) |
|
||||||
|
((data & 0xff000000) >> 24)
|
||||||
|
}
|
||||||
|
|
||||||
func Htons(x uint16) uint16 {
|
func Htons(x uint16) uint16 {
|
||||||
return swapInt16(x)
|
return swapInt16(x)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func Ntohs(x uint16) uint16 {
|
||||||
|
return swapInt16(x)
|
||||||
|
}
|
||||||
|
|
||||||
|
func Htonl(x c.Uint) c.Uint {
|
||||||
|
return swapInt32(x)
|
||||||
|
}
|
||||||
|
|
||||||
|
func Ntohl(x c.Uint) c.Uint {
|
||||||
|
return swapInt32(x)
|
||||||
|
}
|
||||||
|
|
||||||
// -----------------------------------------------------------------------------
|
// -----------------------------------------------------------------------------
|
||||||
|
|||||||
@@ -21,7 +21,6 @@ func main() {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
defer ctx.Free()
|
defer ctx.Free()
|
||||||
|
|
||||||
var ret c.Int = ctx.InitEx(unsafe.Pointer(unsafe.StringData(key)), c.Int(lenKey), openssl.EVP_sha256(), nil)
|
var ret c.Int = ctx.InitEx(unsafe.Pointer(unsafe.StringData(key)), c.Int(lenKey), openssl.EVP_sha256(), nil)
|
||||||
if ret == 0 {
|
if ret == 0 {
|
||||||
c.Fprintf(c.Stderr, c.Str("%s\n"), c.Str("Error initializing HMAC_CTX"))
|
c.Fprintf(c.Stderr, c.Str("%s\n"), c.Str("Error initializing HMAC_CTX"))
|
||||||
@@ -37,9 +36,5 @@ func main() {
|
|||||||
c.Fprintf(c.Stderr, c.Str("%s\n"), c.Str("Error finalizing HMAC_CTX"))
|
c.Fprintf(c.Stderr, c.Str("%s\n"), c.Str("Error finalizing HMAC_CTX"))
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
fmt.Print("HMAC: ")
|
fmt.Printf("HMAC:%x\n", digest[:digestLen])
|
||||||
for i := 0; i < int(digestLen); i++ {
|
|
||||||
fmt.Printf("%02x", digest[i])
|
|
||||||
}
|
|
||||||
fmt.Print("\n")
|
|
||||||
}
|
}
|
||||||
|
|||||||
102
c/openssl/bn.go
102
c/openssl/bn.go
@@ -49,8 +49,19 @@ func BN_CTXSecureNew() *BN_CTX
|
|||||||
func (*BN_CTX) Free() {}
|
func (*BN_CTX) Free() {}
|
||||||
|
|
||||||
// void BN_CTX_start(BN_CTX *ctx);
|
// void BN_CTX_start(BN_CTX *ctx);
|
||||||
|
//
|
||||||
|
// llgo:link (*BN_CTX).Start C.BN_CTX_start
|
||||||
|
func (*BN_CTX) Start() {}
|
||||||
|
|
||||||
// BIGNUM *BN_CTX_get(BN_CTX *ctx);
|
// BIGNUM *BN_CTX_get(BN_CTX *ctx);
|
||||||
|
//
|
||||||
|
// llgo:link (*BN_CTX).Get C.BN_CTX_get
|
||||||
|
func (*BN_CTX) Get() *BIGNUM { return nil }
|
||||||
|
|
||||||
// void BN_CTX_end(BN_CTX *ctx);
|
// void BN_CTX_end(BN_CTX *ctx);
|
||||||
|
//
|
||||||
|
// llgo:link (*BN_CTX).End C.BN_CTX_end
|
||||||
|
func (*BN_CTX) End() {}
|
||||||
|
|
||||||
// -----------------------------------------------------------------------------
|
// -----------------------------------------------------------------------------
|
||||||
|
|
||||||
@@ -191,23 +202,80 @@ func BNDec2bn(a **BIGNUM, str *c.Char) c.Int
|
|||||||
//go:linkname BNAsc2bn C.BN_asc2bn
|
//go:linkname BNAsc2bn C.BN_asc2bn
|
||||||
func BNAsc2bn(a **BIGNUM, str *c.Char) c.Int
|
func BNAsc2bn(a **BIGNUM, str *c.Char) c.Int
|
||||||
|
|
||||||
/*
|
// BIGNUM *BN_bin2bn(const unsigned char *s, int len, BIGNUM *ret);
|
||||||
BIGNUM *BN_bin2bn(const unsigned char *s, int len, BIGNUM *ret);
|
//
|
||||||
BIGNUM *BN_signed_bin2bn(const unsigned char *s, int len, BIGNUM *ret);
|
//go:linkname BNBin2bn C.BN_bin2bn
|
||||||
int BN_bn2bin(const BIGNUM *a, unsigned char *to);
|
func BNBin2bn(s *byte, len c.Int, ret *BIGNUM) *BIGNUM
|
||||||
int BN_bn2binpad(const BIGNUM *a, unsigned char *to, int tolen);
|
|
||||||
int BN_signed_bn2bin(const BIGNUM *a, unsigned char *to, int tolen);
|
// BIGNUM *BN_signed_bin2bn(const unsigned char *s, int len, BIGNUM *ret);
|
||||||
BIGNUM *BN_lebin2bn(const unsigned char *s, int len, BIGNUM *ret);
|
//
|
||||||
BIGNUM *BN_signed_lebin2bn(const unsigned char *s, int len, BIGNUM *ret);
|
//go:linkname BNSignedBin2bn C.BN_signed_bin2bn
|
||||||
int BN_bn2lebinpad(const BIGNUM *a, unsigned char *to, int tolen);
|
func BNSignedBin2bn(s *byte, len c.Int, ret *BIGNUM) *BIGNUM
|
||||||
int BN_signed_bn2lebin(const BIGNUM *a, unsigned char *to, int tolen);
|
|
||||||
BIGNUM *BN_native2bn(const unsigned char *s, int len, BIGNUM *ret);
|
// int BN_bn2bin(const BIGNUM *a, unsigned char *to);
|
||||||
BIGNUM *BN_signed_native2bn(const unsigned char *s, int len, BIGNUM *ret);
|
//
|
||||||
int BN_bn2nativepad(const BIGNUM *a, unsigned char *to, int tolen);
|
// llgo:link (*BIGNUM).Bn2bin C.BN_bn2bin
|
||||||
int BN_signed_bn2native(const BIGNUM *a, unsigned char *to, int tolen);
|
func (bn *BIGNUM) Bn2bin(to *byte) c.Int { return 0 }
|
||||||
BIGNUM *BN_mpi2bn(const unsigned char *s, int len, BIGNUM *ret);
|
|
||||||
int BN_bn2mpi(const BIGNUM *a, unsigned char *to);
|
// int BN_bn2binpad(const BIGNUM *a, unsigned char *to, int tolen);
|
||||||
*/
|
//
|
||||||
|
// llgo:link (*BIGNUM).Bn2binpad C.BN_bn2binpad
|
||||||
|
func (bn *BIGNUM) Bn2binpad(to *byte, tolen c.Int) c.Int { return 0 }
|
||||||
|
|
||||||
|
// int BN_signed_bn2bin(const BIGNUM *a, unsigned char *to, int tolen);
|
||||||
|
//
|
||||||
|
// llgo:link (*BIGNUM).SignedBn2bin C.BN_signed_bn2bin
|
||||||
|
func (bn *BIGNUM) SignedBn2bin(to *byte, tolen c.Int) c.Int { return 0 }
|
||||||
|
|
||||||
|
// BIGNUM *BN_lebin2bn(const unsigned char *s, int len, BIGNUM *ret);
|
||||||
|
//
|
||||||
|
//go:linkname BNLebin2bn C.BN_lebin2bn
|
||||||
|
func BNLebin2bn(s *byte, len c.Int, ret *BIGNUM) *BIGNUM
|
||||||
|
|
||||||
|
// BIGNUM *BN_signed_lebin2bn(const unsigned char *s, int len, BIGNUM *ret);
|
||||||
|
//
|
||||||
|
//go:linkname BNSignedLebin2bn C.BN_signed_lebin2bn
|
||||||
|
func BNSignedLebin2bn(s *byte, len c.Int, ret *BIGNUM) *BIGNUM
|
||||||
|
|
||||||
|
// int BN_bn2lebinpad(const BIGNUM *a, unsigned char *to, int tolen);
|
||||||
|
//
|
||||||
|
// llgo:link (*BIGNUM).Bn2lebinpad C.BN_bn2lebinpad
|
||||||
|
func (bn *BIGNUM) Bn2lebinpad(to *byte, tolen c.Int) c.Int { return 0 }
|
||||||
|
|
||||||
|
// int BN_signed_bn2lebin(const BIGNUM *a, unsigned char *to, int tolen);
|
||||||
|
//
|
||||||
|
// llgo:link (*BIGNUM).SignedBn2lebin C.BN_signed_bn2lebin
|
||||||
|
func (bn *BIGNUM) SignedBn2lebin(to *byte, tolen c.Int) c.Int { return 0 }
|
||||||
|
|
||||||
|
// BIGNUM *BN_native2bn(const unsigned char *s, int len, BIGNUM *ret);
|
||||||
|
//
|
||||||
|
//go:linkname BNNative2bn C.BN_native2bn
|
||||||
|
func BNNative2bn(s *byte, len c.Int, ret *BIGNUM) *BIGNUM
|
||||||
|
|
||||||
|
// BIGNUM *BN_signed_native2bn(const unsigned char *s, int len, BIGNUM *ret);
|
||||||
|
//
|
||||||
|
//go:linkname BNSignedNative2bn C.BN_signed_native2bn
|
||||||
|
func BNSignedNative2bn(s *byte, len c.Int, ret *BIGNUM) *BIGNUM
|
||||||
|
|
||||||
|
// int BN_bn2nativepad(const BIGNUM *a, unsigned char *to, int tolen);
|
||||||
|
//
|
||||||
|
// llgo:link (*BIGNUM).Bn2nativepad C.BN_bn2nativepad
|
||||||
|
func (bn *BIGNUM) Bn2nativepad(to *byte, tolen c.Int) c.Int { return 0 }
|
||||||
|
|
||||||
|
// int BN_signed_bn2native(const BIGNUM *a, unsigned char *to, int tolen);
|
||||||
|
//
|
||||||
|
// llgo:link (*BIGNUM).SignedBn2native C.BN_signed_bn2native
|
||||||
|
func (bn *BIGNUM) SignedBn2native(to *byte, tolen c.Int) c.Int { return 0 }
|
||||||
|
|
||||||
|
// BIGNUM *BN_mpi2bn(const unsigned char *s, int len, BIGNUM *ret);
|
||||||
|
//
|
||||||
|
//go:linkname BNMpi2bn C.BN_mpi2bn
|
||||||
|
func BNMpi2bn(s *byte, len c.Int, ret *BIGNUM) *BIGNUM
|
||||||
|
|
||||||
|
// int BN_bn2mpi(const BIGNUM *a, unsigned char *to);
|
||||||
|
//
|
||||||
|
// llgo:link (*BIGNUM).Bn2mpi C.BN_bn2mpi
|
||||||
|
func (bn *BIGNUM) Bn2mpi(to *byte) c.Int { return 0 }
|
||||||
|
|
||||||
// int BN_sub(BIGNUM *r, const BIGNUM *a, const BIGNUM *b);
|
// int BN_sub(BIGNUM *r, const BIGNUM *a, const BIGNUM *b);
|
||||||
//
|
//
|
||||||
|
|||||||
@@ -6,7 +6,15 @@ import (
|
|||||||
"github.com/goplus/llgo/c"
|
"github.com/goplus/llgo/c"
|
||||||
)
|
)
|
||||||
|
|
||||||
const EVP_MAX_MD_SIZE = 64 /* longest known is SHA512 */
|
const (
|
||||||
|
EVP_MAX_MD_SIZE = 64 /* longest known is SHA512 */
|
||||||
|
)
|
||||||
|
|
||||||
|
// -----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
type EVP_MD struct {
|
||||||
|
Unused [0]byte
|
||||||
|
}
|
||||||
|
|
||||||
// const EVP_MD *EVP_sha1(void)
|
// const EVP_MD *EVP_sha1(void)
|
||||||
//
|
//
|
||||||
@@ -43,13 +51,7 @@ func EVP_sha384() *EVP_MD
|
|||||||
//go:linkname EVP_sha512 C.EVP_sha512
|
//go:linkname EVP_sha512 C.EVP_sha512
|
||||||
func EVP_sha512() *EVP_MD
|
func EVP_sha512() *EVP_MD
|
||||||
|
|
||||||
type EVP_MD struct {
|
// -----------------------------------------------------------------------------
|
||||||
Unused [0]byte
|
|
||||||
}
|
|
||||||
|
|
||||||
type EVP_MD_CTX struct {
|
|
||||||
Unused [0]byte
|
|
||||||
}
|
|
||||||
|
|
||||||
type HMAC_CTX struct {
|
type HMAC_CTX struct {
|
||||||
Unused [0]byte
|
Unused [0]byte
|
||||||
@@ -63,30 +65,38 @@ func NewHMAC_CTX() *HMAC_CTX
|
|||||||
// OSSL_DEPRECATEDIN_3_0 void HMAC_CTX_free(HMAC_CTX *ctx);
|
// OSSL_DEPRECATEDIN_3_0 void HMAC_CTX_free(HMAC_CTX *ctx);
|
||||||
//
|
//
|
||||||
// llgo:link (*HMAC_CTX).Free C.HMAC_CTX_free
|
// llgo:link (*HMAC_CTX).Free C.HMAC_CTX_free
|
||||||
func (c *HMAC_CTX) Free() {}
|
func (ctx *HMAC_CTX) Free() {}
|
||||||
|
|
||||||
// OSSL_DEPRECATEDIN_3_0 size_t HMAC_size(const HMAC_CTX *e);
|
// OSSL_DEPRECATEDIN_3_0 size_t HMAC_size(const HMAC_CTX *e);
|
||||||
//
|
//
|
||||||
// llgo:link (*HMAC_CTX).Size C.HMAC_size
|
// llgo:link (*HMAC_CTX).Size C.HMAC_size
|
||||||
func (c *HMAC_CTX) Size() uintptr { return 0 }
|
func (ctx *HMAC_CTX) Size() uintptr { return 0 }
|
||||||
|
|
||||||
// OSSL_DEPRECATEDIN_3_0 int HMAC_CTX_reset(HMAC_CTX *ctx);
|
// OSSL_DEPRECATEDIN_3_0 int HMAC_CTX_reset(HMAC_CTX *ctx);
|
||||||
//
|
//
|
||||||
// llgo:link (*HMAC_CTX).Reset C.HMAC_CTX_reset
|
// llgo:link (*HMAC_CTX).Reset C.HMAC_CTX_reset
|
||||||
func (c *HMAC_CTX) Reset() c.Int { return 0 }
|
func (ctx *HMAC_CTX) Reset() c.Int { return 0 }
|
||||||
|
|
||||||
// OSSL_DEPRECATEDIN_1_1_0 __owur int HMAC_Init(HMAC_CTX *ctx,
|
// OSSL_DEPRECATEDIN_1_1_0 __owur int HMAC_Init(HMAC_CTX *ctx,
|
||||||
// const void *key, int len,
|
// const void *key, int len,
|
||||||
// const EVP_MD *md);
|
// const EVP_MD *md);
|
||||||
//
|
//
|
||||||
// llgo:link (*HMAC_CTX).Init C.HMAC_Init
|
// llgo:link (*HMAC_CTX).Init C.HMAC_Init
|
||||||
func (c *HMAC_CTX) Init(key unsafe.Pointer, len c.Int, md *EVP_MD) c.Int { return 0 }
|
func (ctx *HMAC_CTX) Init(key unsafe.Pointer, len c.Int, md *EVP_MD) c.Int { return 0 }
|
||||||
|
|
||||||
|
func (ctx *HMAC_CTX) InitBytes(key []byte, md *EVP_MD) c.Int {
|
||||||
|
return ctx.Init(unsafe.Pointer(unsafe.SliceData(key)), c.Int(len(key)), md)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ctx *HMAC_CTX) InitString(key string, md *EVP_MD) c.Int {
|
||||||
|
return ctx.Init(unsafe.Pointer(unsafe.StringData(key)), c.Int(len(key)), md)
|
||||||
|
}
|
||||||
|
|
||||||
// OSSL_DEPRECATEDIN_3_0 int HMAC_Init_ex(HMAC_CTX *ctx, const void *key, int len,
|
// OSSL_DEPRECATEDIN_3_0 int HMAC_Init_ex(HMAC_CTX *ctx, const void *key, int len,
|
||||||
// const EVP_MD *md, ENGINE *impl);
|
// const EVP_MD *md, ENGINE *impl);
|
||||||
//
|
//
|
||||||
// llgo:link (*HMAC_CTX).InitEx C.HMAC_Init_ex
|
// llgo:link (*HMAC_CTX).InitEx C.HMAC_Init_ex
|
||||||
func (c *HMAC_CTX) InitEx(key unsafe.Pointer, len c.Int, md *EVP_MD, impl unsafe.Pointer) c.Int {
|
func (ctx *HMAC_CTX) InitEx(key unsafe.Pointer, len c.Int, md *EVP_MD, impl unsafe.Pointer) c.Int {
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -94,28 +104,30 @@ func (c *HMAC_CTX) InitEx(key unsafe.Pointer, len c.Int, md *EVP_MD, impl unsafe
|
|||||||
// size_t len);
|
// size_t len);
|
||||||
//
|
//
|
||||||
// llgo:link (*HMAC_CTX).Update C.HMAC_Update
|
// llgo:link (*HMAC_CTX).Update C.HMAC_Update
|
||||||
func (c *HMAC_CTX) Update(data unsafe.Pointer, len uintptr) c.Int { return 0 }
|
func (ctx *HMAC_CTX) Update(data unsafe.Pointer, len uintptr) c.Int { return 0 }
|
||||||
|
|
||||||
func (c *HMAC_CTX) UpdateBytes(data []byte) c.Int {
|
func (ctx *HMAC_CTX) UpdateBytes(data []byte) c.Int {
|
||||||
return c.Update(unsafe.Pointer(unsafe.SliceData(data)), uintptr(len(data)))
|
return ctx.Update(unsafe.Pointer(unsafe.SliceData(data)), uintptr(len(data)))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *HMAC_CTX) UpdateString(data string) c.Int {
|
func (ctx *HMAC_CTX) UpdateString(data string) c.Int {
|
||||||
return c.Update(unsafe.Pointer(unsafe.StringData(data)), uintptr(len(data)))
|
return ctx.Update(unsafe.Pointer(unsafe.StringData(data)), uintptr(len(data)))
|
||||||
}
|
}
|
||||||
|
|
||||||
// OSSL_DEPRECATEDIN_3_0 int HMAC_Final(HMAC_CTX *ctx, unsigned char *md,
|
// OSSL_DEPRECATEDIN_3_0 int HMAC_Final(HMAC_CTX *ctx, unsigned char *md,
|
||||||
// unsigned int *len);
|
// unsigned int *len);
|
||||||
//
|
//
|
||||||
// llgo:link (*HMAC_CTX).Final C.HMAC_Final
|
// llgo:link (*HMAC_CTX).Final C.HMAC_Final
|
||||||
func (c *HMAC_CTX) Final(md *byte, len *c.Uint) c.Int { return 0 }
|
func (ctx *HMAC_CTX) Final(md *byte, len *c.Uint) c.Int { return 0 }
|
||||||
|
|
||||||
// OSSL_DEPRECATEDIN_3_0 __owur int HMAC_CTX_copy(HMAC_CTX *dctx, HMAC_CTX *sctx);
|
// OSSL_DEPRECATEDIN_3_0 __owur int HMAC_CTX_copy(HMAC_CTX *dctx, HMAC_CTX *sctx);
|
||||||
//
|
//
|
||||||
// llgo:link (*HMAC_CTX).Copy C.HMAC_CTX_copy
|
// llgo:link (*HMAC_CTX).Copy C.HMAC_CTX_copy
|
||||||
func (c *HMAC_CTX) Copy(sctx *HMAC_CTX) c.Int { return 0 }
|
func (ctx *HMAC_CTX) Copy(sctx *HMAC_CTX) c.Int { return 0 }
|
||||||
|
|
||||||
// OSSL_DEPRECATEDIN_3_0 void HMAC_CTX_set_flags(HMAC_CTX *ctx, unsigned long flags);
|
// OSSL_DEPRECATEDIN_3_0 void HMAC_CTX_set_flags(HMAC_CTX *ctx, unsigned long flags);
|
||||||
//
|
//
|
||||||
// llgo:link (*HMAC_CTX).SetFlags C.HMAC_CTX_set_flags
|
// llgo:link (*HMAC_CTX).SetFlags C.HMAC_CTX_set_flags
|
||||||
func (c *HMAC_CTX) SetFlags(flags c.Ulong) {}
|
func (ctx *HMAC_CTX) SetFlags(flags c.Ulong) {}
|
||||||
|
|
||||||
|
// -----------------------------------------------------------------------------
|
||||||
|
|||||||
217
chore/_xtool/castdump/castdump.go
Normal file
217
chore/_xtool/castdump/castdump.go
Normal file
@@ -0,0 +1,217 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
"unsafe"
|
||||||
|
|
||||||
|
"github.com/goplus/llgo/c"
|
||||||
|
"github.com/goplus/llgo/c/clang"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Data struct {
|
||||||
|
Depth c.Uint
|
||||||
|
Unit *clang.TranslationUnit
|
||||||
|
}
|
||||||
|
|
||||||
|
var accessMap = map[clang.CXXAccessSpecifier]string{
|
||||||
|
clang.CXXInvalidAccessSpecifier: "invalid",
|
||||||
|
clang.CXXPublic: "public",
|
||||||
|
clang.CXXProtected: "protected",
|
||||||
|
clang.CXXPrivate: "private",
|
||||||
|
}
|
||||||
|
|
||||||
|
func printIndent(depth c.Uint) {
|
||||||
|
fmt.Print(strings.Repeat(" ", int(depth)))
|
||||||
|
}
|
||||||
|
|
||||||
|
func accessToString(spec clang.CXXAccessSpecifier) string {
|
||||||
|
if str, ok := accessMap[spec]; ok {
|
||||||
|
return str
|
||||||
|
}
|
||||||
|
return "unknown"
|
||||||
|
}
|
||||||
|
|
||||||
|
func visit(cursor, parent clang.Cursor, ClientData c.Pointer) clang.ChildVisitResult {
|
||||||
|
data := (*Data)(ClientData)
|
||||||
|
printAST(cursor, data)
|
||||||
|
return clang.ChildVisit_Continue
|
||||||
|
}
|
||||||
|
|
||||||
|
func printType(t clang.Type, data *Data) {
|
||||||
|
printIndent(data.Depth)
|
||||||
|
|
||||||
|
typeSpell := t.String()
|
||||||
|
typeKind := t.Kind.String()
|
||||||
|
|
||||||
|
if t.Kind == clang.TypeInvalid {
|
||||||
|
} else if t.Kind == clang.TypeUnexposed {
|
||||||
|
c.Printf(c.Str("<UnexposedType|%s>: %s\n"), typeKind.CStr(), typeSpell.CStr())
|
||||||
|
} else if t.Kind >= clang.TypeFirstBuiltin && t.Kind <= clang.TypeLastBuiltin {
|
||||||
|
c.Printf(c.Str("<BuiltinType|%s>: %s\n"), typeKind.CStr(), typeSpell.CStr())
|
||||||
|
} else if t.Kind > clang.TypeComplex {
|
||||||
|
c.Printf(c.Str("<ComplexType|%s>: %s\n"), typeKind.CStr(), typeSpell.CStr())
|
||||||
|
}
|
||||||
|
|
||||||
|
data.Depth++
|
||||||
|
switch t.Kind {
|
||||||
|
case clang.TypePointer:
|
||||||
|
printType(t.PointeeType(), data)
|
||||||
|
case clang.TypeIncompleteArray, clang.TypeVariableArray, clang.TypeDependentSizedArray, clang.TypeConstantArray:
|
||||||
|
printType(t.ArrayElementType(), data)
|
||||||
|
case clang.TypeTypedef:
|
||||||
|
printType(t.CanonicalType(), data)
|
||||||
|
case clang.TypeFunctionProto:
|
||||||
|
printType(t.ResultType(), data)
|
||||||
|
for i := 0; i < int(t.NumArgTypes()); i++ {
|
||||||
|
printType(t.ArgType(c.Uint(i)), data)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
data.Depth--
|
||||||
|
|
||||||
|
typeKind.Dispose()
|
||||||
|
typeSpell.Dispose()
|
||||||
|
}
|
||||||
|
|
||||||
|
func printLocation(cursor clang.Cursor) {
|
||||||
|
loc := cursor.Location()
|
||||||
|
var file clang.File
|
||||||
|
var line, column c.Uint
|
||||||
|
|
||||||
|
loc.SpellingLocation(&file, &line, &column, nil)
|
||||||
|
filename := file.FileName()
|
||||||
|
defer filename.Dispose()
|
||||||
|
|
||||||
|
c.Printf(c.Str("(Loc:%s:%d:%d)\n"), filename.CStr(), line, column)
|
||||||
|
}
|
||||||
|
|
||||||
|
func printAccess(cursor clang.Cursor) {
|
||||||
|
kind := cursor.Kind.String()
|
||||||
|
spell := cursor.String()
|
||||||
|
defer kind.Dispose()
|
||||||
|
defer spell.Dispose()
|
||||||
|
|
||||||
|
c.Printf(c.Str("%s: %s %s"), kind.CStr(), spell.CStr(), c.AllocaCStr(accessToString(cursor.CXXAccessSpecifier())))
|
||||||
|
printLocation(cursor)
|
||||||
|
}
|
||||||
|
|
||||||
|
func printMacro(cursor clang.Cursor, unit *clang.TranslationUnit) {
|
||||||
|
kind := cursor.Kind.String()
|
||||||
|
defer kind.Dispose()
|
||||||
|
|
||||||
|
c.Printf(c.Str("%s: "), kind.CStr())
|
||||||
|
ran := cursor.Extent()
|
||||||
|
var numTokens c.Uint
|
||||||
|
var tokens *clang.Token
|
||||||
|
unit.Tokenize(ran, &tokens, &numTokens)
|
||||||
|
tokensSlice := unsafe.Slice(tokens, int(numTokens))
|
||||||
|
for _, tok := range tokensSlice {
|
||||||
|
tokStr := unit.Token(tok)
|
||||||
|
c.Printf(c.Str("%s "), tokStr.CStr())
|
||||||
|
tokStr.Dispose()
|
||||||
|
}
|
||||||
|
printLocation(cursor)
|
||||||
|
}
|
||||||
|
|
||||||
|
func printFunc(cursor clang.Cursor, data *Data) {
|
||||||
|
kind := cursor.Kind.String()
|
||||||
|
spell := cursor.String()
|
||||||
|
symbol := cursor.Mangling()
|
||||||
|
defer symbol.Dispose()
|
||||||
|
defer kind.Dispose()
|
||||||
|
defer spell.Dispose()
|
||||||
|
|
||||||
|
c.Printf(c.Str("%s: %s (Symbol: %s)"), kind.CStr(), spell.CStr(), symbol.CStr())
|
||||||
|
printLocation(cursor)
|
||||||
|
printType(cursor.Type(), data)
|
||||||
|
}
|
||||||
|
|
||||||
|
func printEnumConstant(cursor clang.Cursor) {
|
||||||
|
kind := cursor.Kind.String()
|
||||||
|
spell := cursor.String()
|
||||||
|
defer kind.Dispose()
|
||||||
|
defer spell.Dispose()
|
||||||
|
|
||||||
|
c.Printf(c.Str("%s: %s:%lld"), kind.CStr(), spell.CStr(), cursor.EnumConstantDeclValue())
|
||||||
|
printLocation(cursor)
|
||||||
|
}
|
||||||
|
|
||||||
|
func printDefault(cursor clang.Cursor, data *Data) {
|
||||||
|
kind := cursor.Kind.String()
|
||||||
|
spell := cursor.String()
|
||||||
|
defer kind.Dispose()
|
||||||
|
defer spell.Dispose()
|
||||||
|
|
||||||
|
// node which has type
|
||||||
|
if cursor.Type().Kind != clang.TypeInvalid {
|
||||||
|
c.Printf(c.Str("%s: %s"), kind.CStr(), spell.CStr())
|
||||||
|
printLocation(cursor)
|
||||||
|
printType(cursor.Type(), data)
|
||||||
|
} else {
|
||||||
|
c.Printf(c.Str("%s: %s\n"), kind.CStr(), spell.CStr())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func printAST(cursor clang.Cursor, data *Data) {
|
||||||
|
kind := cursor.Kind.String()
|
||||||
|
spell := cursor.String()
|
||||||
|
|
||||||
|
printIndent(data.Depth)
|
||||||
|
|
||||||
|
switch cursor.Kind {
|
||||||
|
case clang.CursorCXXAccessSpecifier:
|
||||||
|
printAccess(cursor)
|
||||||
|
case clang.CursorMacroDefinition:
|
||||||
|
printMacro(cursor, data.Unit)
|
||||||
|
case clang.CursorFunctionDecl, clang.CursorCXXMethod, clang.CursorConstructor, clang.CursorDestructor:
|
||||||
|
printFunc(cursor, data)
|
||||||
|
case clang.CursorEnumConstantDecl:
|
||||||
|
printEnumConstant(cursor)
|
||||||
|
default:
|
||||||
|
printDefault(cursor, data)
|
||||||
|
}
|
||||||
|
|
||||||
|
data.Depth++
|
||||||
|
clang.VisitChildren(cursor, visit, c.Pointer(data))
|
||||||
|
data.Depth--
|
||||||
|
|
||||||
|
kind.Dispose()
|
||||||
|
spell.Dispose()
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
if c.Argc != 2 {
|
||||||
|
fmt.Fprintln(os.Stderr, "Usage: castdump <headerFile>")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
args := make([]*c.Char, 3)
|
||||||
|
args[0] = c.Str("-x")
|
||||||
|
args[1] = c.Str("c++")
|
||||||
|
args[2] = c.Str("-std=c++11")
|
||||||
|
|
||||||
|
sourceFile := *c.Advance(c.Argv, 1)
|
||||||
|
index := clang.CreateIndex(0, 0)
|
||||||
|
unit := index.ParseTranslationUnit(
|
||||||
|
sourceFile,
|
||||||
|
unsafe.SliceData(args), 3,
|
||||||
|
nil, 0,
|
||||||
|
clang.DetailedPreprocessingRecord,
|
||||||
|
)
|
||||||
|
defer index.Dispose()
|
||||||
|
defer unit.Dispose()
|
||||||
|
|
||||||
|
if unit == nil {
|
||||||
|
println("Unable to parse translation unit. Quitting.")
|
||||||
|
c.Exit(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
cursor := unit.Cursor()
|
||||||
|
|
||||||
|
Data := &Data{
|
||||||
|
Depth: 0,
|
||||||
|
Unit: unit,
|
||||||
|
}
|
||||||
|
printAST(cursor, Data)
|
||||||
|
}
|
||||||
@@ -22,7 +22,6 @@ import (
|
|||||||
"io"
|
"io"
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"strconv"
|
|
||||||
"strings"
|
"strings"
|
||||||
"unsafe"
|
"unsafe"
|
||||||
|
|
||||||
@@ -61,11 +60,11 @@ func main() {
|
|||||||
|
|
||||||
check(err)
|
check(err)
|
||||||
|
|
||||||
filepaths := generateHeaderFilePath(conf.CFlags, conf.Include)
|
filepaths := genHeaderFilePath(conf.CFlags, conf.Include)
|
||||||
astInfos, err := parse.ParseHeaderFile(filepaths)
|
headerInfos, err := parse.ParseHeaderFile(filepaths, conf.TrimPrefixes)
|
||||||
check(err)
|
check(err)
|
||||||
|
|
||||||
symbolInfo := getCommonSymbols(symbols, astInfos, conf.TrimPrefixes)
|
symbolInfo := getCommonSymbols(symbols, headerInfos, conf.TrimPrefixes)
|
||||||
|
|
||||||
err = genSymbolTableFile(symbolInfo)
|
err = genSymbolTableFile(symbolInfo)
|
||||||
check(err)
|
check(err)
|
||||||
@@ -77,8 +76,8 @@ func check(err error) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func parseDylibSymbols(lib string) ([]types.CPPSymbol, error) {
|
func parseDylibSymbols(lib string) ([]*nm.Symbol, error) {
|
||||||
dylibPath, err := generateDylibPath(lib)
|
dylibPath, err := genDylibPath(lib)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, errors.New("failed to generate dylib path")
|
return nil, errors.New("failed to generate dylib path")
|
||||||
}
|
}
|
||||||
@@ -88,22 +87,15 @@ func parseDylibSymbols(lib string) ([]types.CPPSymbol, error) {
|
|||||||
return nil, errors.New("failed to list symbols in dylib")
|
return nil, errors.New("failed to list symbols in dylib")
|
||||||
}
|
}
|
||||||
|
|
||||||
var symbols []types.CPPSymbol
|
var symbols []*nm.Symbol
|
||||||
|
|
||||||
for _, file := range files {
|
for _, file := range files {
|
||||||
for _, sym := range file.Symbols {
|
symbols = append(symbols, file.Symbols...)
|
||||||
demangleName := decodeSymbolName(sym.Name)
|
|
||||||
symbols = append(symbols, types.CPPSymbol{
|
|
||||||
Symbol: sym,
|
|
||||||
DemangleName: demangleName,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return symbols, nil
|
return symbols, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func generateDylibPath(lib string) (string, error) {
|
func genDylibPath(lib string) (string, error) {
|
||||||
output := lib
|
output := lib
|
||||||
libPath := ""
|
libPath := ""
|
||||||
libName := ""
|
libName := ""
|
||||||
@@ -123,31 +115,20 @@ func generateDylibPath(lib string) (string, error) {
|
|||||||
return dylibPath, nil
|
return dylibPath, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func decodeSymbolName(symbolName string) string {
|
func decodeSymbol(symbolName string) string {
|
||||||
if symbolName == "" {
|
if symbolName == "" {
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
|
|
||||||
demangled := llvm.ItaniumDemangle(symbolName, true)
|
demangled := llvm.ItaniumDemangle(symbolName, true)
|
||||||
if demangled == nil {
|
if demangled == nil {
|
||||||
return symbolName
|
return symbolName
|
||||||
}
|
}
|
||||||
defer c.Free(unsafe.Pointer(demangled))
|
defer c.Free(unsafe.Pointer(demangled))
|
||||||
|
|
||||||
demangleName := c.GoString(demangled)
|
demangleName := c.GoString(demangled)
|
||||||
if demangleName == "" {
|
return strings.TrimSpace(demangleName)
|
||||||
return symbolName
|
|
||||||
}
|
|
||||||
|
|
||||||
decodedName := strings.TrimSpace(demangleName)
|
|
||||||
decodedName = strings.ReplaceAll(decodedName,
|
|
||||||
"std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> > const",
|
|
||||||
"std::string")
|
|
||||||
|
|
||||||
return decodedName
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func generateHeaderFilePath(cflags string, files []string) []string {
|
func genHeaderFilePath(cflags string, files []string) []string {
|
||||||
prefixPath := cflags
|
prefixPath := cflags
|
||||||
prefixPath = strings.TrimPrefix(prefixPath, "-I")
|
prefixPath = strings.TrimPrefix(prefixPath, "-I")
|
||||||
var includePaths []string
|
var includePaths []string
|
||||||
@@ -157,77 +138,23 @@ func generateHeaderFilePath(cflags string, files []string) []string {
|
|||||||
return includePaths
|
return includePaths
|
||||||
}
|
}
|
||||||
|
|
||||||
func getCommonSymbols(dylibSymbols []types.CPPSymbol, astInfoList []types.ASTInformation, prefix []string) []types.SymbolInfo {
|
func getCommonSymbols(dylibSymbols []*nm.Symbol, symbolMap map[string]string, prefix []string) []*types.SymbolInfo {
|
||||||
var commonSymbols []types.SymbolInfo
|
var commonSymbols []*types.SymbolInfo
|
||||||
functionNameMap := make(map[string]int)
|
|
||||||
|
|
||||||
for _, astInfo := range astInfoList {
|
|
||||||
for _, dylibSym := range dylibSymbols {
|
for _, dylibSym := range dylibSymbols {
|
||||||
if strings.TrimPrefix(dylibSym.Name, "_") == astInfo.Symbol {
|
symName := strings.TrimPrefix(dylibSym.Name, "_")
|
||||||
cppName := generateCPPName(astInfo)
|
if goName, ok := symbolMap[symName]; ok {
|
||||||
functionNameMap[cppName]++
|
symbolInfo := &types.SymbolInfo{
|
||||||
symbolInfo := types.SymbolInfo{
|
Mangle: symName,
|
||||||
Mangle: strings.TrimPrefix(dylibSym.Name, "_"),
|
CPP: decodeSymbol(dylibSym.Name),
|
||||||
CPP: cppName,
|
Go: goName,
|
||||||
Go: generateMangle(astInfo, functionNameMap[cppName], prefix),
|
|
||||||
}
|
}
|
||||||
commonSymbols = append(commonSymbols, symbolInfo)
|
commonSymbols = append(commonSymbols, symbolInfo)
|
||||||
break
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
return commonSymbols
|
return commonSymbols
|
||||||
}
|
}
|
||||||
|
|
||||||
func generateCPPName(astInfo types.ASTInformation) string {
|
func genSymbolTableFile(symbolInfos []*types.SymbolInfo) error {
|
||||||
cppName := astInfo.Name
|
|
||||||
if astInfo.Class != "" {
|
|
||||||
cppName = astInfo.Class + "::" + astInfo.Name
|
|
||||||
}
|
|
||||||
return cppName
|
|
||||||
}
|
|
||||||
|
|
||||||
func generateMangle(astInfo types.ASTInformation, count int, prefixes []string) string {
|
|
||||||
astInfo.Class = removePrefix(astInfo.Class, prefixes)
|
|
||||||
astInfo.Name = removePrefix(astInfo.Name, prefixes)
|
|
||||||
res := ""
|
|
||||||
if astInfo.Class != "" {
|
|
||||||
if astInfo.Class == astInfo.Name {
|
|
||||||
res = "(*" + astInfo.Class + ")." + "Init"
|
|
||||||
if count > 1 {
|
|
||||||
res += "__" + strconv.Itoa(count-1)
|
|
||||||
}
|
|
||||||
} else if astInfo.Name == "~"+astInfo.Class {
|
|
||||||
res = "(*" + astInfo.Class + ")." + "Dispose"
|
|
||||||
if count > 1 {
|
|
||||||
res += "__" + strconv.Itoa(count-1)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
res = "(*" + astInfo.Class + ")." + astInfo.Name
|
|
||||||
if count > 1 {
|
|
||||||
res += "__" + strconv.Itoa(count-1)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
res = astInfo.Name
|
|
||||||
if count > 1 {
|
|
||||||
res += "__" + strconv.Itoa(count-1)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return res
|
|
||||||
}
|
|
||||||
|
|
||||||
func removePrefix(str string, prefixes []string) string {
|
|
||||||
for _, prefix := range prefixes {
|
|
||||||
if strings.HasPrefix(str, prefix) {
|
|
||||||
return strings.TrimPrefix(str, prefix)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return str
|
|
||||||
}
|
|
||||||
|
|
||||||
func genSymbolTableFile(symbolInfos []types.SymbolInfo) error {
|
|
||||||
// keep open follow code block can run successfully
|
// keep open follow code block can run successfully
|
||||||
for i := range symbolInfos {
|
for i := range symbolInfos {
|
||||||
println("symbol", symbolInfos[i].Go)
|
println("symbol", symbolInfos[i].Go)
|
||||||
@@ -269,6 +196,7 @@ func genSymbolTableFile(symbolInfos []types.SymbolInfo) error {
|
|||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func readExistingSymbolTable(fileName string) (map[string]types.SymbolInfo, error) {
|
func readExistingSymbolTable(fileName string) (map[string]types.SymbolInfo, error) {
|
||||||
existingSymbols := make(map[string]types.SymbolInfo)
|
existingSymbols := make(map[string]types.SymbolInfo)
|
||||||
|
|
||||||
|
|||||||
@@ -3,23 +3,27 @@ package parse
|
|||||||
import (
|
import (
|
||||||
"errors"
|
"errors"
|
||||||
"strconv"
|
"strconv"
|
||||||
|
"strings"
|
||||||
"unsafe"
|
"unsafe"
|
||||||
|
|
||||||
"github.com/goplus/llgo/c"
|
"github.com/goplus/llgo/c"
|
||||||
"github.com/goplus/llgo/c/clang"
|
"github.com/goplus/llgo/c/clang"
|
||||||
"github.com/goplus/llgo/chore/llcppg/types"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type Context struct {
|
type Context struct {
|
||||||
namespaceName string
|
namespaceName string
|
||||||
className string
|
className string
|
||||||
astInfo []types.ASTInformation
|
prefixes []string
|
||||||
|
symbolMap map[string]string
|
||||||
currentFile string
|
currentFile string
|
||||||
|
nameCounts map[string]int
|
||||||
}
|
}
|
||||||
|
|
||||||
func newContext() *Context {
|
func newContext(prefixes []string) *Context {
|
||||||
return &Context{
|
return &Context{
|
||||||
astInfo: make([]types.ASTInformation, 0),
|
prefixes: prefixes,
|
||||||
|
symbolMap: make(map[string]string),
|
||||||
|
nameCounts: make(map[string]int),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -35,70 +39,83 @@ func (c *Context) setCurrentFile(filename string) {
|
|||||||
c.currentFile = filename
|
c.currentFile = filename
|
||||||
}
|
}
|
||||||
|
|
||||||
var context = newContext()
|
func (c *Context) removePrefix(str string) string {
|
||||||
|
for _, prefix := range c.prefixes {
|
||||||
|
if strings.HasPrefix(str, prefix) {
|
||||||
|
return strings.TrimPrefix(str, prefix)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return str
|
||||||
|
}
|
||||||
|
|
||||||
func collectFuncInfo(cursor clang.Cursor) types.ASTInformation {
|
func (c *Context) genGoName(name string) string {
|
||||||
|
class := c.removePrefix(c.className)
|
||||||
|
name = c.removePrefix(name)
|
||||||
|
|
||||||
info := types.ASTInformation{
|
var baseName string
|
||||||
Namespace: context.namespaceName,
|
if class == "" {
|
||||||
Class: context.className,
|
baseName = name
|
||||||
|
} else {
|
||||||
|
baseName = c.genMethodName(class, name)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return c.addSuffix(baseName)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Context) genMethodName(class, name string) string {
|
||||||
|
prefix := "(*" + class + ")."
|
||||||
|
if class == name {
|
||||||
|
return prefix + "Init"
|
||||||
|
}
|
||||||
|
if name == "~"+class {
|
||||||
|
return prefix + "Dispose"
|
||||||
|
}
|
||||||
|
return prefix + name
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Context) addSuffix(name string) string {
|
||||||
|
c.nameCounts[name]++
|
||||||
|
count := c.nameCounts[name]
|
||||||
|
if count > 1 {
|
||||||
|
return name + "__" + strconv.Itoa(count-1)
|
||||||
|
}
|
||||||
|
return name
|
||||||
|
}
|
||||||
|
|
||||||
|
var context = newContext([]string{})
|
||||||
|
|
||||||
|
func collectFuncInfo(cursor clang.Cursor) {
|
||||||
cursorStr := cursor.String()
|
cursorStr := cursor.String()
|
||||||
symbol := cursor.Mangling()
|
symbol := cursor.Mangling()
|
||||||
|
|
||||||
info.Name = c.GoString(cursorStr.CStr())
|
name := c.GoString(cursorStr.CStr())
|
||||||
|
symbolName := c.GoString(symbol.CStr())
|
||||||
info.Symbol = c.GoString(symbol.CStr())
|
if len(symbolName) >= 1 && symbolName[0] == '_' {
|
||||||
if len(info.Symbol) >= 1 {
|
symbolName = symbolName[1:]
|
||||||
if info.Symbol[0] == '_' {
|
|
||||||
info.Symbol = info.Symbol[1:]
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
defer symbol.Dispose()
|
defer symbol.Dispose()
|
||||||
defer cursorStr.Dispose()
|
defer cursorStr.Dispose()
|
||||||
|
|
||||||
if context.namespaceName != "" {
|
goName := context.genGoName(name)
|
||||||
info.Namespace = context.namespaceName
|
context.symbolMap[symbolName] = goName
|
||||||
}
|
|
||||||
if context.className != "" {
|
|
||||||
info.Class = context.className
|
|
||||||
}
|
|
||||||
|
|
||||||
typeStr := cursor.ResultType().String()
|
|
||||||
defer typeStr.Dispose()
|
|
||||||
info.ReturnType = c.GoString(typeStr.CStr())
|
|
||||||
|
|
||||||
info.Parameters = make([]types.Parameter, cursor.NumArguments())
|
|
||||||
for i := 0; i < int(cursor.NumArguments()); i++ {
|
|
||||||
argCurSor := cursor.Argument(c.Uint(i))
|
|
||||||
argType := argCurSor.Type().String()
|
|
||||||
argName := argCurSor.String()
|
|
||||||
info.Parameters[i] = types.Parameter{
|
|
||||||
Name: c.GoString(argName.CStr()),
|
|
||||||
Type: c.GoString(argType.CStr()),
|
|
||||||
}
|
|
||||||
|
|
||||||
argType.Dispose()
|
|
||||||
argName.Dispose()
|
|
||||||
}
|
|
||||||
|
|
||||||
return info
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func visit(cursor, parent clang.Cursor, clientData c.Pointer) clang.ChildVisitResult {
|
func visit(cursor, parent clang.Cursor, clientData c.Pointer) clang.ChildVisitResult {
|
||||||
if cursor.Kind == clang.Namespace {
|
if cursor.Kind == clang.CursorNamespace {
|
||||||
nameStr := cursor.String()
|
nameStr := cursor.String()
|
||||||
|
defer nameStr.Dispose()
|
||||||
|
|
||||||
context.setNamespaceName(c.GoString(nameStr.CStr()))
|
context.setNamespaceName(c.GoString(nameStr.CStr()))
|
||||||
clang.VisitChildren(cursor, visit, nil)
|
clang.VisitChildren(cursor, visit, nil)
|
||||||
context.setNamespaceName("")
|
context.setNamespaceName("")
|
||||||
} else if cursor.Kind == clang.ClassDecl {
|
} else if cursor.Kind == clang.CursorClassDecl {
|
||||||
nameStr := cursor.String()
|
nameStr := cursor.String()
|
||||||
|
defer nameStr.Dispose()
|
||||||
|
|
||||||
context.setClassName(c.GoString(nameStr.CStr()))
|
context.setClassName(c.GoString(nameStr.CStr()))
|
||||||
clang.VisitChildren(cursor, visit, nil)
|
clang.VisitChildren(cursor, visit, nil)
|
||||||
context.setClassName("")
|
context.setClassName("")
|
||||||
} else if cursor.Kind == clang.CXXMethod || cursor.Kind == clang.FunctionDecl || cursor.Kind == clang.Constructor || cursor.Kind == clang.Destructor {
|
} else if cursor.Kind == clang.CursorCXXMethod || cursor.Kind == clang.CursorFunctionDecl || cursor.Kind == clang.CursorConstructor || cursor.Kind == clang.CursorDestructor {
|
||||||
loc := cursor.Location()
|
loc := cursor.Location()
|
||||||
var file clang.File
|
var file clang.File
|
||||||
var line, column c.Uint
|
var line, column c.Uint
|
||||||
@@ -107,9 +124,7 @@ func visit(cursor, parent clang.Cursor, clientData c.Pointer) clang.ChildVisitRe
|
|||||||
filename := file.FileName()
|
filename := file.FileName()
|
||||||
|
|
||||||
if c.Strcmp(filename.CStr(), c.AllocaCStr(context.currentFile)) == 0 {
|
if c.Strcmp(filename.CStr(), c.AllocaCStr(context.currentFile)) == 0 {
|
||||||
info := collectFuncInfo(cursor)
|
collectFuncInfo(cursor)
|
||||||
info.Location = c.GoString(filename.CStr()) + ":" + strconv.Itoa(int(line)) + ":" + strconv.Itoa(int(column))
|
|
||||||
context.astInfo = append(context.astInfo, info)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
defer filename.Dispose()
|
defer filename.Dispose()
|
||||||
@@ -118,14 +133,13 @@ func visit(cursor, parent clang.Cursor, clientData c.Pointer) clang.ChildVisitRe
|
|||||||
return clang.ChildVisit_Continue
|
return clang.ChildVisit_Continue
|
||||||
}
|
}
|
||||||
|
|
||||||
func ParseHeaderFile(filepaths []string) ([]types.ASTInformation, error) {
|
func ParseHeaderFile(filepaths []string, prefixes []string) (map[string]string, error) {
|
||||||
|
|
||||||
index := clang.CreateIndex(0, 0)
|
index := clang.CreateIndex(0, 0)
|
||||||
args := make([]*c.Char, 3)
|
args := make([]*c.Char, 3)
|
||||||
args[0] = c.Str("-x")
|
args[0] = c.Str("-x")
|
||||||
args[1] = c.Str("c++")
|
args[1] = c.Str("c++")
|
||||||
args[2] = c.Str("-std=c++11")
|
args[2] = c.Str("-std=c++11")
|
||||||
context = newContext()
|
context = newContext(prefixes)
|
||||||
|
|
||||||
for _, filename := range filepaths {
|
for _, filename := range filepaths {
|
||||||
unit := index.ParseTranslationUnit(
|
unit := index.ParseTranslationUnit(
|
||||||
@@ -149,5 +163,5 @@ func ParseHeaderFile(filepaths []string) ([]types.ASTInformation, error) {
|
|||||||
|
|
||||||
index.Dispose()
|
index.Dispose()
|
||||||
|
|
||||||
return context.astInfo, nil
|
return context.symbolMap, nil
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -36,6 +36,11 @@ type Stmt interface {
|
|||||||
stmtNode()
|
stmtNode()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type PPD interface { // preprocessing directive
|
||||||
|
Node
|
||||||
|
ppdNode()
|
||||||
|
}
|
||||||
|
|
||||||
// =============================================================================
|
// =============================================================================
|
||||||
// Expressions (Types are also expressions)
|
// Expressions (Types are also expressions)
|
||||||
|
|
||||||
@@ -226,6 +231,8 @@ type EnumItem struct {
|
|||||||
Value Expr // optional
|
Value Expr // optional
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (*EnumItem) exprNode() {}
|
||||||
|
|
||||||
// enum Name { Item1, Item2, ... };
|
// enum Name { Item1, Item2, ... };
|
||||||
type EnumTypeDecl struct {
|
type EnumTypeDecl struct {
|
||||||
DeclBase
|
DeclBase
|
||||||
@@ -261,8 +268,25 @@ func (*TypeDecl) declNode() {}
|
|||||||
// =============================================================================
|
// =============================================================================
|
||||||
// AST File
|
// AST File
|
||||||
|
|
||||||
|
type Include struct {
|
||||||
|
Path string `json:"path"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (*Include) ppdNode() {}
|
||||||
|
|
||||||
|
// ------------------------------------------------
|
||||||
|
|
||||||
|
type Macro struct {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (*Macro) ppdNode() {}
|
||||||
|
|
||||||
|
// ------------------------------------------------
|
||||||
|
|
||||||
type File struct {
|
type File struct {
|
||||||
Decls []Decl
|
Decls []Decl `json:"decls"`
|
||||||
|
Includes []*Include `json:"includes,omitempty"`
|
||||||
|
Macros []*Macro `json:"macros,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// =============================================================================
|
// =============================================================================
|
||||||
|
|||||||
@@ -59,8 +59,21 @@ llcppsigfetch - # read config from stdin
|
|||||||
|
|
||||||
It fetches information of C/C++ symbols and print to stdout. Its format is as follows:
|
It fetches information of C/C++ symbols and print to stdout. Its format is as follows:
|
||||||
|
|
||||||
```
|
```json
|
||||||
TODO: see llgo/xtool/clang/ast
|
[
|
||||||
|
{
|
||||||
|
"path": "/path/to/file.h",
|
||||||
|
"doc": {
|
||||||
|
"decls": [],
|
||||||
|
"macros": [],
|
||||||
|
"includes": [
|
||||||
|
{
|
||||||
|
"path": "incfile.h"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
```
|
```
|
||||||
|
|
||||||
### gogensig
|
### gogensig
|
||||||
@@ -69,252 +82,3 @@ TODO: see llgo/xtool/clang/ast
|
|||||||
gogensig ast-file
|
gogensig ast-file
|
||||||
gogensig - # read AST from stdin
|
gogensig - # read AST from stdin
|
||||||
```
|
```
|
||||||
|
|
||||||
## Overall
|
|
||||||
|
|
||||||
### Process
|
|
||||||
|
|
||||||
1. The Parsing Module reads `llcppg.cfg` to obtain dynamic libraries, header files, and the package name. After parsing, it writes the generated `llcppg.symb.json` path into `llcppg.cfg`.
|
|
||||||
2. The Function Declaration Generation Module reads `llcppg.cfg` to get the package name, header files, and the previously generated `llcppg.symb.json`. After parsing, it generates the function prototype `llcppg.function.json`.
|
|
||||||
3. Reads the previously generated `llcppg.information.json`, stores it as a structure, and uses gogen to generate code based on the structure.
|
|
||||||
|
|
||||||
## Parsing Module
|
|
||||||
|
|
||||||
### Input
|
|
||||||
|
|
||||||
Obtains the paths to header files and dynamic library files by reading the JSON file `llcppg.cfg`.
|
|
||||||
|
|
||||||
```json
|
|
||||||
{
|
|
||||||
"name": "inih",
|
|
||||||
"cflags": "$(pkg-config --cflags INIReader)",
|
|
||||||
"include": [
|
|
||||||
"INIReader.h",
|
|
||||||
"AnotherHeaderFile.h"
|
|
||||||
],
|
|
||||||
"libs": "$(pkg-config --libs INIReader)",
|
|
||||||
"trimPrefixes": ["Ini", "INI"]
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
```bash
|
|
||||||
llcppsymg config-file
|
|
||||||
```
|
|
||||||
|
|
||||||
### Implementation Steps
|
|
||||||
|
|
||||||
1. Parse dylib and store:
|
|
||||||
|
|
||||||
```go
|
|
||||||
// types.go
|
|
||||||
type CPPSymbol struct {
|
|
||||||
Symbol string `json:"symbol"`
|
|
||||||
Type string `json:"type"`
|
|
||||||
Name string `json:"name"`
|
|
||||||
}
|
|
||||||
|
|
||||||
// parser_dylib.go
|
|
||||||
func parseDylibSymbols(lib string) ([]common.CPPSymbol, error)
|
|
||||||
```
|
|
||||||
|
|
||||||
2. Parse header files and store:
|
|
||||||
|
|
||||||
```go
|
|
||||||
// common.go
|
|
||||||
type ASTInformation struct {
|
|
||||||
Namespace string `json:"namespace"`
|
|
||||||
Class string `json:"class"`
|
|
||||||
Name string `json:"name"`
|
|
||||||
BaseClasses []string `json:"baseClasses"`
|
|
||||||
ReturnType string `json:"returnType"`
|
|
||||||
Location string `json:"location"`
|
|
||||||
Parameters []Parameter `json:"parameters"`
|
|
||||||
Symbol string `json:"symbol"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type Parameter struct {
|
|
||||||
Name string `json:"name"`
|
|
||||||
Type string `json:"type"`
|
|
||||||
}
|
|
||||||
|
|
||||||
// parser_ast.go
|
|
||||||
func parseHeaderFile(config types.Config) ([]common.ASTInformation, error)
|
|
||||||
```
|
|
||||||
|
|
||||||
3. Cross-reference data from the first two steps to get the final output
|
|
||||||
|
|
||||||
```go
|
|
||||||
// common.go
|
|
||||||
type SymbolInfo struct {
|
|
||||||
Mangle string `json:"mangle"` // C++ Symbol
|
|
||||||
CPP string `json:"c++"` // C++ function name
|
|
||||||
Go string `json:"go"` // Go function name
|
|
||||||
}
|
|
||||||
|
|
||||||
// common_symbols.go
|
|
||||||
func getCommonSymbols(dylibSymbols []common.CPPSymbol, astInfoList []common.ASTInformation) []common.SymbolInfo {
|
|
||||||
```
|
|
||||||
|
|
||||||
4. Generate `llcppg.symb.json` file and store the JSON file path into `llcppg.cfg`
|
|
||||||
|
|
||||||
```go
|
|
||||||
func generateJSON([]CommonSymbolInfo)
|
|
||||||
```
|
|
||||||
|
|
||||||
5. Example `llcppg.symb.json` file
|
|
||||||
|
|
||||||
```json
|
|
||||||
{
|
|
||||||
"FunctionName": "A::B::C",
|
|
||||||
"Symbol": "_ZN9INIReaderC1ERKNSt3__112basic_stringIcNS0_11char_traitsIcEENS0_9allocatorIcEEEE",
|
|
||||||
"Location": "a.h",
|
|
||||||
"UserFunctionName": "CFromA"
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
## Function Declaration Generation Module
|
|
||||||
|
|
||||||
### Input
|
|
||||||
|
|
||||||
No input required, directly reads the `llcppg.cfg` file
|
|
||||||
|
|
||||||
### Implementation Steps
|
|
||||||
|
|
||||||
1. Execute the executable
|
|
||||||
|
|
||||||
```bash
|
|
||||||
llcppsigfetch config-file
|
|
||||||
```
|
|
||||||
|
|
||||||
2. Parse header files
|
|
||||||
|
|
||||||
```go
|
|
||||||
// common.go
|
|
||||||
type ASTInformation struct {
|
|
||||||
Namespace string `json:"namespace"`
|
|
||||||
Class string `json:"class"`
|
|
||||||
Name string `json:"name"`
|
|
||||||
BaseClasses []string `json:"baseClasses"`
|
|
||||||
ReturnType string `json:"returnType"`
|
|
||||||
Location string `json:"location"`
|
|
||||||
Parameters []Parameter `json:"parameters"`
|
|
||||||
Symbol string `json:"symbol"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type Parameter struct {
|
|
||||||
Name string `json:"name"`
|
|
||||||
Type string `json:"type"`
|
|
||||||
}
|
|
||||||
|
|
||||||
// parser_ast.go
|
|
||||||
func ParseHeaderFile(filePath string) ([]common.ASTInformation, error)
|
|
||||||
```
|
|
||||||
|
|
||||||
3. Generate the final JSON mapping file `llcppg.information.json`
|
|
||||||
|
|
||||||
```go
|
|
||||||
func GenerateJSONFile(info []common.ASTInformation)
|
|
||||||
```
|
|
||||||
|
|
||||||
```json
|
|
||||||
{
|
|
||||||
"functionName": "A::B::C",
|
|
||||||
"symbol": "_ZN9INIReaderC1ERKNSt3__112basic_stringIcNS0_11char_traitsIcEENS0_9allocatorIcEEEE",
|
|
||||||
"location": "a.h",
|
|
||||||
"returnType": "int",
|
|
||||||
"userFunctionName": "CFromA",
|
|
||||||
"parameters": [
|
|
||||||
{
|
|
||||||
"arg1": "int"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"arg2": "*char"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
## Code Generation Module
|
|
||||||
|
|
||||||
### Input
|
|
||||||
|
|
||||||
No input required, directly reads `llcppg.information.json` file
|
|
||||||
|
|
||||||
### Implementation Steps
|
|
||||||
|
|
||||||
1. Execute the executable
|
|
||||||
|
|
||||||
```bash
|
|
||||||
gogensig ast-file
|
|
||||||
```
|
|
||||||
|
|
||||||
2. Parse JSON file
|
|
||||||
|
|
||||||
```go
|
|
||||||
// common.go
|
|
||||||
type HeaderFileInfo struct {
|
|
||||||
FunctionName string `json:"functionName"`
|
|
||||||
Symbol string `json:"symbol"`
|
|
||||||
Location string `json:"location"`
|
|
||||||
UserFunctionName string `json:"userFunctionName"`
|
|
||||||
Parameters map[string]string `json:"parameters"`
|
|
||||||
}
|
|
||||||
|
|
||||||
// parse_json.go
|
|
||||||
func ParseJSON(jsonFilePath string) ([]common.HeaderFileInfo, error)
|
|
||||||
```
|
|
||||||
|
|
||||||
3. Generate code using the parsed structure with gogen
|
|
||||||
|
|
||||||
```go
|
|
||||||
// generator.go
|
|
||||||
func GenerateCode(info []common.HeaderFileInfo) {
|
|
||||||
pkg := gogen.NewPackage("", PackageName, nil)
|
|
||||||
cm := comment(fmt.Sprintf("llgo:link %s %s", funcName1, symbol1))
|
|
||||||
pkg.NewFunc(recv, funcName, params, results, variadic).SetComments(pkg, cm).BodyStart(pkg).End()
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
### Output
|
|
||||||
|
|
||||||
1. Directory structure
|
|
||||||
|
|
||||||
```bash
|
|
||||||
package_name/
|
|
||||||
├── _demo
|
|
||||||
├── demo1.go
|
|
||||||
└── llgo_link.go
|
|
||||||
└── a.go
|
|
||||||
└── b.go
|
|
||||||
└── c.go
|
|
||||||
```
|
|
||||||
|
|
||||||
Note that `demo1.go` file needs to be written by the user
|
|
||||||
|
|
||||||
2. `llgo_link.go` is responsible for linking configuration
|
|
||||||
|
|
||||||
```go
|
|
||||||
package inih
|
|
||||||
const (
|
|
||||||
LLGoFiles = "$(pkg-config --cflags INIReader): _wrap/reader.cpp"
|
|
||||||
LLGoPackage = "link: $(pkg-config --libs inih INIReader); -linih -lINIReader"
|
|
||||||
)
|
|
||||||
```
|
|
||||||
|
|
||||||
3. Example content for `a.go`
|
|
||||||
|
|
||||||
```go
|
|
||||||
package inih
|
|
||||||
import (
|
|
||||||
_ "unsafe"
|
|
||||||
"github.com/goplus/llgo/c"
|
|
||||||
)
|
|
||||||
//go:linkname Parse C.ini_parse
|
|
||||||
func Parse(filename *c.Char, handler func(user c.Pointer, section *c.Char, name *c.Char, value *c.Char) c.Int, user c.Pointer) c.Int
|
|
||||||
|
|
||||||
//go:linkname ParseFile C.ini_parse_file
|
|
||||||
func ParseFile(file c.FilePtr, handler func(user c.Pointer, section *c.Char, name *c.Char, value *c.Char) c.Int, user c.Pointer) c.Int
|
|
||||||
|
|
||||||
//go:linkname ParseString C.ini_parse_string
|
|
||||||
func ParseString(str *c.Char, handler func(user c.Pointer, section *c.Char, name *c.Char, value *c.Char) c.Int, user c.Pointer) c.Int
|
|
||||||
```
|
|
||||||
@@ -16,10 +16,6 @@
|
|||||||
|
|
||||||
package types
|
package types
|
||||||
|
|
||||||
import (
|
|
||||||
"github.com/goplus/llgo/xtool/nm"
|
|
||||||
)
|
|
||||||
|
|
||||||
// Config represents a configuration for the llcppg tool.
|
// Config represents a configuration for the llcppg tool.
|
||||||
type Config struct {
|
type Config struct {
|
||||||
Name string `json:"name"`
|
Name string `json:"name"`
|
||||||
@@ -29,27 +25,6 @@ type Config struct {
|
|||||||
TrimPrefixes []string `json:"trimPrefixes"`
|
TrimPrefixes []string `json:"trimPrefixes"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type CPPSymbol struct {
|
|
||||||
DemangleName string
|
|
||||||
*nm.Symbol
|
|
||||||
}
|
|
||||||
|
|
||||||
type ASTInformation struct {
|
|
||||||
Namespace string `json:"namespace"`
|
|
||||||
Class string `json:"class"`
|
|
||||||
Name string `json:"name"`
|
|
||||||
BaseClasses []string `json:"baseClasses"`
|
|
||||||
ReturnType string `json:"returnType"`
|
|
||||||
Location string `json:"location"`
|
|
||||||
Parameters []Parameter `json:"parameters"`
|
|
||||||
Symbol string `json:"symbol"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type Parameter struct {
|
|
||||||
Name string `json:"name"`
|
|
||||||
Type string `json:"type"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type SymbolInfo struct {
|
type SymbolInfo struct {
|
||||||
Mangle string `json:"mangle"` // C++ Symbol
|
Mangle string `json:"mangle"` // C++ Symbol
|
||||||
CPP string `json:"c++"` // C++ function name
|
CPP string `json:"c++"` // C++ function name
|
||||||
|
|||||||
@@ -543,10 +543,12 @@ _llgo_0:
|
|||||||
|
|
||||||
define void @"main.init#7"() {
|
define void @"main.init#7"() {
|
||||||
_llgo_0:
|
_llgo_0:
|
||||||
%0 = load ptr, ptr @"map[_llgo_int]_llgo_string", align 8
|
%0 = load ptr, ptr @_llgo_int, align 8
|
||||||
%1 = call ptr @"github.com/goplus/llgo/internal/runtime.MakeMap"(ptr %0, i64 0)
|
%1 = load ptr, ptr @_llgo_string, align 8
|
||||||
%2 = icmp ne ptr %1, null
|
%2 = load ptr, ptr @"map[_llgo_int]_llgo_string", align 8
|
||||||
call void @main.assert(i1 %2)
|
%3 = call ptr @"github.com/goplus/llgo/internal/runtime.MakeMap"(ptr %2, i64 0)
|
||||||
|
%4 = icmp ne ptr %3, null
|
||||||
|
call void @main.assert(i1 %4)
|
||||||
call void @main.assert(i1 true)
|
call void @main.assert(i1 true)
|
||||||
ret void
|
ret void
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ func main() {
|
|||||||
make4()
|
make4()
|
||||||
make5()
|
make5()
|
||||||
make6()
|
make6()
|
||||||
|
make7()
|
||||||
}
|
}
|
||||||
|
|
||||||
func make1() {
|
func make1() {
|
||||||
@@ -113,3 +114,15 @@ func make6() {
|
|||||||
println(k, v)
|
println(k, v)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func make7() {
|
||||||
|
type N int
|
||||||
|
m := map[N]string{
|
||||||
|
1: "hello",
|
||||||
|
2: "world",
|
||||||
|
}
|
||||||
|
for k, v := range m {
|
||||||
|
println(k, v)
|
||||||
|
}
|
||||||
|
println(m[1])
|
||||||
|
}
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -8,6 +8,7 @@ source_filename = "main"
|
|||||||
@"main.init$guard" = global i1 false, align 1
|
@"main.init$guard" = global i1 false, align 1
|
||||||
@__llgo_argc = global i32 0, align 4
|
@__llgo_argc = global i32 0, align 4
|
||||||
@__llgo_argv = global ptr null, align 8
|
@__llgo_argv = global ptr null, align 8
|
||||||
|
@_llgo_int = linkonce global ptr null, align 8
|
||||||
@"map[_llgo_int]_llgo_int" = linkonce global ptr null, align 8
|
@"map[_llgo_int]_llgo_int" = linkonce global ptr null, align 8
|
||||||
@0 = private unnamed_addr constant [7 x i8] c"topbits", align 1
|
@0 = private unnamed_addr constant [7 x i8] c"topbits", align 1
|
||||||
@1 = private unnamed_addr constant [4 x i8] c"keys", align 1
|
@1 = private unnamed_addr constant [4 x i8] c"keys", align 1
|
||||||
@@ -36,24 +37,32 @@ _llgo_0:
|
|||||||
store ptr %1, ptr @__llgo_argv, align 8
|
store ptr %1, ptr @__llgo_argv, align 8
|
||||||
call void @"github.com/goplus/llgo/internal/runtime.init"()
|
call void @"github.com/goplus/llgo/internal/runtime.init"()
|
||||||
call void @main.init()
|
call void @main.init()
|
||||||
%2 = load ptr, ptr @"map[_llgo_int]_llgo_int", align 8
|
%2 = load ptr, ptr @_llgo_int, align 8
|
||||||
%3 = call ptr @"github.com/goplus/llgo/internal/runtime.MakeMap"(ptr %2, i64 2)
|
%3 = load ptr, ptr @_llgo_int, align 8
|
||||||
%4 = load ptr, ptr @"map[_llgo_int]_llgo_int", align 8
|
%4 = load ptr, ptr @"map[_llgo_int]_llgo_int", align 8
|
||||||
%5 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocU"(i64 8)
|
%5 = call ptr @"github.com/goplus/llgo/internal/runtime.MakeMap"(ptr %4, i64 2)
|
||||||
store i64 23, ptr %5, align 4
|
%6 = load ptr, ptr @_llgo_int, align 8
|
||||||
%6 = call ptr @"github.com/goplus/llgo/internal/runtime.MapAssign"(ptr %4, ptr %3, ptr %5)
|
%7 = load ptr, ptr @_llgo_int, align 8
|
||||||
store i64 100, ptr %6, align 4
|
%8 = load ptr, ptr @"map[_llgo_int]_llgo_int", align 8
|
||||||
%7 = load ptr, ptr @"map[_llgo_int]_llgo_int", align 8
|
%9 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocU"(i64 8)
|
||||||
%8 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocU"(i64 8)
|
store i64 23, ptr %9, align 4
|
||||||
store i64 7, ptr %8, align 4
|
%10 = call ptr @"github.com/goplus/llgo/internal/runtime.MapAssign"(ptr %8, ptr %5, ptr %9)
|
||||||
%9 = call ptr @"github.com/goplus/llgo/internal/runtime.MapAssign"(ptr %7, ptr %3, ptr %8)
|
store i64 100, ptr %10, align 4
|
||||||
store i64 29, ptr %9, align 4
|
%11 = load ptr, ptr @_llgo_int, align 8
|
||||||
%10 = load ptr, ptr @"map[_llgo_int]_llgo_int", align 8
|
%12 = load ptr, ptr @_llgo_int, align 8
|
||||||
%11 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocU"(i64 8)
|
%13 = load ptr, ptr @"map[_llgo_int]_llgo_int", align 8
|
||||||
store i64 23, ptr %11, align 4
|
%14 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocU"(i64 8)
|
||||||
%12 = call ptr @"github.com/goplus/llgo/internal/runtime.MapAccess1"(ptr %10, ptr %3, ptr %11)
|
store i64 7, ptr %14, align 4
|
||||||
%13 = load i64, ptr %12, align 4
|
%15 = call ptr @"github.com/goplus/llgo/internal/runtime.MapAssign"(ptr %13, ptr %5, ptr %14)
|
||||||
%14 = call i32 (ptr, ...) @printf(ptr @5, i64 %13)
|
store i64 29, ptr %15, align 4
|
||||||
|
%16 = load ptr, ptr @_llgo_int, align 8
|
||||||
|
%17 = load ptr, ptr @_llgo_int, align 8
|
||||||
|
%18 = load ptr, ptr @"map[_llgo_int]_llgo_int", align 8
|
||||||
|
%19 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocU"(i64 8)
|
||||||
|
store i64 23, ptr %19, align 4
|
||||||
|
%20 = call ptr @"github.com/goplus/llgo/internal/runtime.MapAccess1"(ptr %18, ptr %5, ptr %19)
|
||||||
|
%21 = load i64, ptr %20, align 4
|
||||||
|
%22 = call i32 (ptr, ...) @printf(ptr @5, i64 %21)
|
||||||
ret i32 0
|
ret i32 0
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -61,109 +70,119 @@ declare void @"github.com/goplus/llgo/internal/runtime.init"()
|
|||||||
|
|
||||||
define void @"main.init$after"() {
|
define void @"main.init$after"() {
|
||||||
_llgo_0:
|
_llgo_0:
|
||||||
%0 = load ptr, ptr @"map[_llgo_int]_llgo_int", align 8
|
%0 = load ptr, ptr @_llgo_int, align 8
|
||||||
%1 = icmp eq ptr %0, null
|
%1 = icmp eq ptr %0, null
|
||||||
br i1 %1, label %_llgo_1, label %_llgo_2
|
br i1 %1, label %_llgo_1, label %_llgo_2
|
||||||
|
|
||||||
_llgo_1: ; preds = %_llgo_0
|
_llgo_1: ; preds = %_llgo_0
|
||||||
%2 = call ptr @"github.com/goplus/llgo/internal/runtime.Basic"(i64 34)
|
%2 = call ptr @"github.com/goplus/llgo/internal/runtime.Basic"(i64 34)
|
||||||
%3 = call ptr @"github.com/goplus/llgo/internal/runtime.Basic"(i64 34)
|
store ptr %2, ptr @_llgo_int, align 8
|
||||||
%4 = alloca %"github.com/goplus/llgo/internal/runtime.String", align 8
|
|
||||||
%5 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %4, i32 0, i32 0
|
|
||||||
store ptr @0, ptr %5, align 8
|
|
||||||
%6 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %4, i32 0, i32 1
|
|
||||||
store i64 7, ptr %6, align 4
|
|
||||||
%7 = load %"github.com/goplus/llgo/internal/runtime.String", ptr %4, align 8
|
|
||||||
%8 = alloca %"github.com/goplus/llgo/internal/runtime.String", align 8
|
|
||||||
%9 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %8, i32 0, i32 0
|
|
||||||
store ptr null, ptr %9, align 8
|
|
||||||
%10 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %8, i32 0, i32 1
|
|
||||||
store i64 0, ptr %10, align 4
|
|
||||||
%11 = load %"github.com/goplus/llgo/internal/runtime.String", ptr %8, align 8
|
|
||||||
%12 = call ptr @"github.com/goplus/llgo/internal/runtime.Basic"(i64 40)
|
|
||||||
%13 = call ptr @"github.com/goplus/llgo/internal/runtime.ArrayOf"(i64 8, ptr %12)
|
|
||||||
%14 = call %"github.com/goplus/llgo/internal/abi.StructField" @"github.com/goplus/llgo/internal/runtime.StructField"(%"github.com/goplus/llgo/internal/runtime.String" %7, ptr %13, i64 0, %"github.com/goplus/llgo/internal/runtime.String" %11, i1 false)
|
|
||||||
%15 = alloca %"github.com/goplus/llgo/internal/runtime.String", align 8
|
|
||||||
%16 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %15, i32 0, i32 0
|
|
||||||
store ptr @1, ptr %16, align 8
|
|
||||||
%17 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %15, i32 0, i32 1
|
|
||||||
store i64 4, ptr %17, align 4
|
|
||||||
%18 = load %"github.com/goplus/llgo/internal/runtime.String", ptr %15, align 8
|
|
||||||
%19 = alloca %"github.com/goplus/llgo/internal/runtime.String", align 8
|
|
||||||
%20 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %19, i32 0, i32 0
|
|
||||||
store ptr null, ptr %20, align 8
|
|
||||||
%21 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %19, i32 0, i32 1
|
|
||||||
store i64 0, ptr %21, align 4
|
|
||||||
%22 = load %"github.com/goplus/llgo/internal/runtime.String", ptr %19, align 8
|
|
||||||
%23 = call ptr @"github.com/goplus/llgo/internal/runtime.Basic"(i64 34)
|
|
||||||
%24 = call ptr @"github.com/goplus/llgo/internal/runtime.ArrayOf"(i64 8, ptr %23)
|
|
||||||
%25 = call %"github.com/goplus/llgo/internal/abi.StructField" @"github.com/goplus/llgo/internal/runtime.StructField"(%"github.com/goplus/llgo/internal/runtime.String" %18, ptr %24, i64 8, %"github.com/goplus/llgo/internal/runtime.String" %22, i1 false)
|
|
||||||
%26 = alloca %"github.com/goplus/llgo/internal/runtime.String", align 8
|
|
||||||
%27 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %26, i32 0, i32 0
|
|
||||||
store ptr @2, ptr %27, align 8
|
|
||||||
%28 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %26, i32 0, i32 1
|
|
||||||
store i64 5, ptr %28, align 4
|
|
||||||
%29 = load %"github.com/goplus/llgo/internal/runtime.String", ptr %26, align 8
|
|
||||||
%30 = alloca %"github.com/goplus/llgo/internal/runtime.String", align 8
|
|
||||||
%31 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %30, i32 0, i32 0
|
|
||||||
store ptr null, ptr %31, align 8
|
|
||||||
%32 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %30, i32 0, i32 1
|
|
||||||
store i64 0, ptr %32, align 4
|
|
||||||
%33 = load %"github.com/goplus/llgo/internal/runtime.String", ptr %30, align 8
|
|
||||||
%34 = call ptr @"github.com/goplus/llgo/internal/runtime.Basic"(i64 34)
|
|
||||||
%35 = call ptr @"github.com/goplus/llgo/internal/runtime.ArrayOf"(i64 8, ptr %34)
|
|
||||||
%36 = call %"github.com/goplus/llgo/internal/abi.StructField" @"github.com/goplus/llgo/internal/runtime.StructField"(%"github.com/goplus/llgo/internal/runtime.String" %29, ptr %35, i64 72, %"github.com/goplus/llgo/internal/runtime.String" %33, i1 false)
|
|
||||||
%37 = alloca %"github.com/goplus/llgo/internal/runtime.String", align 8
|
|
||||||
%38 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %37, i32 0, i32 0
|
|
||||||
store ptr @3, ptr %38, align 8
|
|
||||||
%39 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %37, i32 0, i32 1
|
|
||||||
store i64 8, ptr %39, align 4
|
|
||||||
%40 = load %"github.com/goplus/llgo/internal/runtime.String", ptr %37, align 8
|
|
||||||
%41 = alloca %"github.com/goplus/llgo/internal/runtime.String", align 8
|
|
||||||
%42 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %41, i32 0, i32 0
|
|
||||||
store ptr null, ptr %42, align 8
|
|
||||||
%43 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %41, i32 0, i32 1
|
|
||||||
store i64 0, ptr %43, align 4
|
|
||||||
%44 = load %"github.com/goplus/llgo/internal/runtime.String", ptr %41, align 8
|
|
||||||
%45 = call ptr @"github.com/goplus/llgo/internal/runtime.Basic"(i64 44)
|
|
||||||
%46 = call %"github.com/goplus/llgo/internal/abi.StructField" @"github.com/goplus/llgo/internal/runtime.StructField"(%"github.com/goplus/llgo/internal/runtime.String" %40, ptr %45, i64 136, %"github.com/goplus/llgo/internal/runtime.String" %44, i1 false)
|
|
||||||
%47 = alloca %"github.com/goplus/llgo/internal/runtime.String", align 8
|
|
||||||
%48 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %47, i32 0, i32 0
|
|
||||||
store ptr @4, ptr %48, align 8
|
|
||||||
%49 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %47, i32 0, i32 1
|
|
||||||
store i64 4, ptr %49, align 4
|
|
||||||
%50 = load %"github.com/goplus/llgo/internal/runtime.String", ptr %47, align 8
|
|
||||||
%51 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocU"(i64 224)
|
|
||||||
%52 = getelementptr %"github.com/goplus/llgo/internal/abi.StructField", ptr %51, i64 0
|
|
||||||
store %"github.com/goplus/llgo/internal/abi.StructField" %14, ptr %52, align 8
|
|
||||||
%53 = getelementptr %"github.com/goplus/llgo/internal/abi.StructField", ptr %51, i64 1
|
|
||||||
store %"github.com/goplus/llgo/internal/abi.StructField" %25, ptr %53, align 8
|
|
||||||
%54 = getelementptr %"github.com/goplus/llgo/internal/abi.StructField", ptr %51, i64 2
|
|
||||||
store %"github.com/goplus/llgo/internal/abi.StructField" %36, ptr %54, align 8
|
|
||||||
%55 = getelementptr %"github.com/goplus/llgo/internal/abi.StructField", ptr %51, i64 3
|
|
||||||
store %"github.com/goplus/llgo/internal/abi.StructField" %46, ptr %55, align 8
|
|
||||||
%56 = alloca %"github.com/goplus/llgo/internal/runtime.Slice", align 8
|
|
||||||
%57 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.Slice", ptr %56, i32 0, i32 0
|
|
||||||
store ptr %51, ptr %57, align 8
|
|
||||||
%58 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.Slice", ptr %56, i32 0, i32 1
|
|
||||||
store i64 4, ptr %58, align 4
|
|
||||||
%59 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.Slice", ptr %56, i32 0, i32 2
|
|
||||||
store i64 4, ptr %59, align 4
|
|
||||||
%60 = load %"github.com/goplus/llgo/internal/runtime.Slice", ptr %56, align 8
|
|
||||||
%61 = call ptr @"github.com/goplus/llgo/internal/runtime.Struct"(%"github.com/goplus/llgo/internal/runtime.String" %50, i64 144, %"github.com/goplus/llgo/internal/runtime.Slice" %60)
|
|
||||||
%62 = call ptr @"github.com/goplus/llgo/internal/runtime.MapOf"(ptr %2, ptr %3, ptr %61, i64 4)
|
|
||||||
call void @"github.com/goplus/llgo/internal/runtime.SetDirectIface"(ptr %62)
|
|
||||||
store ptr %62, ptr @"map[_llgo_int]_llgo_int", align 8
|
|
||||||
br label %_llgo_2
|
br label %_llgo_2
|
||||||
|
|
||||||
_llgo_2: ; preds = %_llgo_1, %_llgo_0
|
_llgo_2: ; preds = %_llgo_1, %_llgo_0
|
||||||
|
%3 = load ptr, ptr @"map[_llgo_int]_llgo_int", align 8
|
||||||
|
%4 = icmp eq ptr %3, null
|
||||||
|
br i1 %4, label %_llgo_3, label %_llgo_4
|
||||||
|
|
||||||
|
_llgo_3: ; preds = %_llgo_2
|
||||||
|
%5 = call ptr @"github.com/goplus/llgo/internal/runtime.Basic"(i64 34)
|
||||||
|
%6 = call ptr @"github.com/goplus/llgo/internal/runtime.Basic"(i64 34)
|
||||||
|
%7 = alloca %"github.com/goplus/llgo/internal/runtime.String", align 8
|
||||||
|
%8 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %7, i32 0, i32 0
|
||||||
|
store ptr @0, ptr %8, align 8
|
||||||
|
%9 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %7, i32 0, i32 1
|
||||||
|
store i64 7, ptr %9, align 4
|
||||||
|
%10 = load %"github.com/goplus/llgo/internal/runtime.String", ptr %7, align 8
|
||||||
|
%11 = alloca %"github.com/goplus/llgo/internal/runtime.String", align 8
|
||||||
|
%12 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %11, i32 0, i32 0
|
||||||
|
store ptr null, ptr %12, align 8
|
||||||
|
%13 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %11, i32 0, i32 1
|
||||||
|
store i64 0, ptr %13, align 4
|
||||||
|
%14 = load %"github.com/goplus/llgo/internal/runtime.String", ptr %11, align 8
|
||||||
|
%15 = call ptr @"github.com/goplus/llgo/internal/runtime.Basic"(i64 40)
|
||||||
|
%16 = call ptr @"github.com/goplus/llgo/internal/runtime.ArrayOf"(i64 8, ptr %15)
|
||||||
|
%17 = call %"github.com/goplus/llgo/internal/abi.StructField" @"github.com/goplus/llgo/internal/runtime.StructField"(%"github.com/goplus/llgo/internal/runtime.String" %10, ptr %16, i64 0, %"github.com/goplus/llgo/internal/runtime.String" %14, i1 false)
|
||||||
|
%18 = alloca %"github.com/goplus/llgo/internal/runtime.String", align 8
|
||||||
|
%19 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %18, i32 0, i32 0
|
||||||
|
store ptr @1, ptr %19, align 8
|
||||||
|
%20 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %18, i32 0, i32 1
|
||||||
|
store i64 4, ptr %20, align 4
|
||||||
|
%21 = load %"github.com/goplus/llgo/internal/runtime.String", ptr %18, align 8
|
||||||
|
%22 = alloca %"github.com/goplus/llgo/internal/runtime.String", align 8
|
||||||
|
%23 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %22, i32 0, i32 0
|
||||||
|
store ptr null, ptr %23, align 8
|
||||||
|
%24 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %22, i32 0, i32 1
|
||||||
|
store i64 0, ptr %24, align 4
|
||||||
|
%25 = load %"github.com/goplus/llgo/internal/runtime.String", ptr %22, align 8
|
||||||
|
%26 = call ptr @"github.com/goplus/llgo/internal/runtime.Basic"(i64 34)
|
||||||
|
%27 = call ptr @"github.com/goplus/llgo/internal/runtime.ArrayOf"(i64 8, ptr %26)
|
||||||
|
%28 = call %"github.com/goplus/llgo/internal/abi.StructField" @"github.com/goplus/llgo/internal/runtime.StructField"(%"github.com/goplus/llgo/internal/runtime.String" %21, ptr %27, i64 8, %"github.com/goplus/llgo/internal/runtime.String" %25, i1 false)
|
||||||
|
%29 = alloca %"github.com/goplus/llgo/internal/runtime.String", align 8
|
||||||
|
%30 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %29, i32 0, i32 0
|
||||||
|
store ptr @2, ptr %30, align 8
|
||||||
|
%31 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %29, i32 0, i32 1
|
||||||
|
store i64 5, ptr %31, align 4
|
||||||
|
%32 = load %"github.com/goplus/llgo/internal/runtime.String", ptr %29, align 8
|
||||||
|
%33 = alloca %"github.com/goplus/llgo/internal/runtime.String", align 8
|
||||||
|
%34 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %33, i32 0, i32 0
|
||||||
|
store ptr null, ptr %34, align 8
|
||||||
|
%35 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %33, i32 0, i32 1
|
||||||
|
store i64 0, ptr %35, align 4
|
||||||
|
%36 = load %"github.com/goplus/llgo/internal/runtime.String", ptr %33, align 8
|
||||||
|
%37 = call ptr @"github.com/goplus/llgo/internal/runtime.Basic"(i64 34)
|
||||||
|
%38 = call ptr @"github.com/goplus/llgo/internal/runtime.ArrayOf"(i64 8, ptr %37)
|
||||||
|
%39 = call %"github.com/goplus/llgo/internal/abi.StructField" @"github.com/goplus/llgo/internal/runtime.StructField"(%"github.com/goplus/llgo/internal/runtime.String" %32, ptr %38, i64 72, %"github.com/goplus/llgo/internal/runtime.String" %36, i1 false)
|
||||||
|
%40 = alloca %"github.com/goplus/llgo/internal/runtime.String", align 8
|
||||||
|
%41 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %40, i32 0, i32 0
|
||||||
|
store ptr @3, ptr %41, align 8
|
||||||
|
%42 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %40, i32 0, i32 1
|
||||||
|
store i64 8, ptr %42, align 4
|
||||||
|
%43 = load %"github.com/goplus/llgo/internal/runtime.String", ptr %40, align 8
|
||||||
|
%44 = alloca %"github.com/goplus/llgo/internal/runtime.String", align 8
|
||||||
|
%45 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %44, i32 0, i32 0
|
||||||
|
store ptr null, ptr %45, align 8
|
||||||
|
%46 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %44, i32 0, i32 1
|
||||||
|
store i64 0, ptr %46, align 4
|
||||||
|
%47 = load %"github.com/goplus/llgo/internal/runtime.String", ptr %44, align 8
|
||||||
|
%48 = call ptr @"github.com/goplus/llgo/internal/runtime.Basic"(i64 44)
|
||||||
|
%49 = call %"github.com/goplus/llgo/internal/abi.StructField" @"github.com/goplus/llgo/internal/runtime.StructField"(%"github.com/goplus/llgo/internal/runtime.String" %43, ptr %48, i64 136, %"github.com/goplus/llgo/internal/runtime.String" %47, i1 false)
|
||||||
|
%50 = alloca %"github.com/goplus/llgo/internal/runtime.String", align 8
|
||||||
|
%51 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %50, i32 0, i32 0
|
||||||
|
store ptr @4, ptr %51, align 8
|
||||||
|
%52 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %50, i32 0, i32 1
|
||||||
|
store i64 4, ptr %52, align 4
|
||||||
|
%53 = load %"github.com/goplus/llgo/internal/runtime.String", ptr %50, align 8
|
||||||
|
%54 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocU"(i64 224)
|
||||||
|
%55 = getelementptr %"github.com/goplus/llgo/internal/abi.StructField", ptr %54, i64 0
|
||||||
|
store %"github.com/goplus/llgo/internal/abi.StructField" %17, ptr %55, align 8
|
||||||
|
%56 = getelementptr %"github.com/goplus/llgo/internal/abi.StructField", ptr %54, i64 1
|
||||||
|
store %"github.com/goplus/llgo/internal/abi.StructField" %28, ptr %56, align 8
|
||||||
|
%57 = getelementptr %"github.com/goplus/llgo/internal/abi.StructField", ptr %54, i64 2
|
||||||
|
store %"github.com/goplus/llgo/internal/abi.StructField" %39, ptr %57, align 8
|
||||||
|
%58 = getelementptr %"github.com/goplus/llgo/internal/abi.StructField", ptr %54, i64 3
|
||||||
|
store %"github.com/goplus/llgo/internal/abi.StructField" %49, ptr %58, align 8
|
||||||
|
%59 = alloca %"github.com/goplus/llgo/internal/runtime.Slice", align 8
|
||||||
|
%60 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.Slice", ptr %59, i32 0, i32 0
|
||||||
|
store ptr %54, ptr %60, align 8
|
||||||
|
%61 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.Slice", ptr %59, i32 0, i32 1
|
||||||
|
store i64 4, ptr %61, align 4
|
||||||
|
%62 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.Slice", ptr %59, i32 0, i32 2
|
||||||
|
store i64 4, ptr %62, align 4
|
||||||
|
%63 = load %"github.com/goplus/llgo/internal/runtime.Slice", ptr %59, align 8
|
||||||
|
%64 = call ptr @"github.com/goplus/llgo/internal/runtime.Struct"(%"github.com/goplus/llgo/internal/runtime.String" %53, i64 144, %"github.com/goplus/llgo/internal/runtime.Slice" %63)
|
||||||
|
%65 = call ptr @"github.com/goplus/llgo/internal/runtime.MapOf"(ptr %5, ptr %6, ptr %64, i64 4)
|
||||||
|
call void @"github.com/goplus/llgo/internal/runtime.SetDirectIface"(ptr %65)
|
||||||
|
store ptr %65, ptr @"map[_llgo_int]_llgo_int", align 8
|
||||||
|
br label %_llgo_4
|
||||||
|
|
||||||
|
_llgo_4: ; preds = %_llgo_3, %_llgo_2
|
||||||
ret void
|
ret void
|
||||||
}
|
}
|
||||||
|
|
||||||
declare ptr @"github.com/goplus/llgo/internal/runtime.MapOf"(ptr, ptr, ptr, i64)
|
|
||||||
|
|
||||||
declare ptr @"github.com/goplus/llgo/internal/runtime.Basic"(i64)
|
declare ptr @"github.com/goplus/llgo/internal/runtime.Basic"(i64)
|
||||||
|
|
||||||
|
declare ptr @"github.com/goplus/llgo/internal/runtime.MapOf"(ptr, ptr, ptr, i64)
|
||||||
|
|
||||||
declare ptr @"github.com/goplus/llgo/internal/runtime.Struct"(%"github.com/goplus/llgo/internal/runtime.String", i64, %"github.com/goplus/llgo/internal/runtime.Slice")
|
declare ptr @"github.com/goplus/llgo/internal/runtime.Struct"(%"github.com/goplus/llgo/internal/runtime.String", i64, %"github.com/goplus/llgo/internal/runtime.Slice")
|
||||||
|
|
||||||
declare %"github.com/goplus/llgo/internal/abi.StructField" @"github.com/goplus/llgo/internal/runtime.StructField"(%"github.com/goplus/llgo/internal/runtime.String", ptr, i64, %"github.com/goplus/llgo/internal/runtime.String", i1)
|
declare %"github.com/goplus/llgo/internal/abi.StructField" @"github.com/goplus/llgo/internal/runtime.StructField"(%"github.com/goplus/llgo/internal/runtime.String", ptr, i64, %"github.com/goplus/llgo/internal/runtime.String", i1)
|
||||||
|
|||||||
@@ -100,6 +100,7 @@ type context struct {
|
|||||||
|
|
||||||
inits []func()
|
inits []func()
|
||||||
phis []func()
|
phis []func()
|
||||||
|
initAfter func()
|
||||||
|
|
||||||
state pkgState
|
state pkgState
|
||||||
inCFunc bool
|
inCFunc bool
|
||||||
@@ -292,9 +293,9 @@ func (p *context) compileBlock(b llssa.Builder, block *ssa.BasicBlock, n int, do
|
|||||||
instrs = instrs[:last]
|
instrs = instrs[:last]
|
||||||
} else if p.state != pkgHasPatch {
|
} else if p.state != pkgHasPatch {
|
||||||
// TODO(xsw): confirm pyMod don't need to call AfterInit
|
// TODO(xsw): confirm pyMod don't need to call AfterInit
|
||||||
p.inits = append(p.inits, func() {
|
p.initAfter = func() {
|
||||||
pkg.AfterInit(b, ret)
|
pkg.AfterInit(b, ret)
|
||||||
})
|
}
|
||||||
}
|
}
|
||||||
} else if doMainInit {
|
} else if doMainInit {
|
||||||
argc := pkg.NewVar("__llgo_argc", types.NewPointer(types.Typ[types.Int32]), llssa.InC)
|
argc := pkg.NewVar("__llgo_argc", types.NewPointer(types.Typ[types.Int32]), llssa.InC)
|
||||||
@@ -837,6 +838,10 @@ func NewPackageEx(prog llssa.Program, patches Patches, pkg *ssa.Package, files [
|
|||||||
ini()
|
ini()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if fn := ctx.initAfter; fn != nil {
|
||||||
|
ctx.initAfter = nil
|
||||||
|
fn()
|
||||||
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -133,7 +133,7 @@ func (p *context) stringData(b llssa.Builder, args []ssa.Value) (ret llssa.Expr)
|
|||||||
}
|
}
|
||||||
|
|
||||||
// func funcAddr(fn any) unsafe.Pointer
|
// func funcAddr(fn any) unsafe.Pointer
|
||||||
func (p *context) funcAddr(b llssa.Builder, args []ssa.Value) llssa.Expr {
|
func (p *context) funcAddr(_ llssa.Builder, args []ssa.Value) llssa.Expr {
|
||||||
if len(args) == 1 {
|
if len(args) == 1 {
|
||||||
if fn, ok := args[0].(*ssa.MakeInterface); ok {
|
if fn, ok := args[0].(*ssa.MakeInterface); ok {
|
||||||
if fnDecl, ok := fn.X.(*ssa.Function); ok {
|
if fnDecl, ok := fn.X.(*ssa.Function); ok {
|
||||||
|
|||||||
2
go.mod
2
go.mod
@@ -7,7 +7,7 @@ require (
|
|||||||
github.com/goplus/llvm v0.8.0
|
github.com/goplus/llvm v0.8.0
|
||||||
github.com/goplus/mod v0.13.12
|
github.com/goplus/mod v0.13.12
|
||||||
github.com/qiniu/x v1.13.10
|
github.com/qiniu/x v1.13.10
|
||||||
golang.org/x/tools v0.22.0
|
golang.org/x/tools v0.19.0
|
||||||
)
|
)
|
||||||
|
|
||||||
require (
|
require (
|
||||||
|
|||||||
4
go.sum
4
go.sum
@@ -10,5 +10,5 @@ golang.org/x/mod v0.19.0 h1:fEdghXQSo20giMthA7cd28ZC+jts4amQ3YMXiP5oMQ8=
|
|||||||
golang.org/x/mod v0.19.0/go.mod h1:hTbmBsO62+eylJbnUtE2MGJUyE7QWk4xUqPFrRgJ+7c=
|
golang.org/x/mod v0.19.0/go.mod h1:hTbmBsO62+eylJbnUtE2MGJUyE7QWk4xUqPFrRgJ+7c=
|
||||||
golang.org/x/sync v0.7.0 h1:YsImfSBoP9QPYL0xyKJPq0gcaJdG3rInoqxTWbfQu9M=
|
golang.org/x/sync v0.7.0 h1:YsImfSBoP9QPYL0xyKJPq0gcaJdG3rInoqxTWbfQu9M=
|
||||||
golang.org/x/sync v0.7.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk=
|
golang.org/x/sync v0.7.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk=
|
||||||
golang.org/x/tools v0.22.0 h1:gqSGLZqv+AI9lIQzniJ0nZDRG5GBPsSi+DRNHWNz6yA=
|
golang.org/x/tools v0.19.0 h1:tfGCXNR1OsFG+sVdLAitlpjAvD/I6dHDKnYrpEZUHkw=
|
||||||
golang.org/x/tools v0.22.0/go.mod h1:aCwcsjqvq7Yqt6TNyX7QMU2enbQ/Gt0bo6krSeEri+c=
|
golang.org/x/tools v0.19.0/go.mod h1:qoJWxmGSIBmAeriMx19ogtrEPrGtDbPK634QFIcLAhc=
|
||||||
|
|||||||
@@ -757,11 +757,13 @@ func findDylibDep(exe, lib string) string {
|
|||||||
type none struct{}
|
type none struct{}
|
||||||
|
|
||||||
var hasAltPkg = map[string]none{
|
var hasAltPkg = map[string]none{
|
||||||
|
"crypto/hmac": {},
|
||||||
"crypto/md5": {},
|
"crypto/md5": {},
|
||||||
|
"crypto/rand": {},
|
||||||
"crypto/sha1": {},
|
"crypto/sha1": {},
|
||||||
"crypto/sha256": {},
|
"crypto/sha256": {},
|
||||||
"crypto/sha512": {},
|
"crypto/sha512": {},
|
||||||
"crypto/rand": {},
|
"crypto/subtle": {},
|
||||||
"fmt": {},
|
"fmt": {},
|
||||||
"hash/crc32": {},
|
"hash/crc32": {},
|
||||||
"internal/abi": {},
|
"internal/abi": {},
|
||||||
|
|||||||
72
internal/lib/crypto/hmac/hmac.go
Normal file
72
internal/lib/crypto/hmac/hmac.go
Normal file
@@ -0,0 +1,72 @@
|
|||||||
|
package hmac
|
||||||
|
|
||||||
|
// llgo:skipall
|
||||||
|
import (
|
||||||
|
"crypto/sha256"
|
||||||
|
"crypto/subtle"
|
||||||
|
"hash"
|
||||||
|
"unsafe"
|
||||||
|
|
||||||
|
"github.com/goplus/llgo/c"
|
||||||
|
"github.com/goplus/llgo/c/openssl"
|
||||||
|
)
|
||||||
|
|
||||||
|
type eface struct {
|
||||||
|
_type unsafe.Pointer
|
||||||
|
funcPtr *unsafe.Pointer
|
||||||
|
}
|
||||||
|
|
||||||
|
func funcOf(a any) unsafe.Pointer {
|
||||||
|
e := (*eface)(unsafe.Pointer(&a))
|
||||||
|
return *e.funcPtr
|
||||||
|
}
|
||||||
|
|
||||||
|
type digest openssl.HMAC_CTX
|
||||||
|
|
||||||
|
func (d *digest) Size() int { panic("todo: hmac.(*digest).Size") }
|
||||||
|
|
||||||
|
func (d *digest) BlockSize() int { panic("todo: hmac.(*digest).BlockSize") }
|
||||||
|
|
||||||
|
func (d *digest) Reset() {
|
||||||
|
(*openssl.HMAC_CTX)(d).Reset()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *digest) Write(p []byte) (nn int, err error) {
|
||||||
|
(*openssl.HMAC_CTX)(d).UpdateBytes(p)
|
||||||
|
return len(p), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *digest) Sum(in []byte) []byte {
|
||||||
|
const Size = openssl.EVP_MAX_MD_SIZE
|
||||||
|
var digestLen c.Uint
|
||||||
|
hash := (*[Size]byte)(c.Alloca(Size))
|
||||||
|
(*openssl.HMAC_CTX)(d).Final(&hash[0], &digestLen)
|
||||||
|
return append(in, hash[:digestLen]...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// New returns a new HMAC hash using the given [hash.Hash] type and key.
|
||||||
|
// New functions like sha256.New from [crypto/sha256] can be used as h.
|
||||||
|
// h must return a new Hash every time it is called.
|
||||||
|
// Note that unlike other hash implementations in the standard library,
|
||||||
|
// the returned Hash does not implement [encoding.BinaryMarshaler]
|
||||||
|
// or [encoding.BinaryUnmarshaler].
|
||||||
|
func New(h func() hash.Hash, key []byte) hash.Hash {
|
||||||
|
var md *openssl.EVP_MD
|
||||||
|
switch funcOf(h) {
|
||||||
|
case c.Func(sha256.New):
|
||||||
|
md = openssl.EVP_sha256()
|
||||||
|
default:
|
||||||
|
panic("todo: hmac.New: unsupported hash function")
|
||||||
|
}
|
||||||
|
ctx := openssl.NewHMAC_CTX()
|
||||||
|
ctx.InitBytes(key, md)
|
||||||
|
return (*digest)(ctx)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Equal compares two MACs for equality without leaking timing information.
|
||||||
|
func Equal(mac1, mac2 []byte) bool {
|
||||||
|
// We don't have to be constant time if the lengths of the MACs are
|
||||||
|
// different as that suggests that a completely different hash function
|
||||||
|
// was used.
|
||||||
|
return subtle.ConstantTimeCompare(mac1, mac2) == 1
|
||||||
|
}
|
||||||
@@ -1,3 +1,19 @@
|
|||||||
|
/*
|
||||||
|
* 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 sha1
|
package sha1
|
||||||
|
|
||||||
// llgo:skipall
|
// llgo:skipall
|
||||||
|
|||||||
@@ -1,3 +1,19 @@
|
|||||||
|
/*
|
||||||
|
* 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 sha256
|
package sha256
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
|||||||
@@ -1,3 +1,19 @@
|
|||||||
|
/*
|
||||||
|
* 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 sha256
|
package sha256
|
||||||
|
|
||||||
// llgo:skipall
|
// llgo:skipall
|
||||||
|
|||||||
@@ -1,3 +1,19 @@
|
|||||||
|
/*
|
||||||
|
* 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 sha512
|
package sha512
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
|||||||
@@ -1,3 +1,19 @@
|
|||||||
|
/*
|
||||||
|
* 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 sha512
|
package sha512
|
||||||
|
|
||||||
// llgo:skipall
|
// llgo:skipall
|
||||||
|
|||||||
22
internal/lib/crypto/subtle/xor.go
Normal file
22
internal/lib/crypto/subtle/xor.go
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
/*
|
||||||
|
* 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 subtle
|
||||||
|
|
||||||
|
// llgo:skip XORBytes
|
||||||
|
import (
|
||||||
|
_ "unsafe"
|
||||||
|
)
|
||||||
220
internal/lib/internal/fmtsort/sort.go
Normal file
220
internal/lib/internal/fmtsort/sort.go
Normal file
@@ -0,0 +1,220 @@
|
|||||||
|
// Copyright 2018 The Go Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
// Package fmtsort provides a general stable ordering mechanism
|
||||||
|
// for maps, on behalf of the fmt and text/template packages.
|
||||||
|
// It is not guaranteed to be efficient and works only for types
|
||||||
|
// that are valid map keys.
|
||||||
|
package fmtsort
|
||||||
|
|
||||||
|
// llgo:skipall
|
||||||
|
import (
|
||||||
|
"reflect"
|
||||||
|
"sort"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Note: Throughout this package we avoid calling reflect.Value.Interface as
|
||||||
|
// it is not always legal to do so and it's easier to avoid the issue than to face it.
|
||||||
|
|
||||||
|
// SortedMap represents a map's keys and values. The keys and values are
|
||||||
|
// aligned in index order: Value[i] is the value in the map corresponding to Key[i].
|
||||||
|
type SortedMap struct {
|
||||||
|
Key []reflect.Value
|
||||||
|
Value []reflect.Value
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o *SortedMap) Len() int { return len(o.Key) }
|
||||||
|
func (o *SortedMap) Less(i, j int) bool { return compare(o.Key[i], o.Key[j]) < 0 }
|
||||||
|
func (o *SortedMap) Swap(i, j int) {
|
||||||
|
o.Key[i], o.Key[j] = o.Key[j], o.Key[i]
|
||||||
|
o.Value[i], o.Value[j] = o.Value[j], o.Value[i]
|
||||||
|
}
|
||||||
|
|
||||||
|
// Sort accepts a map and returns a SortedMap that has the same keys and
|
||||||
|
// values but in a stable sorted order according to the keys, modulo issues
|
||||||
|
// raised by unorderable key values such as NaNs.
|
||||||
|
//
|
||||||
|
// The ordering rules are more general than with Go's < operator:
|
||||||
|
//
|
||||||
|
// - when applicable, nil compares low
|
||||||
|
// - ints, floats, and strings order by <
|
||||||
|
// - NaN compares less than non-NaN floats
|
||||||
|
// - bool compares false before true
|
||||||
|
// - complex compares real, then imag
|
||||||
|
// - pointers compare by machine address
|
||||||
|
// - channel values compare by machine address
|
||||||
|
// - structs compare each field in turn
|
||||||
|
// - arrays compare each element in turn.
|
||||||
|
// Otherwise identical arrays compare by length.
|
||||||
|
// - interface values compare first by reflect.Type describing the concrete type
|
||||||
|
// and then by concrete value as described in the previous rules.
|
||||||
|
func Sort(mapValue reflect.Value) *SortedMap {
|
||||||
|
if mapValue.Type().Kind() != reflect.Map {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
// Note: this code is arranged to not panic even in the presence
|
||||||
|
// of a concurrent map update. The runtime is responsible for
|
||||||
|
// yelling loudly if that happens. See issue 33275.
|
||||||
|
n := mapValue.Len()
|
||||||
|
key := make([]reflect.Value, 0, n)
|
||||||
|
value := make([]reflect.Value, 0, n)
|
||||||
|
iter := mapValue.MapRange()
|
||||||
|
for iter.Next() {
|
||||||
|
key = append(key, iter.Key())
|
||||||
|
value = append(value, iter.Value())
|
||||||
|
}
|
||||||
|
sorted := &SortedMap{
|
||||||
|
Key: key,
|
||||||
|
Value: value,
|
||||||
|
}
|
||||||
|
sort.Stable(sorted)
|
||||||
|
return sorted
|
||||||
|
}
|
||||||
|
|
||||||
|
// compare compares two values of the same type. It returns -1, 0, 1
|
||||||
|
// according to whether a > b (1), a == b (0), or a < b (-1).
|
||||||
|
// If the types differ, it returns -1.
|
||||||
|
// See the comment on Sort for the comparison rules.
|
||||||
|
func compare(aVal, bVal reflect.Value) int {
|
||||||
|
aType, bType := aVal.Type(), bVal.Type()
|
||||||
|
if aType != bType {
|
||||||
|
return -1 // No good answer possible, but don't return 0: they're not equal.
|
||||||
|
}
|
||||||
|
switch aVal.Kind() {
|
||||||
|
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
|
||||||
|
a, b := aVal.Int(), bVal.Int()
|
||||||
|
switch {
|
||||||
|
case a < b:
|
||||||
|
return -1
|
||||||
|
case a > b:
|
||||||
|
return 1
|
||||||
|
default:
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr:
|
||||||
|
a, b := aVal.Uint(), bVal.Uint()
|
||||||
|
switch {
|
||||||
|
case a < b:
|
||||||
|
return -1
|
||||||
|
case a > b:
|
||||||
|
return 1
|
||||||
|
default:
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
case reflect.String:
|
||||||
|
a, b := aVal.String(), bVal.String()
|
||||||
|
switch {
|
||||||
|
case a < b:
|
||||||
|
return -1
|
||||||
|
case a > b:
|
||||||
|
return 1
|
||||||
|
default:
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
case reflect.Float32, reflect.Float64:
|
||||||
|
return floatCompare(aVal.Float(), bVal.Float())
|
||||||
|
case reflect.Complex64, reflect.Complex128:
|
||||||
|
a, b := aVal.Complex(), bVal.Complex()
|
||||||
|
if c := floatCompare(real(a), real(b)); c != 0 {
|
||||||
|
return c
|
||||||
|
}
|
||||||
|
return floatCompare(imag(a), imag(b))
|
||||||
|
case reflect.Bool:
|
||||||
|
a, b := aVal.Bool(), bVal.Bool()
|
||||||
|
switch {
|
||||||
|
case a == b:
|
||||||
|
return 0
|
||||||
|
case a:
|
||||||
|
return 1
|
||||||
|
default:
|
||||||
|
return -1
|
||||||
|
}
|
||||||
|
case reflect.Pointer, reflect.UnsafePointer:
|
||||||
|
a, b := aVal.Pointer(), bVal.Pointer()
|
||||||
|
switch {
|
||||||
|
case a < b:
|
||||||
|
return -1
|
||||||
|
case a > b:
|
||||||
|
return 1
|
||||||
|
default:
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
case reflect.Chan:
|
||||||
|
if c, ok := nilCompare(aVal, bVal); ok {
|
||||||
|
return c
|
||||||
|
}
|
||||||
|
ap, bp := aVal.Pointer(), bVal.Pointer()
|
||||||
|
switch {
|
||||||
|
case ap < bp:
|
||||||
|
return -1
|
||||||
|
case ap > bp:
|
||||||
|
return 1
|
||||||
|
default:
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
case reflect.Struct:
|
||||||
|
for i := 0; i < aVal.NumField(); i++ {
|
||||||
|
if c := compare(aVal.Field(i), bVal.Field(i)); c != 0 {
|
||||||
|
return c
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 0
|
||||||
|
case reflect.Array:
|
||||||
|
for i := 0; i < aVal.Len(); i++ {
|
||||||
|
if c := compare(aVal.Index(i), bVal.Index(i)); c != 0 {
|
||||||
|
return c
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 0
|
||||||
|
case reflect.Interface:
|
||||||
|
if c, ok := nilCompare(aVal, bVal); ok {
|
||||||
|
return c
|
||||||
|
}
|
||||||
|
c := compare(reflect.ValueOf(aVal.Elem().Type()), reflect.ValueOf(bVal.Elem().Type()))
|
||||||
|
if c != 0 {
|
||||||
|
return c
|
||||||
|
}
|
||||||
|
return compare(aVal.Elem(), bVal.Elem())
|
||||||
|
default:
|
||||||
|
// Certain types cannot appear as keys (maps, funcs, slices), but be explicit.
|
||||||
|
panic("bad type in compare: " + aType.String())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// nilCompare checks whether either value is nil. If not, the boolean is false.
|
||||||
|
// If either value is nil, the boolean is true and the integer is the comparison
|
||||||
|
// value. The comparison is defined to be 0 if both are nil, otherwise the one
|
||||||
|
// nil value compares low. Both arguments must represent a chan, func,
|
||||||
|
// interface, map, pointer, or slice.
|
||||||
|
func nilCompare(aVal, bVal reflect.Value) (int, bool) {
|
||||||
|
if aVal.IsNil() {
|
||||||
|
if bVal.IsNil() {
|
||||||
|
return 0, true
|
||||||
|
}
|
||||||
|
return -1, true
|
||||||
|
}
|
||||||
|
if bVal.IsNil() {
|
||||||
|
return 1, true
|
||||||
|
}
|
||||||
|
return 0, false
|
||||||
|
}
|
||||||
|
|
||||||
|
// floatCompare compares two floating-point values. NaNs compare low.
|
||||||
|
func floatCompare(a, b float64) int {
|
||||||
|
switch {
|
||||||
|
case isNaN(a):
|
||||||
|
return -1 // No good answer if b is a NaN so don't bother checking.
|
||||||
|
case isNaN(b):
|
||||||
|
return 1
|
||||||
|
case a < b:
|
||||||
|
return -1
|
||||||
|
case a > b:
|
||||||
|
return 1
|
||||||
|
}
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
func isNaN(a float64) bool {
|
||||||
|
return a != a
|
||||||
|
}
|
||||||
@@ -18,6 +18,9 @@ package big
|
|||||||
|
|
||||||
// llgo:skipall
|
// llgo:skipall
|
||||||
import (
|
import (
|
||||||
|
"math/rand"
|
||||||
|
|
||||||
|
"github.com/goplus/llgo/c"
|
||||||
"github.com/goplus/llgo/c/openssl"
|
"github.com/goplus/llgo/c/openssl"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -81,9 +84,34 @@ func NewInt(x int64) *Int {
|
|||||||
return z.SetInt64(x)
|
return z.SetInt64(x)
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
// Set sets z to x and returns z.
|
// Set sets z to x and returns z.
|
||||||
func (z *Int) Set(x *Int) *Int {
|
func (z *Int) Set(x *Int) *Int {
|
||||||
|
if z != x {
|
||||||
|
a := (*openssl.BIGNUM)(z)
|
||||||
|
b := (*openssl.BIGNUM)(x)
|
||||||
|
a.Copy(b)
|
||||||
|
}
|
||||||
|
return z
|
||||||
|
}
|
||||||
|
|
||||||
|
// Abs sets z to |x| (the absolute value of x) and returns z.
|
||||||
|
func (z *Int) Abs(x *Int) *Int {
|
||||||
|
z.Set(x)
|
||||||
|
a := (*openssl.BIGNUM)(z)
|
||||||
|
a.SetNegative(0)
|
||||||
|
return z
|
||||||
|
}
|
||||||
|
|
||||||
|
// Neg sets z to -x and returns z.
|
||||||
|
func (z *Int) Neg(x *Int) *Int {
|
||||||
|
z.Set(x)
|
||||||
|
a := (*openssl.BIGNUM)(z)
|
||||||
|
if a.IsNegative() != 0 {
|
||||||
|
a.SetNegative(0)
|
||||||
|
} else {
|
||||||
|
a.SetNegative(1)
|
||||||
|
}
|
||||||
|
return z
|
||||||
}
|
}
|
||||||
|
|
||||||
// Bits provides raw (unchecked but fast) access to x by returning its
|
// Bits provides raw (unchecked but fast) access to x by returning its
|
||||||
@@ -92,6 +120,7 @@ func (z *Int) Set(x *Int) *Int {
|
|||||||
// Bits is intended to support implementation of missing low-level Int
|
// Bits is intended to support implementation of missing low-level Int
|
||||||
// functionality outside this package; it should be avoided otherwise.
|
// functionality outside this package; it should be avoided otherwise.
|
||||||
func (x *Int) Bits() []Word {
|
func (x *Int) Bits() []Word {
|
||||||
|
panic("todo big.Bits")
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetBits provides raw (unchecked but fast) access to z by setting its
|
// SetBits provides raw (unchecked but fast) access to z by setting its
|
||||||
@@ -100,17 +129,9 @@ func (x *Int) Bits() []Word {
|
|||||||
// SetBits is intended to support implementation of missing low-level Int
|
// SetBits is intended to support implementation of missing low-level Int
|
||||||
// functionality outside this package; it should be avoided otherwise.
|
// functionality outside this package; it should be avoided otherwise.
|
||||||
func (z *Int) SetBits(abs []Word) *Int {
|
func (z *Int) SetBits(abs []Word) *Int {
|
||||||
|
panic("todo big.SetBits")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Abs sets z to |x| (the absolute value of x) and returns z.
|
|
||||||
func (z *Int) Abs(x *Int) *Int {
|
|
||||||
}
|
|
||||||
|
|
||||||
// Neg sets z to -x and returns z.
|
|
||||||
func (z *Int) Neg(x *Int) *Int {
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
|
|
||||||
// Add sets z to the sum x+y and returns z.
|
// Add sets z to the sum x+y and returns z.
|
||||||
func (z *Int) Add(x, y *Int) *Int {
|
func (z *Int) Add(x, y *Int) *Int {
|
||||||
(*openssl.BIGNUM)(z).Add((*openssl.BIGNUM)(x), (*openssl.BIGNUM)(y))
|
(*openssl.BIGNUM)(z).Add((*openssl.BIGNUM)(x), (*openssl.BIGNUM)(y))
|
||||||
@@ -123,31 +144,35 @@ func (z *Int) Sub(x, y *Int) *Int {
|
|||||||
return z
|
return z
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
// Mul sets z to the product x*y and returns z.
|
// Mul sets z to the product x*y and returns z.
|
||||||
func (z *Int) Mul(x, y *Int) *Int {
|
func (z *Int) Mul(x, y *Int) *Int {
|
||||||
|
panic("todo big.Mul")
|
||||||
}
|
}
|
||||||
|
|
||||||
// MulRange sets z to the product of all integers
|
// MulRange sets z to the product of all integers
|
||||||
// in the range [a, b] inclusively and returns z.
|
// in the range [a, b] inclusively and returns z.
|
||||||
// If a > b (empty range), the result is 1.
|
// If a > b (empty range), the result is 1.
|
||||||
func (z *Int) MulRange(a, b int64) *Int {
|
func (z *Int) MulRange(a, b int64) *Int {
|
||||||
|
panic("todo big.MulRange")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Binomial sets z to the binomial coefficient C(n, k) and returns z.
|
// Binomial sets z to the binomial coefficient C(n, k) and returns z.
|
||||||
func (z *Int) Binomial(n, k int64) *Int {
|
func (z *Int) Binomial(n, k int64) *Int {
|
||||||
|
panic("todo big.Binomial")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Quo sets z to the quotient x/y for y != 0 and returns z.
|
// Quo sets z to the quotient x/y for y != 0 and returns z.
|
||||||
// If y == 0, a division-by-zero run-time panic occurs.
|
// If y == 0, a division-by-zero run-time panic occurs.
|
||||||
// Quo implements truncated division (like Go); see QuoRem for more details.
|
// Quo implements truncated division (like Go); see QuoRem for more details.
|
||||||
func (z *Int) Quo(x, y *Int) *Int {
|
func (z *Int) Quo(x, y *Int) *Int {
|
||||||
|
panic("todo big.Quo")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Rem sets z to the remainder x%y for y != 0 and returns z.
|
// Rem sets z to the remainder x%y for y != 0 and returns z.
|
||||||
// If y == 0, a division-by-zero run-time panic occurs.
|
// If y == 0, a division-by-zero run-time panic occurs.
|
||||||
// Rem implements truncated modulus (like Go); see QuoRem for more details.
|
// Rem implements truncated modulus (like Go); see QuoRem for more details.
|
||||||
func (z *Int) Rem(x, y *Int) *Int {
|
func (z *Int) Rem(x, y *Int) *Int {
|
||||||
|
panic("todo big.Rem")
|
||||||
}
|
}
|
||||||
|
|
||||||
// QuoRem sets z to the quotient x/y and r to the remainder x%y
|
// QuoRem sets z to the quotient x/y and r to the remainder x%y
|
||||||
@@ -162,18 +187,21 @@ func (z *Int) Rem(x, y *Int) *Int {
|
|||||||
// (See Daan Leijen, “Division and Modulus for Computer Scientists”.)
|
// (See Daan Leijen, “Division and Modulus for Computer Scientists”.)
|
||||||
// See DivMod for Euclidean division and modulus (unlike Go).
|
// See DivMod for Euclidean division and modulus (unlike Go).
|
||||||
func (z *Int) QuoRem(x, y, r *Int) (*Int, *Int) {
|
func (z *Int) QuoRem(x, y, r *Int) (*Int, *Int) {
|
||||||
|
panic("todo big.QuoRem")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Div sets z to the quotient x/y for y != 0 and returns z.
|
// Div sets z to the quotient x/y for y != 0 and returns z.
|
||||||
// If y == 0, a division-by-zero run-time panic occurs.
|
// If y == 0, a division-by-zero run-time panic occurs.
|
||||||
// Div implements Euclidean division (unlike Go); see DivMod for more details.
|
// Div implements Euclidean division (unlike Go); see DivMod for more details.
|
||||||
func (z *Int) Div(x, y *Int) *Int {
|
func (z *Int) Div(x, y *Int) *Int {
|
||||||
|
panic("todo big.Div")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Mod sets z to the modulus x%y for y != 0 and returns z.
|
// Mod sets z to the modulus x%y for y != 0 and returns z.
|
||||||
// If y == 0, a division-by-zero run-time panic occurs.
|
// If y == 0, a division-by-zero run-time panic occurs.
|
||||||
// Mod implements Euclidean modulus (unlike Go); see DivMod for more details.
|
// Mod implements Euclidean modulus (unlike Go); see DivMod for more details.
|
||||||
func (z *Int) Mod(x, y *Int) *Int {
|
func (z *Int) Mod(x, y *Int) *Int {
|
||||||
|
panic("todo big.Mod")
|
||||||
}
|
}
|
||||||
|
|
||||||
// DivMod sets z to the quotient x div y and m to the modulus x mod y
|
// DivMod sets z to the quotient x div y and m to the modulus x mod y
|
||||||
@@ -191,8 +219,8 @@ func (z *Int) Mod(x, y *Int) *Int {
|
|||||||
// ACM press.)
|
// ACM press.)
|
||||||
// See QuoRem for T-division and modulus (like Go).
|
// See QuoRem for T-division and modulus (like Go).
|
||||||
func (z *Int) DivMod(x, y, m *Int) (*Int, *Int) {
|
func (z *Int) DivMod(x, y, m *Int) (*Int, *Int) {
|
||||||
|
panic("big.DivMod")
|
||||||
}
|
}
|
||||||
*/
|
|
||||||
|
|
||||||
// Cmp compares x and y and returns:
|
// Cmp compares x and y and returns:
|
||||||
//
|
//
|
||||||
@@ -212,29 +240,35 @@ func (x *Int) CmpAbs(y *Int) int {
|
|||||||
return int((*openssl.BIGNUM)(x).Ucmp((*openssl.BIGNUM)(y)))
|
return int((*openssl.BIGNUM)(x).Ucmp((*openssl.BIGNUM)(y)))
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
// Int64 returns the int64 representation of x.
|
// Int64 returns the int64 representation of x.
|
||||||
// If x cannot be represented in an int64, the result is undefined.
|
// If x cannot be represented in an int64, the result is undefined.
|
||||||
func (x *Int) Int64() int64 {
|
func (x *Int) Int64() int64 {
|
||||||
|
panic("todo big.Int64")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Uint64 returns the uint64 representation of x.
|
// Uint64 returns the uint64 representation of x.
|
||||||
// If x cannot be represented in a uint64, the result is undefined.
|
// If x cannot be represented in a uint64, the result is undefined.
|
||||||
func (x *Int) Uint64() uint64 {
|
func (x *Int) Uint64() uint64 {
|
||||||
|
panic("todo big.Uint64")
|
||||||
}
|
}
|
||||||
|
|
||||||
// IsInt64 reports whether x can be represented as an int64.
|
// IsInt64 reports whether x can be represented as an int64.
|
||||||
func (x *Int) IsInt64() bool {
|
func (x *Int) IsInt64() bool {
|
||||||
|
panic("todo big.IsInt64")
|
||||||
}
|
}
|
||||||
|
|
||||||
// IsUint64 reports whether x can be represented as a uint64.
|
// IsUint64 reports whether x can be represented as a uint64.
|
||||||
func (x *Int) IsUint64() bool {
|
func (x *Int) IsUint64() bool {
|
||||||
|
panic("todo big.IsUint64")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Float64 returns the float64 value nearest x,
|
// Float64 returns the float64 value nearest x,
|
||||||
// and an indication of any rounding that occurred.
|
// and an indication of any rounding that occurred.
|
||||||
// TODO(xsw):
|
// TODO(xsw):
|
||||||
// func (x *Int) Float64() (float64, Accuracy)
|
/*
|
||||||
|
func (x *Int) Float64() (float64, Accuracy) {
|
||||||
|
panic("todo big.Float64")
|
||||||
|
}*/
|
||||||
|
|
||||||
// SetString sets z to the value of s, interpreted in the given base,
|
// SetString sets z to the value of s, interpreted in the given base,
|
||||||
// and returns z and a boolean indicating success. The entire string
|
// and returns z and a boolean indicating success. The entire string
|
||||||
@@ -259,17 +293,20 @@ func (x *Int) IsUint64() bool {
|
|||||||
// are no other errors. If base != 0, underscores are not recognized
|
// are no other errors. If base != 0, underscores are not recognized
|
||||||
// and act like any other character that is not a valid digit.
|
// and act like any other character that is not a valid digit.
|
||||||
func (z *Int) SetString(s string, base int) (*Int, bool) {
|
func (z *Int) SetString(s string, base int) (*Int, bool) {
|
||||||
|
panic("todo big.SetString")
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetBytes interprets buf as the bytes of a big-endian unsigned
|
// SetBytes interprets buf as the bytes of a big-endian unsigned
|
||||||
// integer, sets z to that value, and returns z.
|
// integer, sets z to that value, and returns z.
|
||||||
func (z *Int) SetBytes(buf []byte) *Int {
|
func (z *Int) SetBytes(buf []byte) *Int {
|
||||||
|
panic("todo big.SetBytes")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Bytes returns the absolute value of x as a big-endian byte slice.
|
// Bytes returns the absolute value of x as a big-endian byte slice.
|
||||||
//
|
//
|
||||||
// To use a fixed length slice, or a preallocated one, use FillBytes.
|
// To use a fixed length slice, or a preallocated one, use FillBytes.
|
||||||
func (x *Int) Bytes() []byte {
|
func (x *Int) Bytes() []byte {
|
||||||
|
panic("todo big.Bytes")
|
||||||
}
|
}
|
||||||
|
|
||||||
// FillBytes sets buf to the absolute value of x, storing it as a zero-extended
|
// FillBytes sets buf to the absolute value of x, storing it as a zero-extended
|
||||||
@@ -277,18 +314,20 @@ func (x *Int) Bytes() []byte {
|
|||||||
//
|
//
|
||||||
// If the absolute value of x doesn't fit in buf, FillBytes will panic.
|
// If the absolute value of x doesn't fit in buf, FillBytes will panic.
|
||||||
func (x *Int) FillBytes(buf []byte) []byte {
|
func (x *Int) FillBytes(buf []byte) []byte {
|
||||||
|
panic("todo big.FillBytes")
|
||||||
}
|
}
|
||||||
|
|
||||||
// BitLen returns the length of the absolute value of x in bits.
|
// BitLen returns the length of the absolute value of x in bits.
|
||||||
// The bit length of 0 is 0.
|
// The bit length of 0 is 0.
|
||||||
func (x *Int) BitLen() int {
|
func (x *Int) BitLen() int {
|
||||||
|
panic("todo big.BitLen")
|
||||||
}
|
}
|
||||||
|
|
||||||
// TrailingZeroBits returns the number of consecutive least significant zero
|
// TrailingZeroBits returns the number of consecutive least significant zero
|
||||||
// bits of |x|.
|
// bits of |x|.
|
||||||
func (x *Int) TrailingZeroBits() uint {
|
func (x *Int) TrailingZeroBits() uint {
|
||||||
|
panic("todo big.TrailingZeroBits")
|
||||||
}
|
}
|
||||||
*/
|
|
||||||
|
|
||||||
// Exp sets z = x**y mod |m| (i.e. the sign of m is ignored), and returns z.
|
// Exp sets z = x**y mod |m| (i.e. the sign of m is ignored), and returns z.
|
||||||
// If m == nil or m == 0, z = x**y unless y <= 0 then z = 1. If m != 0, y < 0,
|
// If m == nil or m == 0, z = x**y unless y <= 0 then z = 1. If m != 0, y < 0,
|
||||||
@@ -308,7 +347,6 @@ func (z *Int) Exp(x, y, m *Int) *Int {
|
|||||||
return z
|
return z
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
// GCD sets z to the greatest common divisor of a and b and returns z.
|
// GCD sets z to the greatest common divisor of a and b and returns z.
|
||||||
// If x or y are not nil, GCD sets their value such that z = a*x + b*y.
|
// If x or y are not nil, GCD sets their value such that z = a*x + b*y.
|
||||||
//
|
//
|
||||||
@@ -321,6 +359,7 @@ func (z *Int) Exp(x, y, m *Int) *Int {
|
|||||||
//
|
//
|
||||||
// If a != 0 and b == 0, GCD sets z = |a|, x = sign(a) * 1, y = 0.
|
// If a != 0 and b == 0, GCD sets z = |a|, x = sign(a) * 1, y = 0.
|
||||||
func (z *Int) GCD(x, y, a, b *Int) *Int {
|
func (z *Int) GCD(x, y, a, b *Int) *Int {
|
||||||
|
panic("todo big.GCD")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Rand sets z to a pseudo-random number in [0, n) and returns z.
|
// Rand sets z to a pseudo-random number in [0, n) and returns z.
|
||||||
@@ -328,6 +367,7 @@ func (z *Int) GCD(x, y, a, b *Int) *Int {
|
|||||||
// As this uses the math/rand package, it must not be used for
|
// As this uses the math/rand package, it must not be used for
|
||||||
// security-sensitive work. Use crypto/rand.Int instead.
|
// security-sensitive work. Use crypto/rand.Int instead.
|
||||||
func (z *Int) Rand(rnd *rand.Rand, n *Int) *Int {
|
func (z *Int) Rand(rnd *rand.Rand, n *Int) *Int {
|
||||||
|
panic("todo big.Rand")
|
||||||
}
|
}
|
||||||
|
|
||||||
// ModInverse sets z to the multiplicative inverse of g in the ring ℤ/nℤ
|
// ModInverse sets z to the multiplicative inverse of g in the ring ℤ/nℤ
|
||||||
@@ -335,11 +375,13 @@ func (z *Int) Rand(rnd *rand.Rand, n *Int) *Int {
|
|||||||
// inverse in the ring ℤ/nℤ. In this case, z is unchanged and the return value
|
// inverse in the ring ℤ/nℤ. In this case, z is unchanged and the return value
|
||||||
// is nil. If n == 0, a division-by-zero run-time panic occurs.
|
// is nil. If n == 0, a division-by-zero run-time panic occurs.
|
||||||
func (z *Int) ModInverse(g, n *Int) *Int {
|
func (z *Int) ModInverse(g, n *Int) *Int {
|
||||||
|
panic("todo big.ModInverse")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Jacobi returns the Jacobi symbol (x/y), either +1, -1, or 0.
|
// Jacobi returns the Jacobi symbol (x/y), either +1, -1, or 0.
|
||||||
// The y argument must be an odd integer.
|
// The y argument must be an odd integer.
|
||||||
func Jacobi(x, y *Int) int {
|
func Jacobi(x, y *Int) int {
|
||||||
|
panic("todo big.Jacobi")
|
||||||
}
|
}
|
||||||
|
|
||||||
// ModSqrt sets z to a square root of x mod p if such a square root exists, and
|
// ModSqrt sets z to a square root of x mod p if such a square root exists, and
|
||||||
@@ -347,19 +389,29 @@ func Jacobi(x, y *Int) int {
|
|||||||
// ModSqrt leaves z unchanged and returns nil. This function panics if p is
|
// ModSqrt leaves z unchanged and returns nil. This function panics if p is
|
||||||
// not an odd integer, its behavior is undefined if p is odd but not prime.
|
// not an odd integer, its behavior is undefined if p is odd but not prime.
|
||||||
func (z *Int) ModSqrt(x, p *Int) *Int {
|
func (z *Int) ModSqrt(x, p *Int) *Int {
|
||||||
|
panic("todo big.ModSqrt")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Lsh sets z = x << n and returns z.
|
// Lsh sets z = x << n and returns z.
|
||||||
func (z *Int) Lsh(x *Int, n uint) *Int {
|
func (z *Int) Lsh(x *Int, n uint) *Int {
|
||||||
|
a := (*openssl.BIGNUM)(z)
|
||||||
|
b := (*openssl.BIGNUM)(x)
|
||||||
|
a.Lshift(b, c.Int(n))
|
||||||
|
return z
|
||||||
}
|
}
|
||||||
|
|
||||||
// Rsh sets z = x >> n and returns z.
|
// Rsh sets z = x >> n and returns z.
|
||||||
func (z *Int) Rsh(x *Int, n uint) *Int {
|
func (z *Int) Rsh(x *Int, n uint) *Int {
|
||||||
|
a := (*openssl.BIGNUM)(z)
|
||||||
|
b := (*openssl.BIGNUM)(x)
|
||||||
|
a.Rshift(b, c.Int(n))
|
||||||
|
return z
|
||||||
}
|
}
|
||||||
|
|
||||||
// Bit returns the value of the i'th bit of x. That is, it
|
// Bit returns the value of the i'th bit of x. That is, it
|
||||||
// returns (x>>i)&1. The bit index i must be >= 0.
|
// returns (x>>i)&1. The bit index i must be >= 0.
|
||||||
func (x *Int) Bit(i int) uint {
|
func (x *Int) Bit(i int) uint {
|
||||||
|
panic("todo big.Bit")
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetBit sets z to x, with x's i'th bit set to b (0 or 1).
|
// SetBit sets z to x, with x's i'th bit set to b (0 or 1).
|
||||||
@@ -367,32 +419,38 @@ func (x *Int) Bit(i int) uint {
|
|||||||
// if b is 0 SetBit sets z = x &^ (1 << i). If b is not 0 or 1,
|
// if b is 0 SetBit sets z = x &^ (1 << i). If b is not 0 or 1,
|
||||||
// SetBit will panic.
|
// SetBit will panic.
|
||||||
func (z *Int) SetBit(x *Int, i int, b uint) *Int {
|
func (z *Int) SetBit(x *Int, i int, b uint) *Int {
|
||||||
|
panic("todo big.SetBit")
|
||||||
}
|
}
|
||||||
|
|
||||||
// And sets z = x & y and returns z.
|
// And sets z = x & y and returns z.
|
||||||
func (z *Int) And(x, y *Int) *Int {
|
func (z *Int) And(x, y *Int) *Int {
|
||||||
|
panic("todo big.And")
|
||||||
}
|
}
|
||||||
|
|
||||||
// AndNot sets z = x &^ y and returns z.
|
// AndNot sets z = x &^ y and returns z.
|
||||||
func (z *Int) AndNot(x, y *Int) *Int {
|
func (z *Int) AndNot(x, y *Int) *Int {
|
||||||
|
panic("todo big.AndNot")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Or sets z = x | y and returns z.
|
// Or sets z = x | y and returns z.
|
||||||
func (z *Int) Or(x, y *Int) *Int {
|
func (z *Int) Or(x, y *Int) *Int {
|
||||||
|
panic("todo big.Or")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Xor sets z = x ^ y and returns z.
|
// Xor sets z = x ^ y and returns z.
|
||||||
func (z *Int) Xor(x, y *Int) *Int {
|
func (z *Int) Xor(x, y *Int) *Int {
|
||||||
|
panic("todo big.Xor")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Not sets z = ^x and returns z.
|
// Not sets z = ^x and returns z.
|
||||||
func (z *Int) Not(x *Int) *Int {
|
func (z *Int) Not(x *Int) *Int {
|
||||||
|
panic("todo big.Not")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Sqrt sets z to ⌊√x⌋, the largest integer such that z² ≤ x, and returns z.
|
// Sqrt sets z to ⌊√x⌋, the largest integer such that z² ≤ x, and returns z.
|
||||||
// It panics if x is negative.
|
// It panics if x is negative.
|
||||||
func (z *Int) Sqrt(x *Int) *Int {
|
func (z *Int) Sqrt(x *Int) *Int {
|
||||||
|
panic("todo big.Sqrt")
|
||||||
}
|
}
|
||||||
*/
|
|
||||||
|
|
||||||
// -----------------------------------------------------------------------------
|
// -----------------------------------------------------------------------------
|
||||||
|
|||||||
@@ -70,14 +70,16 @@ func (w WaitStatus) StopSignal() Signal {
|
|||||||
return Signal(w>>shift) & 0xFF
|
return Signal(w>>shift) & 0xFF
|
||||||
}
|
}
|
||||||
|
|
||||||
/* TODO(xsw):
|
|
||||||
func (w WaitStatus) TrapCause() int {
|
func (w WaitStatus) TrapCause() int {
|
||||||
|
/*
|
||||||
if w.StopSignal() != SIGTRAP {
|
if w.StopSignal() != SIGTRAP {
|
||||||
return -1
|
return -1
|
||||||
}
|
}
|
||||||
|
|
||||||
return int(w>>shift) >> 8
|
return int(w>>shift) >> 8
|
||||||
|
*/
|
||||||
|
panic("todo: syscall.WaitStatus.TrapCause")
|
||||||
}
|
}
|
||||||
*/
|
|
||||||
|
|
||||||
func Wait4(pid int, wstatus *WaitStatus, options int, rusage *syscall.Rusage) (wpid int, err error) {
|
func Wait4(pid int, wstatus *WaitStatus, options int, rusage *syscall.Rusage) (wpid int, err error) {
|
||||||
var status c.Int
|
var status c.Int
|
||||||
|
|||||||
@@ -414,6 +414,9 @@ func (b Builder) abiType(t types.Type) Expr {
|
|||||||
b.loadType(t.Elem())
|
b.loadType(t.Elem())
|
||||||
case *types.Array:
|
case *types.Array:
|
||||||
b.abiType(t.Elem())
|
b.abiType(t.Elem())
|
||||||
|
case *types.Map:
|
||||||
|
b.abiType(t.Key())
|
||||||
|
b.abiType(t.Elem())
|
||||||
}
|
}
|
||||||
g := b.loadType(t)
|
g := b.loadType(t)
|
||||||
return b.Load(g.Expr)
|
return b.Load(g.Expr)
|
||||||
|
|||||||
Reference in New Issue
Block a user