| // Copyright (c) 2020, the Dart project authors. Please see the AUTHORS file |
| // for details. All rights reserved. Use of this source code is governed by a |
| // BSD-style license that can be found in the LICENSE file. |
| |
| import 'package:ffigen/src/code_generator.dart'; |
| import 'package:ffigen/src/config_provider/config_types.dart'; |
| import 'package:ffigen/src/header_parser/data.dart'; |
| import 'package:logging/logging.dart'; |
| |
| import '../clang_bindings/clang_bindings.dart' as clang_types; |
| import '../includer.dart'; |
| import '../utils.dart'; |
| |
| final _logger = Logger('ffigen.header_parser.functiondecl_parser'); |
| |
| /// Holds temporary information regarding [Func] while parsing. |
| class _ParserFunc { |
| /// Multiple values are since there may be more than one instance of the |
| /// same base C function with different variadic arguments. |
| List<Func> funcs = []; |
| bool incompleteStructParameter = false; |
| bool unimplementedParameterType = false; |
| _ParserFunc(); |
| } |
| |
| final _stack = Stack<_ParserFunc>(); |
| |
| /// Parses a function declaration. |
| List<Func>? parseFunctionDeclaration(clang_types.CXCursor cursor) { |
| _stack.push(_ParserFunc()); |
| |
| final funcUsr = cursor.usr(); |
| final funcName = cursor.spelling(); |
| if (shouldIncludeFunc(funcUsr, funcName)) { |
| _logger.fine('++++ Adding Function: ${cursor.completeStringRepr()}'); |
| |
| final rt = _getFunctionReturnType(cursor); |
| final parameters = _getParameters(cursor, funcName); |
| |
| if (clang.clang_Cursor_isFunctionInlined(cursor) != 0) { |
| _logger.fine('---- Removed Function, reason: inline function: ' |
| '${cursor.completeStringRepr()}'); |
| _logger.warning( |
| "Skipped Function '$funcName', inline functions are not supported."); |
| // Returning empty so that [addToBindings] function excludes this. |
| return _stack.pop().funcs; |
| } |
| |
| if (rt.isIncompleteCompound || _stack.top.incompleteStructParameter) { |
| _logger.fine( |
| '---- Removed Function, reason: Incomplete struct pass/return by ' |
| 'value: ${cursor.completeStringRepr()}'); |
| _logger.warning( |
| "Skipped Function '$funcName', Incomplete struct pass/return by " |
| 'value not supported.'); |
| // Returning null so that [addToBindings] function excludes this. |
| return _stack.pop().funcs; |
| } |
| |
| if (rt.baseType is UnimplementedType || |
| _stack.top.unimplementedParameterType) { |
| _logger.fine('---- Removed Function, reason: unsupported return type or ' |
| 'parameter type: ${cursor.completeStringRepr()}'); |
| _logger.warning( |
| "Skipped Function '$funcName', function has unsupported return type " |
| 'or parameter type.'); |
| // Returning null so that [addToBindings] function excludes this. |
| return _stack.pop().funcs; |
| } |
| |
| // Initialized with a single value with no prefix and empty var args. |
| var varArgFunctions = [VarArgFunction('', [])]; |
| if (config.varArgFunctions.containsKey(funcName)) { |
| if (clang.clang_isFunctionTypeVariadic(cursor.type()) == 1) { |
| varArgFunctions = config.varArgFunctions[funcName]!; |
| } else { |
| _logger.warning( |
| "Skipping variadic-argument config for function $funcName since its not variadic."); |
| } |
| } |
| for (final vaFunc in varArgFunctions) { |
| _stack.top.funcs.add(Func( |
| dartDoc: getCursorDocComment( |
| cursor, |
| nesting.length + commentPrefix.length, |
| ), |
| usr: funcUsr + vaFunc.postfix, |
| name: config.functionDecl.renameUsingConfig(funcName) + vaFunc.postfix, |
| originalName: funcName, |
| returnType: rt, |
| parameters: parameters, |
| varArgParameters: |
| vaFunc.types.map((ta) => Parameter(type: ta, name: 'va')).toList(), |
| exposeSymbolAddress: |
| config.functionDecl.shouldIncludeSymbolAddress(funcName), |
| exposeFunctionTypedefs: |
| config.exposeFunctionTypedefs.shouldInclude(funcName), |
| isLeaf: config.leafFunctions.shouldInclude(funcName), |
| ffiNativeConfig: config.ffiNativeConfig, |
| )); |
| } |
| bindingsIndex.addFuncToSeen(funcUsr, _stack.top.funcs.last); |
| } else if (bindingsIndex.isSeenFunc(funcUsr)) { |
| _stack.top.funcs.add(bindingsIndex.getSeenFunc(funcUsr)!); |
| } |
| |
| return _stack.pop().funcs; |
| } |
| |
| Type _getFunctionReturnType(clang_types.CXCursor cursor) { |
| return cursor.returnType().toCodeGenType(); |
| } |
| |
| List<Parameter> _getParameters(clang_types.CXCursor cursor, String funcName) { |
| final parameters = <Parameter>[]; |
| |
| final totalArgs = clang.clang_Cursor_getNumArguments(cursor); |
| for (var i = 0; i < totalArgs; i++) { |
| final paramCursor = clang.clang_Cursor_getArgument(cursor, i); |
| |
| _logger.finer('===== parameter: ${paramCursor.completeStringRepr()}'); |
| |
| final pt = _getParameterType(paramCursor); |
| if (pt.isIncompleteCompound) { |
| _stack.top.incompleteStructParameter = true; |
| } else if (pt.baseType is UnimplementedType) { |
| _logger.finer('Unimplemented type: ${pt.baseType}'); |
| _stack.top.unimplementedParameterType = true; |
| } |
| |
| final pn = paramCursor.spelling(); |
| |
| /// If [pn] is null or empty, its set to `arg$i` by code_generator. |
| parameters.add( |
| Parameter( |
| originalName: pn, |
| name: config.functionDecl.renameMemberUsingConfig(funcName, pn), |
| type: pt, |
| ), |
| ); |
| } |
| |
| return parameters; |
| } |
| |
| Type _getParameterType(clang_types.CXCursor cursor) { |
| return cursor.toCodeGenType(); |
| } |