123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597 |
- #!/usr/bin/env python
- # -*- Mode: Python -*-
- # GObject-Introspection - a framework for introspecting GObject libraries
- # Copyright (C) 2008-2010 Johan Dahlin
- # Copyright (C) 2009 Red Hat, Inc.
- #
- # This program is free software; you can redistribute it and/or
- # modify it under the terms of the GNU General Public License
- # as published by the Free Software Foundation; either version 2
- # of the License, or (at your option) any later version.
- #
- # This program is distributed in the hope that it will be useful,
- # but WITHOUT ANY WARRANTY; without even the implied warranty of
- # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- # GNU General Public License for more details.
- #
- # You should have received a copy of the GNU General Public License
- # along with this program; if not, write to the Free Software
- # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
- # 02110-1301, USA.
- #
- from __future__ import absolute_import
- from __future__ import division
- from __future__ import print_function
- from __future__ import unicode_literals
- import errno
- import optparse
- import os
- import shutil
- import stat
- import subprocess
- import sys
- import tempfile
- import platform
- from giscanner import message
- from giscanner.annotationparser import GtkDocCommentBlockParser
- from giscanner.ast import Include, Namespace
- from giscanner.dumper import compile_introspection_binary
- from giscanner.gdumpparser import GDumpParser, IntrospectionBinary
- from giscanner.introspectablepass import IntrospectablePass
- from giscanner.girparser import GIRParser
- from giscanner.girwriter import GIRWriter
- from giscanner.maintransformer import MainTransformer
- from giscanner.shlibs import resolve_shlibs
- from giscanner.sourcescanner import SourceScanner, ALL_EXTS
- from giscanner.transformer import Transformer
- from . import utils
- def process_cflags_begin(option, opt, value, parser):
- cflags = getattr(parser.values, option.dest)
- while len(parser.rargs) > 0 and parser.rargs[0] != '--cflags-end':
- arg = parser.rargs.pop(0)
- if arg == "-I" and parser.rargs and parser.rargs[0] != '--cflags-end':
- # This is a special case where there's a space between -I and the path.
- arg += parser.rargs.pop(0)
- cflags.append(utils.cflag_real_include_path(arg))
- def process_cflags_end(option, opt, value, parser):
- pass
- def process_cpp_includes(option, opt, value, parser):
- cpp_includes = getattr(parser.values, option.dest)
- cpp_includes.append(os.path.realpath(value))
- def get_preprocessor_option_group(parser):
- group = optparse.OptionGroup(parser, "Preprocessor options")
- group.add_option("", "--cflags-begin",
- help="Start preprocessor/compiler flags",
- dest="cflags", default=[],
- action="callback", callback=process_cflags_begin)
- group.add_option("", "--cflags-end",
- help="End preprocessor/compiler flags",
- action="callback", callback=process_cflags_end)
- group.add_option("-I", help="Pre-processor include file",
- dest="cpp_includes", default=[], type="string",
- action="callback", callback=process_cpp_includes)
- group.add_option("-D", help="Pre-processor define",
- action="append", dest="cpp_defines",
- default=[])
- group.add_option("-U", help="Pre-processor undefine",
- action="append", dest="cpp_undefines",
- default=[])
- group.add_option("-p", dest="", help="Ignored")
- return group
- def get_windows_option_group(parser):
- group = optparse.OptionGroup(parser, "Machine Dependent Options")
- group.add_option("-m", help="some machine dependent option",
- action="append", dest='m_option',
- default=[])
- return group
- def _get_option_parser():
- parser = optparse.OptionParser('%prog [options] sources')
- parser.add_option('', "--quiet",
- action="store_true", dest="quiet",
- default=False,
- help="If passed, do not print details of normal operation")
- parser.add_option("", "--format",
- action="store", dest="format",
- default="gir",
- help="format to use, one of gidl, gir")
- parser.add_option("-i", "--include",
- action="append", dest="includes", default=[],
- help="Add specified gir file as dependency")
- parser.add_option("", "--include-uninstalled",
- action="append", dest="includes_uninstalled", default=[],
- help=("""A file path to a dependency; only use this "
- "when building multiple .gir files inside a "
- "single module."""))
- parser.add_option("", "--add-include-path",
- action="append", dest="include_paths", default=[],
- help="include paths for other GIR files")
- parser.add_option("", "--program",
- action="store", dest="program", default=None,
- help="program to execute")
- parser.add_option("", "--use-binary-wrapper",
- action="store", dest="wrapper", default=None,
- help="wrapper to use for running programs (useful when cross-compiling)")
- parser.add_option("", "--use-ldd-wrapper",
- action="store", dest="ldd_wrapper", default=None,
- help="wrapper to use instead of ldd (useful when cross-compiling)")
- parser.add_option("", "--program-arg",
- action="append", dest="program_args", default=[],
- help="extra arguments to program")
- parser.add_option("", "--libtool",
- action="store", dest="libtool_path", default=None,
- help="full path to libtool")
- parser.add_option("", "--no-libtool",
- action="store_true", dest="nolibtool", default=False,
- help="do not use libtool")
- parser.add_option("", "--external-library",
- action="store_true", dest="external_library", default=False,
- help=("""If true, the library is located on the system,""" +
- """not in the current directory"""))
- parser.add_option("-l", "--library",
- action="append", dest="libraries", default=[],
- help="libraries of this unit")
- parser.add_option("-L", "--library-path",
- action="append", dest="library_paths", default=[],
- help="directories to search for libraries")
- parser.add_option("", "--header-only",
- action="store_true", dest="header_only", default=[],
- help="If specified, just generate a GIR for the given header files")
- parser.add_option("-n", "--namespace",
- action="store", dest="namespace_name",
- help=("name of namespace for this unit, also "
- "used to compute --identifier-prefix and --symbol-prefix"))
- parser.add_option("", "--nsversion",
- action="store", dest="namespace_version",
- help="version of namespace for this unit")
- parser.add_option("", "--strip-prefix",
- action="store", dest="strip_prefix",
- help="""Option --strip-prefix is deprecated, please see --identifier-prefix
- and --symbol-prefix.""")
- parser.add_option("", "--identifier-prefix",
- action="append", dest="identifier_prefixes", default=[],
- help="""Remove this prefix from C identifiers (structure typedefs, etc.).
- May be specified multiple times. This is also used as the default for --symbol-prefix if
- the latter is not specified.""")
- parser.add_option("", "--identifier-filter-cmd",
- action="store", dest="identifier_filter_cmd", default='',
- help='Filter identifiers (struct and union typedefs) through the given '
- 'shell command which will receive the identifier name as input '
- 'to stdin and is expected to output the filtered results to stdout.')
- parser.add_option("", "--symbol-prefix",
- action="append", dest="symbol_prefixes", default=[],
- help="Remove this prefix from C symbols (function names)")
- parser.add_option("", "--symbol-filter-cmd",
- action="store", dest="symbol_filter_cmd", default='',
- help='Filter symbols (function names) through the given '
- 'shell command which will receive the symbol name as input '
- 'to stdin and is expected to output the filtered results to stdout.')
- parser.add_option("", "--accept-unprefixed",
- action="store_true", dest="accept_unprefixed", default=False,
- help="""If specified, accept symbols and identifiers that do not
- match the namespace prefix.""")
- parser.add_option("", "--add-init-section",
- action="append", dest="init_sections", default=[],
- help="add extra initialization code in the introspection program")
- parser.add_option("-o", "--output",
- action="store", dest="output", default="-",
- help="output filename to write to, defaults to - (stdout)")
- parser.add_option("", "--pkg",
- action="append", dest="packages", default=[],
- help="pkg-config packages to get cflags from")
- parser.add_option("", "--pkg-export",
- action="append", dest="packages_export", default=[],
- help="Associated pkg-config packages for this library")
- parser.add_option('', "--warn-all",
- action="store_true", dest="warn_all", default=False,
- help="If true, enable all warnings for introspection")
- parser.add_option('', "--warn-error",
- action="store_true", dest="warn_fatal",
- help="Turn warnings into fatal errors")
- parser.add_option("-v", "--verbose",
- action="store_true", dest="verbose",
- help="be verbose")
- parser.add_option("", "--c-include",
- action="append", dest="c_includes", default=[],
- help="headers which should be included in C programs")
- parser.add_option("", "--filelist",
- action="store", dest="filelist", default=[],
- help="file containing headers and sources to be scanned")
- group = get_preprocessor_option_group(parser)
- parser.add_option_group(group)
- msystemenv = os.environ.get('MSYSTEM')
- if msystemenv and msystemenv.startswith('MINGW'):
- group = get_windows_option_group(parser)
- parser.add_option_group(group)
- # Private options
- parser.add_option('', "--generate-typelib-tests",
- action="store", dest="test_codegen", default=None,
- help=optparse.SUPPRESS_HELP)
- parser.add_option('', "--passthrough-gir",
- action="store", dest="passthrough_gir", default=None,
- help=optparse.SUPPRESS_HELP)
- parser.add_option('', "--reparse-validate",
- action="store_true", dest="reparse_validate_gir", default=False,
- help=optparse.SUPPRESS_HELP)
- parser.add_option("", "--typelib-xml",
- action="store_true", dest="typelib_xml",
- help=optparse.SUPPRESS_HELP)
- parser.add_option("", "--function-decoration",
- action="append", dest="function_decoration", default=[],
- help="Macro to decorate functions in generated code")
- parser.add_option("", "--include-first-in-header",
- action="append", dest="include_first_header", default=[],
- help="Header to include first in generated header")
- parser.add_option("", "--include-last-in-header",
- action="append", dest="include_last_header", default=[],
- help="Header to include after the other headers in generated header")
- parser.add_option("", "--include-first-in-src",
- action="append", dest="include_first_src", default=[],
- help="Header to include first in generated sources")
- parser.add_option("", "--include-last-in-src",
- action="append", dest="include_last_src", default=[],
- help="Header to include after the other headers in generated sources")
- return parser
- def _error(msg):
- raise SystemExit('ERROR: %s' % (msg, ))
- def passthrough_gir(path, f):
- parser = GIRParser()
- parser.parse(path)
- writer = GIRWriter(parser.get_namespace())
- f.write(writer.get_encoded_xml())
- def test_codegen(optstring,
- function_decoration,
- include_first_header,
- include_last_header,
- include_first_src,
- include_last_src):
- (namespace, out_h_filename, out_c_filename) = optstring.split(',')
- if namespace == 'Everything':
- from .testcodegen import EverythingCodeGenerator
- gen = EverythingCodeGenerator(out_h_filename,
- out_c_filename,
- function_decoration,
- include_first_header,
- include_last_header,
- include_first_src,
- include_last_src)
- gen.write()
- else:
- _error("Invaild namespace '%s'" % (namespace, ))
- return 0
- def process_options(output, allowed_flags):
- for option in output.split():
- for flag in allowed_flags:
- if not option.startswith(flag):
- continue
- yield option
- break
- def process_packages(options, packages):
- args = [os.environ.get('PKG_CONFIG', 'pkg-config'), '--cflags']
- args.extend(packages)
- output = subprocess.Popen(args,
- stdout=subprocess.PIPE).communicate()[0]
- if output is None:
- # the error output should have already appeared on our stderr,
- # so we just exit
- return 1
- output = output.decode('ascii')
- # Some pkg-config files on Windows have options we don't understand,
- # so we explicitly filter to only the ones we need.
- options_whitelist = ['-I', '-D', '-U', '-l', '-L']
- filtered_output = list(process_options(output, options_whitelist))
- parser = _get_option_parser()
- pkg_options, unused = parser.parse_args(filtered_output)
- options.cpp_includes.extend([os.path.realpath(f) for f in pkg_options.cpp_includes])
- options.cpp_defines.extend(pkg_options.cpp_defines)
- options.cpp_undefines.extend(pkg_options.cpp_undefines)
- def extract_filenames(args):
- filenames = []
- for arg in args:
- # We don't support real C++ parsing yet, but we should be able
- # to understand C API implemented in C++ files.
- if os.path.splitext(arg)[1] in ALL_EXTS:
- if not os.path.exists(arg):
- _error('%s: no such a file or directory' % (arg, ))
- # Make absolute, because we do comparisons inside scannerparser.c
- # against the absolute path that cpp will give us
- filenames.append(arg)
- return filenames
- def extract_filelist(options):
- filenames = []
- if not os.path.exists(options.filelist):
- _error('%s: no such filelist file' % (options.filelist, ))
- with open(options.filelist, "r") as filelist_file:
- lines = filelist_file.readlines()
- for line in lines:
- # We don't support real C++ parsing yet, but we should be able
- # to understand C API implemented in C++ files.
- filename = line.strip()
- if (filename.endswith('.c') or filename.endswith('.cpp')
- or filename.endswith('.cc') or filename.endswith('.cxx')
- or filename.endswith('.h') or filename.endswith('.hpp')
- or filename.endswith('.hxx')):
- if not os.path.exists(filename):
- _error('%s: Invalid filelist entry-no such file or directory' % (line, ))
- # Make absolute, because we do comparisons inside scannerparser.c
- # against the absolute path that cpp will give us
- filenames.append(filename)
- return filenames
- def create_namespace(options):
- if options.strip_prefix:
- print("""g-ir-scanner: warning: Option --strip-prefix has been deprecated;
- see --identifier-prefix and --symbol-prefix.""")
- options.identifier_prefixes.append(options.strip_prefix)
- # We do this dance because the empty list has different semantics from
- # None; if the user didn't specify the options, we want to use None so
- # the Namespace constructor picks the defaults.
- if options.identifier_prefixes:
- identifier_prefixes = options.identifier_prefixes
- else:
- identifier_prefixes = None
- if options.symbol_prefixes:
- for prefix in options.symbol_prefixes:
- # See Transformer._split_c_string_for_namespace_matches() for
- # why this check is needed
- if prefix.lower() != prefix:
- _error("Values for --symbol-prefix must be entirely lowercase")
- symbol_prefixes = options.symbol_prefixes
- else:
- symbol_prefixes = None
- return Namespace(options.namespace_name,
- options.namespace_version,
- identifier_prefixes=identifier_prefixes,
- symbol_prefixes=symbol_prefixes)
- def create_transformer(namespace, options):
- transformer = Transformer(namespace,
- accept_unprefixed=options.accept_unprefixed,
- identifier_filter_cmd=options.identifier_filter_cmd,
- symbol_filter_cmd=options.symbol_filter_cmd)
- transformer.set_include_paths(options.include_paths)
- if options.passthrough_gir or options.reparse_validate_gir:
- transformer.disable_cache()
- transformer.set_passthrough_mode()
- for include in options.includes:
- if os.sep in include:
- _error("Invalid include path '%s'" % (include, ))
- try:
- include_obj = Include.from_string(include)
- except:
- _error("Malformed include '%s'\n" % (include, ))
- transformer.register_include(include_obj)
- for include_path in options.includes_uninstalled:
- transformer.register_include_uninstalled(include_path)
- return transformer
- def create_binary(transformer, options, args):
- # Transform the C AST nodes into higher level
- # GLib/GObject nodes
- gdump_parser = GDumpParser(transformer)
- # Do enough parsing that we have the get_type() functions to reference
- # when creating the introspection binary
- gdump_parser.init_parse()
- if options.program:
- args = [options.program]
- args.extend(options.program_args)
- binary = IntrospectionBinary(args)
- else:
- binary = compile_introspection_binary(options,
- gdump_parser.get_get_type_functions(),
- gdump_parser.get_error_quark_functions())
- shlibs = resolve_shlibs(options, binary, options.libraries)
- if options.wrapper:
- # The wrapper needs the binary itself, not the libtool wrapper script,
- # so we check if libtool has sneaked the binary into .libs subdirectory
- # and adjust the path accordingly
- import os.path
- dir_name, binary_name = os.path.split(binary.args[0])
- libtool_binary = os.path.join(dir_name, '.libs', binary_name)
- if os.path.exists(libtool_binary):
- binary.args[0] = libtool_binary
- # Then prepend the wrapper to the command line to execute
- binary.args = [options.wrapper] + binary.args
- gdump_parser.set_introspection_binary(binary)
- gdump_parser.parse()
- return shlibs
- def create_source_scanner(options, args):
- if hasattr(options, 'filelist') and options.filelist:
- filenames = extract_filelist(options)
- else:
- filenames = extract_filenames(args)
- if platform.system() == 'Darwin':
- options.cpp_undefines.append('__BLOCKS__')
- # Run the preprocessor, tokenize and construct simple
- # objects representing the raw C symbols
- ss = SourceScanner()
- ss.set_cpp_options(options.cpp_includes,
- options.cpp_defines,
- options.cpp_undefines,
- cflags=options.cflags)
- ss.parse_files(filenames)
- ss.parse_macros(filenames)
- return ss
- def write_output(data, options):
- """Write encoded XML 'data' to the filename specified in 'options'."""
- if options.output == "-":
- output = sys.stdout
- elif options.reparse_validate_gir:
- main_f, main_f_name = tempfile.mkstemp(suffix='.gir')
- if (os.path.isfile(options.output)):
- shutil.copystat(options.output, main_f_name)
- else:
- os.chmod(main_f_name,
- stat.S_IWUSR | stat.S_IRUSR | stat.S_IRGRP | stat.S_IROTH)
- with os.fdopen(main_f, 'wb') as main_f:
- main_f.write(data)
- temp_f, temp_f_name = tempfile.mkstemp(suffix='.gir')
- with os.fdopen(temp_f, 'wb') as temp_f:
- passthrough_gir(main_f_name, temp_f)
- if not utils.files_are_identical(main_f_name, temp_f_name):
- _error("Failed to re-parse gir file; scanned='%s' passthrough='%s'" % (
- main_f_name, temp_f_name))
- os.unlink(temp_f_name)
- try:
- shutil.move(main_f_name, options.output)
- except OSError as e:
- if e.errno == errno.EPERM:
- os.unlink(main_f_name)
- raise
- return 0
- else:
- try:
- output = open(options.output, 'wb')
- except IOError as e:
- _error("opening output for writing: %s" % (e.strerror, ))
- try:
- output.write(data)
- except IOError as e:
- _error("while writing output: %s" % (e.strerror, ))
- def scanner_main(args):
- parser = _get_option_parser()
- (options, args) = parser.parse_args(args)
- if options.passthrough_gir:
- passthrough_gir(options.passthrough_gir, sys.stdout)
- if options.test_codegen:
- return test_codegen(options.test_codegen,
- options.function_decoration,
- options.include_first_header,
- options.include_last_header,
- options.include_first_src,
- options.include_last_src)
- if hasattr(options, 'filelist') and not options.filelist:
- if len(args) <= 1:
- _error('Need at least one filename')
- if not options.namespace_name:
- _error('Namespace name missing')
- if options.format == 'gir':
- from giscanner.girwriter import GIRWriter as Writer
- else:
- _error("Unknown format: %s" % (options.format, ))
- if not (options.libraries
- or options.program
- or options.header_only):
- _error("Must specify --program or --library")
- namespace = create_namespace(options)
- logger = message.MessageLogger.get(namespace=namespace)
- if options.warn_all:
- logger.enable_warnings((message.WARNING, message.ERROR, message.FATAL))
- transformer = create_transformer(namespace, options)
- packages = set(options.packages)
- packages.update(transformer.get_pkgconfig_packages())
- if packages:
- exit_code = process_packages(options, packages)
- if exit_code:
- return exit_code
- ss = create_source_scanner(options, args)
- cbp = GtkDocCommentBlockParser()
- blocks = cbp.parse_comment_blocks(ss.get_comments())
- # Transform the C symbols into AST nodes
- transformer.parse(ss.get_symbols())
- if not options.header_only:
- shlibs = create_binary(transformer, options, args)
- else:
- shlibs = []
- transformer.namespace.shared_libraries = shlibs
- main = MainTransformer(transformer, blocks)
- main.transform()
- utils.break_on_debug_flag('tree')
- final = IntrospectablePass(transformer, blocks)
- final.validate()
- warning_count = logger.get_warning_count()
- if options.warn_fatal and warning_count > 0:
- message.fatal("warnings configured as fatal")
- return 1
- elif warning_count > 0 and options.warn_all is False:
- print("g-ir-scanner: %s: warning: %d warnings suppressed "
- "(use --warn-all to see them)" %
- (transformer.namespace.name, warning_count, ))
- # Write out AST
- if options.packages_export:
- exported_packages = options.packages_export
- else:
- exported_packages = options.packages
- transformer.namespace.c_includes = options.c_includes
- transformer.namespace.exported_packages = exported_packages
- writer = Writer(transformer.namespace)
- data = writer.get_encoded_xml()
- write_output(data, options)
- return 0
|