blob: 29a86e5031a370bf1a4ce4cf26ee92c6e603b04b [file] [log] [blame]
// 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 'dart:ffi';
import 'package:ffi/ffi.dart';
import 'package:ffigen/src/code_generator.dart';
import 'package:ffigen/src/config_provider.dart';
import 'package:ffigen/src/header_parser/sub_parsers/macro_parser.dart';
import 'package:ffigen/src/config_provider/config_types.dart';
import 'package:ffigen/src/find_resource.dart';
import 'package:ffigen/src/header_parser/sub_parsers/unnamed_enumdecl_parser.dart';
import 'package:ffigen/src/header_parser/translation_unit_parser.dart';
import 'package:ffigen/src/strings.dart' as strings;
import 'package:logging/logging.dart';
import 'package:path/path.dart' as path;
import 'clang_bindings/clang_bindings.dart' as clang_types;
import 'data.dart';
import 'utils.dart';
/// Main entrypoint for header_parser.
Library parse(Config conf, {bool sort = false}) {
initParser(conf);
final bindings = parseToBindings();
final library = Library(
bindings: bindings,
name: config.wrapperName,
description: config.wrapperDocComment,
header: config.preamble,
);
if (sort) {
library.sort();
}
return library;
}
// ===================================================================================
// BELOW FUNCTIONS ARE MEANT FOR INTERNAL USE AND TESTING
// ===================================================================================
var _logger = Logger('ffigen.header_parser.parser');
/// Initialises parser, clears any previous values.
void initParser(Config c) {
// Set global configurations.
config = c;
// Find full path of dynamic library and initialise bindings.
if (findDotDartTool() == null) {
throw Exception('Unable to find .dart_tool.');
} else {
final fullDylibPath = path.join(
findDotDartTool().toFilePath(),
strings.ffigenFolderName,
strings.dylibFileName,
);
clang = clang_types.Clang(DynamicLibrary.open(fullDylibPath));
}
}
/// Parses source files and adds generated bindings to [bindings].
List<Binding> parseToBindings() {
final index = clang.clang_createIndex(0, 0);
Pointer<Pointer<Utf8>> clangCmdArgs = nullptr;
var cmdLen = 0;
/// Add compiler opt for comment parsing for clang based on config.
if (config.commentType.length != CommentLength.none &&
config.commentType.style == CommentStyle.any) {
config.compilerOpts ??= [];
config.compilerOpts.add(strings.fparseAllComments);
}
if (config.compilerOpts != null) {
clangCmdArgs = createDynamicStringArray(config.compilerOpts);
cmdLen = config.compilerOpts.length;
}
// Contains all bindings.
final bindings = <Binding>[];
// Log all headers for user.
_logger.info('Input Headers: ${config.headers.entryPoints}');
for (final headerLocation in config.headers.entryPoints) {
_logger.fine('Creating TranslationUnit for header: $headerLocation');
final tu = clang.clang_parseTranslationUnit(
index,
Utf8.toUtf8(headerLocation).cast(),
clangCmdArgs.cast(),
cmdLen,
nullptr,
0,
clang_types.CXTranslationUnit_Flags.CXTranslationUnit_SkipFunctionBodies |
clang_types.CXTranslationUnit_Flags
.CXTranslationUnit_DetailedPreprocessingRecord,
);
if (tu == nullptr) {
_logger.severe(
"Skipped header/file: $headerLocation, couldn't parse source.");
// Skip parsing this header.
continue;
}
logTuDiagnostics(tu, _logger, headerLocation);
final rootCursor = clang.clang_getTranslationUnitCursor_wrap(tu);
bindings.addAll(parseTranslationUnit(rootCursor));
// Cleanup.
rootCursor.dispose();
clang.clang_disposeTranslationUnit(tu);
}
// Add all saved unnamed enums.
bindings.addAll(getSavedUnNamedEnums());
// Parse all saved macros.
bindings.addAll(parseSavedMacros());
if (config.compilerOpts != null) {
clangCmdArgs.dispose(config.compilerOpts.length);
}
clang.clang_disposeIndex(index);
return bindings;
}