genbindings: parse and call const methods for disambiguation

This commit is contained in:
mappu 2024-08-28 18:47:23 +12:00
parent 488e3cc6c4
commit 56395eaa7a
3 changed files with 21 additions and 9 deletions

View File

@ -491,7 +491,7 @@ func parseMethod(node map[string]interface{}, mm *CppMethod) error {
// If anything here is too complicated, skip the whole method // If anything here is too complicated, skip the whole method
var err error = nil var err error = nil
mm.ReturnType, mm.Parameters, err = parseTypeString(qualType) mm.ReturnType, mm.Parameters, mm.IsConst, err = parseTypeString(qualType)
if err != nil { if err != nil {
return err return err
} }
@ -575,15 +575,15 @@ func parseMethod(node map[string]interface{}, mm *CppMethod) error {
} }
// parseTypeString converts a function/method type string such as // parseTypeString converts a function/method type string such as
// - `QString (const char *, const char *, int)` // - `QString (const char *, const char *, int) const`
// - `void (const QKeySequence \u0026)` // - `void (const QKeySequence \u0026)`
// into its (A) return type and (B) separate parameter types. // into its (A) return type and (B) separate parameter types.
// These clang strings never contain the parameter's name, so the names here are // These clang strings never contain the parameter's name, so the names here are
// not filled in. // not filled in.
func parseTypeString(typeString string) (CppParameter, []CppParameter, error) { func parseTypeString(typeString string) (CppParameter, []CppParameter, bool, error) {
if strings.Contains(typeString, `&&`) { // TODO Rvalue references if strings.Contains(typeString, `&&`) { // TODO Rvalue references
return CppParameter{}, nil, ErrTooComplex return CppParameter{}, nil, false, ErrTooComplex
} }
// Cut to exterior-most (, ) pair // Cut to exterior-most (, ) pair
@ -591,7 +591,12 @@ func parseTypeString(typeString string) (CppParameter, []CppParameter, error) {
epos := strings.LastIndex(typeString, `)`) epos := strings.LastIndex(typeString, `)`)
if opos == -1 || epos == -1 { if opos == -1 || epos == -1 {
return CppParameter{}, nil, fmt.Errorf("Type string %q missing brackets", typeString) return CppParameter{}, nil, false, fmt.Errorf("Type string %q missing brackets", typeString)
}
isConst := false
if strings.Contains(typeString[epos:], `const`) {
isConst = true
} }
returnType := parseSingleTypeString(strings.TrimSpace(typeString[0:opos])) returnType := parseSingleTypeString(strings.TrimSpace(typeString[0:opos]))
@ -599,14 +604,14 @@ func parseTypeString(typeString string) (CppParameter, []CppParameter, error) {
// Skip functions that return ints-by-reference since the ergonomics don't // Skip functions that return ints-by-reference since the ergonomics don't
// go through the binding // go through the binding
if returnType.IntType() && returnType.ByRef { if returnType.IntType() && returnType.ByRef {
return CppParameter{}, nil, ErrTooComplex // e.g. QSize::rheight() return CppParameter{}, nil, false, ErrTooComplex // e.g. QSize::rheight()
} }
inner := typeString[opos+1 : epos] inner := typeString[opos+1 : epos]
// Should be no more brackets // Should be no more brackets
if strings.ContainsAny(inner, `()`) { if strings.ContainsAny(inner, `()`) {
return CppParameter{}, nil, ErrTooComplex return CppParameter{}, nil, false, ErrTooComplex
} }
// Parameters are separated by commas and nesting can not be possible // Parameters are separated by commas and nesting can not be possible
@ -622,7 +627,7 @@ func parseTypeString(typeString string) (CppParameter, []CppParameter, error) {
} }
} }
return returnType, ret, nil return returnType, ret, isConst, nil
} }
func tokenizeMultipleParameters(p string) []string { func tokenizeMultipleParameters(p string) []string {

View File

@ -600,7 +600,13 @@ extern "C" {
} }
} else if m.ReturnType.QtClassType() && !m.ReturnType.Pointer { } else if m.ReturnType.QtClassType() && !m.ReturnType.Pointer {
// Still need const/nonconst in order to properly call function overloads that are only disambiguated by const-ness
// e.g. QCborArray::begin() can return an Iterator or a ConstIterator
if m.IsConst {
shouldReturn = "const " + m.ReturnType.ParameterType + " ret = "
} else {
shouldReturn = m.ReturnType.ParameterType + " ret = " shouldReturn = m.ReturnType.ParameterType + " ret = "
}
afterCall = "\t// Copy-construct value returned type into heap-allocated copy\n" afterCall = "\t// Copy-construct value returned type into heap-allocated copy\n"
afterCall += "\treturn static_cast<" + m.ReturnType.ParameterType + "*>(new " + m.ReturnType.ParameterType + "(ret));\n" afterCall += "\treturn static_cast<" + m.ReturnType.ParameterType + "*>(new " + m.ReturnType.ParameterType + "(ret));\n"

View File

@ -159,6 +159,7 @@ type CppMethod struct {
Parameters []CppParameter Parameters []CppParameter
IsStatic bool IsStatic bool
IsSignal bool IsSignal bool
IsConst bool
HasHiddenParams bool // Set to true if there is an overload with more parameters HasHiddenParams bool // Set to true if there is an overload with more parameters
} }