207 lines
4.9 KiB
Go
207 lines
4.9 KiB
Go
/*
|
|
* 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 cl
|
|
|
|
import (
|
|
"bytes"
|
|
"go/ast"
|
|
"go/constant"
|
|
"go/token"
|
|
"go/types"
|
|
"os"
|
|
"strings"
|
|
|
|
llssa "github.com/goplus/llgo/ssa"
|
|
"golang.org/x/tools/go/ssa"
|
|
)
|
|
|
|
type contentLines = [][]byte
|
|
type contentMap = map[string]contentLines
|
|
|
|
func contentOf(m contentMap, file string) (lines contentLines, err error) {
|
|
if v, ok := m[file]; ok {
|
|
return v, nil
|
|
}
|
|
b, err := os.ReadFile(file)
|
|
if err == nil {
|
|
lines = bytes.Split(b, []byte{'\n'})
|
|
m[file] = lines
|
|
}
|
|
return
|
|
}
|
|
|
|
// decl: a package that only contains declarations
|
|
// noinit: a package that does not need to be initialized
|
|
func pkgKind(v string) int {
|
|
switch v {
|
|
case "decl":
|
|
return pkgDeclOnly
|
|
case "noinit":
|
|
return pkgNoInit
|
|
}
|
|
return pkgNormal
|
|
}
|
|
|
|
func (p *context) importPkg(pkg *types.Package, i *pkgInfo) {
|
|
scope := pkg.Scope()
|
|
llpkg, ok := scope.Lookup("LLGoPackage").(*types.Const)
|
|
if !ok {
|
|
return
|
|
}
|
|
if v := llpkg.Val(); v.Kind() == constant.String {
|
|
i.kind = pkgKind(constant.StringVal(v))
|
|
}
|
|
fset := p.fset
|
|
names := scope.Names()
|
|
contents := make(contentMap)
|
|
pkgPath := llssa.PathOf(pkg)
|
|
for _, name := range names {
|
|
if token.IsExported(name) {
|
|
obj := scope.Lookup(name)
|
|
if obj, ok := obj.(*types.Func); ok {
|
|
if pos := obj.Pos(); pos != token.NoPos {
|
|
f := fset.File(pos)
|
|
if fp := f.Position(pos); fp.Line > 2 {
|
|
lines, err := contentOf(contents, fp.Filename)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
if i := fp.Line - 2; i < len(lines) {
|
|
line := string(lines[i])
|
|
p.initLinkname(pkgPath, line)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func (p *context) initFiles(pkgPath string, files []*ast.File) {
|
|
for _, file := range files {
|
|
for _, decl := range file.Decls {
|
|
if decl, ok := decl.(*ast.FuncDecl); ok {
|
|
if decl.Recv == nil {
|
|
if doc := decl.Doc; doc != nil {
|
|
if n := len(doc.List); n > 0 {
|
|
line := doc.List[n-1].Text
|
|
p.initLinkname(pkgPath, line)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func (p *context) initLinkname(pkgPath, line string) {
|
|
const (
|
|
linkname = "//go:linkname "
|
|
)
|
|
if strings.HasPrefix(line, linkname) {
|
|
text := strings.TrimSpace(line[len(linkname):])
|
|
if idx := strings.IndexByte(text, ' '); idx > 0 {
|
|
link := strings.TrimLeft(text[idx+1:], " ")
|
|
if strings.Contains(link, ".") { // eg. C.printf, C.strlen
|
|
name := pkgPath + "." + text[:idx]
|
|
p.link[name] = link[2:]
|
|
} else {
|
|
panic(line + ": no specified call convention. eg. //go:linkname Printf C.printf")
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// func: pkg.name
|
|
// method: (pkg.T).name, (*pkg.T).name
|
|
func funcName(pkg *types.Package, fn *ssa.Function) string {
|
|
sig := fn.Signature
|
|
name := fn.Name()
|
|
if recv := sig.Recv(); recv != nil {
|
|
var tName string
|
|
t := recv.Type()
|
|
if tp, ok := t.(*types.Pointer); ok {
|
|
t, tName = tp.Elem(), "*"
|
|
}
|
|
tName += llssa.NameOf(t.(*types.Named))
|
|
return "(" + tName + ")." + name
|
|
}
|
|
ret := llssa.FullName(pkg, name)
|
|
if ret == "main.main" {
|
|
ret = "main"
|
|
}
|
|
return ret
|
|
}
|
|
|
|
func checkCgo(fnName string) bool {
|
|
return len(fnName) > 4 && fnName[0] == '_' && fnName[2] == 'g' && fnName[3] == 'o' &&
|
|
(fnName[1] == 'C' || fnName[1] == 'c') &&
|
|
(fnName[4] == '_' || strings.HasPrefix(fnName[4:], "Check"))
|
|
}
|
|
|
|
func (p *context) funcName(pkg *types.Package, fn *ssa.Function, ignore bool) (string, bool) {
|
|
name := funcName(pkg, fn)
|
|
if ignore && ignoreName(name) || checkCgo(fn.Name()) {
|
|
return name, false
|
|
}
|
|
if v, ok := p.link[name]; ok {
|
|
return v, true
|
|
}
|
|
return name, true
|
|
}
|
|
|
|
func (p *context) funcOf(fn *ssa.Function) llssa.Function {
|
|
pkgTypes := p.ensureLoaded(fn.Pkg.Pkg)
|
|
pkg := p.pkg
|
|
name, _ := p.funcName(pkgTypes, fn, false)
|
|
if ret := pkg.FuncOf(name); ret != nil {
|
|
return ret
|
|
}
|
|
return pkg.NewFunc(name, fn.Signature)
|
|
}
|
|
|
|
func (p *context) varOf(v *ssa.Global) llssa.Global {
|
|
pkgTypes := p.ensureLoaded(v.Pkg.Pkg)
|
|
pkg := p.pkg
|
|
name := llssa.FullName(pkgTypes, v.Name())
|
|
if ret := pkg.VarOf(name); ret != nil {
|
|
return ret
|
|
}
|
|
return pkg.NewVar(name, v.Type())
|
|
}
|
|
|
|
func (p *context) ensureLoaded(pkgTypes *types.Package) *types.Package {
|
|
if p.goTyps != pkgTypes {
|
|
if _, ok := p.loaded[pkgTypes]; !ok {
|
|
i := &pkgInfo{
|
|
kind: pkgKindByPath(pkgTypes.Path()),
|
|
}
|
|
p.loaded[pkgTypes] = i
|
|
p.importPkg(pkgTypes, i)
|
|
}
|
|
}
|
|
return pkgTypes
|
|
}
|
|
|
|
func pkgKindByPath(pkgPath string) int {
|
|
switch pkgPath {
|
|
case "syscall", "runtime/cgo":
|
|
return pkgNoInit
|
|
}
|
|
return pkgNormal
|
|
}
|