Files
llgo/chore/_xtool/llcppsymg/parse/parse.go

168 lines
3.7 KiB
Go
Raw Normal View History

package parse
2024-07-26 13:45:08 +08:00
import (
"errors"
2024-07-26 13:45:08 +08:00
"strconv"
2024-08-08 14:14:20 +08:00
"strings"
2024-07-26 13:45:08 +08:00
"unsafe"
"github.com/goplus/llgo/c"
"github.com/goplus/llgo/c/clang"
)
type Context struct {
namespaceName string
className string
2024-08-08 14:14:20 +08:00
prefixes []string
symbolMap map[string]string
currentFile string
2024-08-08 14:14:20 +08:00
nameCounts map[string]int
2024-07-26 13:45:08 +08:00
}
2024-08-08 14:14:20 +08:00
func newContext(prefixes []string) *Context {
2024-07-26 13:45:08 +08:00
return &Context{
2024-08-08 14:14:20 +08:00
prefixes: prefixes,
symbolMap: make(map[string]string),
nameCounts: make(map[string]int),
2024-07-26 13:45:08 +08:00
}
}
func (c *Context) setNamespaceName(name string) {
c.namespaceName = name
}
func (c *Context) setClassName(name string) {
c.className = name
}
func (c *Context) setCurrentFile(filename string) {
2024-07-26 17:09:26 +08:00
c.currentFile = filename
2024-07-26 15:56:47 +08:00
}
2024-08-08 14:14:20 +08:00
func (c *Context) removePrefix(str string) string {
for _, prefix := range c.prefixes {
if strings.HasPrefix(str, prefix) {
return strings.TrimPrefix(str, prefix)
}
2024-07-26 13:45:08 +08:00
}
2024-08-08 14:14:20 +08:00
return str
}
2024-07-26 13:45:08 +08:00
2024-08-08 14:14:20 +08:00
func (c *Context) genGoName(name string) string {
class := c.removePrefix(c.className)
name = c.removePrefix(name)
2024-07-26 14:37:26 +08:00
2024-08-08 14:14:20 +08:00
var baseName string
if class == "" {
baseName = name
} else {
baseName = c.genMethodName(class, name)
2024-07-26 14:37:26 +08:00
}
2024-07-26 13:45:08 +08:00
2024-08-08 14:14:20 +08:00
return c.addSuffix(baseName)
}
2024-07-26 13:45:08 +08:00
2024-08-08 14:14:20 +08:00
func (c *Context) genMethodName(class, name string) string {
prefix := "(*" + class + ")."
if class == name {
return prefix + "Init"
}
if name == "~"+class {
return prefix + "Dispose"
2024-07-26 13:45:08 +08:00
}
2024-08-08 14:14:20 +08:00
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)
2024-07-26 13:45:08 +08:00
}
2024-08-08 14:14:20 +08:00
return name
}
2024-07-26 13:45:08 +08:00
2024-08-08 14:14:20 +08:00
var context = newContext([]string{})
func collectFuncInfo(cursor clang.Cursor) {
cursorStr := cursor.String()
symbol := cursor.Mangling()
2024-07-26 13:45:08 +08:00
2024-08-08 14:14:20 +08:00
name := c.GoString(cursorStr.CStr())
symbolName := c.GoString(symbol.CStr())
if len(symbolName) >= 1 && symbolName[0] == '_' {
symbolName = symbolName[1:]
2024-07-26 13:45:08 +08:00
}
2024-08-08 14:14:20 +08:00
defer symbol.Dispose()
defer cursorStr.Dispose()
2024-07-26 13:45:08 +08:00
2024-08-08 14:14:20 +08:00
goName := context.genGoName(name)
context.symbolMap[symbolName] = goName
2024-07-26 13:45:08 +08:00
}
func visit(cursor, parent clang.Cursor, clientData c.Pointer) clang.ChildVisitResult {
2024-08-08 14:14:20 +08:00
if cursor.Kind == clang.CursorNamespace {
2024-07-26 13:45:08 +08:00
nameStr := cursor.String()
2024-08-08 14:14:20 +08:00
defer nameStr.Dispose()
2024-07-26 13:45:08 +08:00
context.setNamespaceName(c.GoString(nameStr.CStr()))
clang.VisitChildren(cursor, visit, nil)
context.setNamespaceName("")
2024-08-08 14:14:20 +08:00
} else if cursor.Kind == clang.CursorClassDecl {
2024-07-26 13:45:08 +08:00
nameStr := cursor.String()
2024-08-08 14:14:20 +08:00
defer nameStr.Dispose()
2024-07-26 13:45:08 +08:00
context.setClassName(c.GoString(nameStr.CStr()))
clang.VisitChildren(cursor, visit, nil)
context.setClassName("")
2024-08-08 14:14:20 +08:00
} else if cursor.Kind == clang.CursorCXXMethod || cursor.Kind == clang.CursorFunctionDecl || cursor.Kind == clang.CursorConstructor || cursor.Kind == clang.CursorDestructor {
2024-07-26 15:56:47 +08:00
loc := cursor.Location()
var file clang.File
var line, column c.Uint
loc.SpellingLocation(&file, &line, &column, nil)
filename := file.FileName()
if c.Strcmp(filename.CStr(), c.AllocaCStr(context.currentFile)) == 0 {
2024-08-08 14:14:20 +08:00
collectFuncInfo(cursor)
2024-07-26 15:56:47 +08:00
}
defer filename.Dispose()
2024-07-26 13:45:08 +08:00
}
return clang.ChildVisit_Continue
}
2024-08-08 14:14:20 +08:00
func ParseHeaderFile(filepaths []string, prefixes []string) (map[string]string, error) {
2024-07-26 13:45:08 +08:00
index := clang.CreateIndex(0, 0)
args := make([]*c.Char, 3)
args[0] = c.Str("-x")
args[1] = c.Str("c++")
args[2] = c.Str("-std=c++11")
2024-08-08 14:14:20 +08:00
context = newContext(prefixes)
2024-07-26 13:45:08 +08:00
for _, filename := range filepaths {
2024-07-26 17:09:26 +08:00
unit := index.ParseTranslationUnit(
c.AllocaCStr(filename),
2024-07-26 17:09:26 +08:00
unsafe.SliceData(args), 3,
nil, 0,
clang.TranslationUnit_None,
)
if unit == nil {
return nil, errors.New("Unable to parse translation unit for file " + filename)
2024-07-26 17:09:26 +08:00
}
2024-07-26 13:45:08 +08:00
2024-07-26 17:09:26 +08:00
cursor := unit.Cursor()
context.setCurrentFile(filename)
clang.VisitChildren(cursor, visit, nil)
unit.Dispose()
}
2024-07-26 13:45:08 +08:00
index.Dispose()
2024-08-08 14:14:20 +08:00
return context.symbolMap, nil
2024-07-26 13:45:08 +08:00
}