2024-08-06 01:03:23 +00:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
2024-08-06 02:29:12 +00:00
|
|
|
"log"
|
|
|
|
"strings"
|
2024-08-06 01:03:23 +00:00
|
|
|
)
|
|
|
|
|
2024-08-07 06:56:14 +00:00
|
|
|
func parseHeader(inner []interface{}) (*CppParsedHeader, error) {
|
2024-08-06 01:03:23 +00:00
|
|
|
|
2024-08-07 06:56:14 +00:00
|
|
|
var ret CppParsedHeader
|
2024-08-06 01:03:23 +00:00
|
|
|
|
|
|
|
for _, node := range inner {
|
|
|
|
|
|
|
|
node, ok := node.(map[string]interface{})
|
|
|
|
if !ok {
|
|
|
|
return nil, errors.New("inner[] element not an object")
|
|
|
|
}
|
|
|
|
|
|
|
|
kind, ok := node["kind"].(string)
|
|
|
|
if !ok {
|
|
|
|
return nil, errors.New("node has no kind")
|
|
|
|
}
|
|
|
|
|
|
|
|
switch kind {
|
|
|
|
|
|
|
|
case "CXXRecordDecl":
|
|
|
|
// Must have a name
|
|
|
|
nodename, ok := node["name"].(string)
|
|
|
|
if !ok {
|
|
|
|
return nil, errors.New("node has no name")
|
|
|
|
}
|
|
|
|
|
|
|
|
fmt.Printf("-> %q name=%q\n", kind, nodename)
|
|
|
|
if classInner, ok := node["inner"].([]interface{}); ok {
|
2024-08-07 06:50:39 +00:00
|
|
|
|
|
|
|
// Check if this was 'struct' (default visible) or 'class' (default invisible)
|
|
|
|
visible := true
|
|
|
|
if tagUsed, ok := node["tagUsed"].(string); ok && tagUsed == "class" {
|
|
|
|
visible = false
|
|
|
|
}
|
|
|
|
|
|
|
|
// Process the inner class definition
|
|
|
|
obj, err := processType(classInner, nodename, visible)
|
2024-08-06 01:03:23 +00:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
2024-08-07 06:56:14 +00:00
|
|
|
ret.Classes = append(ret.Classes, obj)
|
2024-08-06 01:03:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
case "StaticAssertDecl":
|
|
|
|
// ignore
|
|
|
|
|
|
|
|
default:
|
|
|
|
return nil, fmt.Errorf("missing handling for clang ast node type %q", kind)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return &ret, nil // done
|
|
|
|
}
|
|
|
|
|
2024-08-07 06:56:14 +00:00
|
|
|
func processType(inner []interface{}, className string, visibility bool) (CppClass, error) {
|
|
|
|
var ret CppClass
|
|
|
|
ret.ClassName = className
|
2024-08-06 01:03:23 +00:00
|
|
|
|
2024-08-06 02:29:12 +00:00
|
|
|
nextMethod:
|
2024-08-06 01:03:23 +00:00
|
|
|
for _, node := range inner {
|
|
|
|
node, ok := node.(map[string]interface{})
|
|
|
|
if !ok {
|
2024-08-07 06:56:14 +00:00
|
|
|
return CppClass{}, errors.New("inner[] element not an object")
|
2024-08-06 01:03:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
kind, ok := node["kind"]
|
|
|
|
if !ok {
|
|
|
|
panic("inner element has no kind")
|
|
|
|
}
|
|
|
|
|
|
|
|
switch kind {
|
2024-08-07 06:50:39 +00:00
|
|
|
case "AccessSpecDecl":
|
|
|
|
// Swap between visible/invisible
|
|
|
|
access, ok := node["access"].(string)
|
|
|
|
if !ok {
|
|
|
|
panic("AccessSpecDecl missing `access` field")
|
|
|
|
}
|
|
|
|
|
|
|
|
switch access {
|
|
|
|
case "public":
|
|
|
|
visibility = true
|
|
|
|
case "private", "protected":
|
|
|
|
visibility = false
|
|
|
|
default:
|
|
|
|
panic("unexpected access visibility '" + access + "'")
|
|
|
|
}
|
|
|
|
|
2024-08-08 05:51:10 +00:00
|
|
|
case "FriendDecl":
|
|
|
|
// Safe to ignore
|
|
|
|
|
|
|
|
case "CXXConstructorDecl":
|
|
|
|
// panic("TODO")
|
|
|
|
|
|
|
|
case "CXXDestructorDecl":
|
2024-08-07 06:51:51 +00:00
|
|
|
// panic("TODO")
|
|
|
|
|
2024-08-06 02:29:12 +00:00
|
|
|
case "CXXMethodDecl":
|
2024-08-07 06:50:39 +00:00
|
|
|
if !visibility {
|
|
|
|
continue // Skip private/protected
|
|
|
|
}
|
|
|
|
|
2024-08-06 02:29:12 +00:00
|
|
|
// Method
|
|
|
|
methodName, ok := node["name"].(string)
|
|
|
|
if !ok {
|
2024-08-07 06:56:14 +00:00
|
|
|
return CppClass{}, errors.New("method has no name")
|
2024-08-06 02:29:12 +00:00
|
|
|
}
|
|
|
|
|
2024-08-07 06:56:14 +00:00
|
|
|
var mm CppMethod
|
|
|
|
mm.MethodName = methodName
|
2024-08-06 02:29:12 +00:00
|
|
|
|
|
|
|
if typobj, ok := node["type"].(map[string]interface{}); ok {
|
|
|
|
if qualType, ok := typobj["qualType"].(string); ok {
|
|
|
|
// The qualType is the whole type of the method, including its parameter types
|
|
|
|
// If anything here is too complicated, skip the whole method
|
2024-08-08 05:51:10 +00:00
|
|
|
|
|
|
|
var err error = nil
|
|
|
|
mm.ReturnType, mm.Parameters, err = parseTypeString(qualType)
|
|
|
|
if err != nil {
|
|
|
|
if errors.Is(err, ErrTooComplex) {
|
|
|
|
log.Printf("Skipping method %q with complex type %q", mm.MethodName, qualType)
|
|
|
|
continue nextMethod
|
|
|
|
}
|
|
|
|
// Real error
|
|
|
|
return CppClass{}, err
|
2024-08-06 02:29:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if methodInner, ok := node["inner"].([]interface{}); ok {
|
2024-08-08 05:51:10 +00:00
|
|
|
paramCounter := 0
|
2024-08-06 02:29:12 +00:00
|
|
|
for _, methodObj := range methodInner {
|
|
|
|
methodObj, ok := methodObj.(map[string]interface{})
|
|
|
|
if !ok {
|
2024-08-07 06:56:14 +00:00
|
|
|
return CppClass{}, errors.New("inner[] element not an object")
|
2024-08-06 02:29:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
switch methodObj["kind"] {
|
|
|
|
case "ParmVarDecl":
|
|
|
|
// Parameter variable
|
|
|
|
parmName, _ := methodObj["name"].(string) // n.b. may be unnamed
|
|
|
|
if parmName == "" {
|
2024-08-08 05:51:10 +00:00
|
|
|
parmName = fmt.Sprintf("param%d", paramCounter+1)
|
2024-08-06 02:29:12 +00:00
|
|
|
}
|
|
|
|
|
2024-08-08 05:51:10 +00:00
|
|
|
// Update the name for the existing nth parameter
|
|
|
|
mm.Parameters[paramCounter].ParameterName = parmName
|
|
|
|
paramCounter++
|
2024-08-07 06:51:51 +00:00
|
|
|
|
2024-08-08 05:51:10 +00:00
|
|
|
// TODO If this parameter is optional, expand it into multiple function overloads
|
2024-08-06 02:29:12 +00:00
|
|
|
|
|
|
|
default:
|
|
|
|
// Something else inside a declaration??
|
|
|
|
fmt.Printf("==> NOT IMPLEMENTED CXXMethodDecl->%q\n", kind)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-08-07 06:56:14 +00:00
|
|
|
ret.Methods = append(ret.Methods, mm)
|
2024-08-06 02:29:12 +00:00
|
|
|
|
2024-08-06 01:03:23 +00:00
|
|
|
default:
|
2024-08-06 02:29:12 +00:00
|
|
|
fmt.Printf("==> NOT IMPLEMENTED %q\n", kind)
|
2024-08-06 01:03:23 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret, nil // done
|
|
|
|
}
|
2024-08-08 05:51:10 +00:00
|
|
|
|
|
|
|
var ErrTooComplex error = errors.New("Type declaration is too complex to parse")
|
|
|
|
|
|
|
|
// parseTypeString converts a string like
|
|
|
|
// - `QString (const char *, const char *, int)`
|
|
|
|
// - `void (const QKeySequence \u0026)`
|
|
|
|
// into its (A) return type and (B) separate parameter types.
|
|
|
|
// These clang strings never contain the parameter's name, so the names here are
|
|
|
|
// not filled in.
|
|
|
|
func parseTypeString(typeString string) (CppParameter, []CppParameter, error) {
|
|
|
|
|
|
|
|
if strings.Contains(typeString, `::`) {
|
|
|
|
return CppParameter{}, nil, ErrTooComplex
|
|
|
|
}
|
|
|
|
|
|
|
|
// Cut to exterior-most (, ) pair
|
|
|
|
opos := strings.Index(typeString, `(`)
|
|
|
|
epos := strings.LastIndex(typeString, `)`)
|
|
|
|
|
|
|
|
if opos == -1 || epos == -1 {
|
|
|
|
return CppParameter{}, nil, fmt.Errorf("Type string %q missing brackets")
|
|
|
|
}
|
|
|
|
|
|
|
|
returnType := parseSingleTypeString(strings.TrimSpace(typeString[0:opos]))
|
|
|
|
|
|
|
|
inner := typeString[opos+1 : epos]
|
|
|
|
|
|
|
|
// Should be no more brackets
|
|
|
|
if strings.ContainsAny(inner, `()`) {
|
|
|
|
return CppParameter{}, nil, ErrTooComplex
|
|
|
|
}
|
|
|
|
|
|
|
|
// Parameters are separated by commas and nesting can not be possible
|
|
|
|
params := strings.Split(inner, `,`)
|
|
|
|
|
|
|
|
ret := make([]CppParameter, 0, len(params))
|
|
|
|
for _, p := range params {
|
|
|
|
|
|
|
|
insert := parseSingleTypeString(p)
|
|
|
|
|
|
|
|
if insert.ParameterType != "" {
|
|
|
|
ret = append(ret, insert)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return returnType, ret, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func parseSingleTypeString(p string) CppParameter {
|
|
|
|
|
|
|
|
tokens := strings.Split(strings.TrimSpace(p), " ")
|
|
|
|
insert := CppParameter{}
|
|
|
|
for _, tok := range tokens {
|
|
|
|
if tok == "const" {
|
|
|
|
insert.Const = true
|
|
|
|
} else if tok == "&" { // U+0026
|
|
|
|
insert.ByRef = true
|
|
|
|
} else if tok == "*" {
|
|
|
|
insert.Pointer = true
|
|
|
|
} else {
|
|
|
|
// Valid part of the type name
|
|
|
|
insert.ParameterType += " " + tok
|
|
|
|
}
|
|
|
|
}
|
|
|
|
insert.ParameterType = strings.TrimSpace(insert.ParameterType)
|
|
|
|
|
|
|
|
return insert
|
|
|
|
}
|