# Copyright (c) 2013 The Chromium Authors. All rights reserved. # Use of this source code is governed by a BSD-style license that can be # found in the LICENSE file. import json import os import subprocess import sys import re from optparse import OptionParser # This script runs pkg-config, optionally filtering out some results, and # returns the result. # # The result will be [ , , , , ] # where each member is itself a list of strings. # # You can filter out matches using "-v " where all results from # pkgconfig matching the given regular expression will be ignored. You can # specify more than one regular expression my specifying "-v" more than once. # # You can specify a sysroot using "-s " where sysroot is the absolute # system path to the sysroot used for compiling. This script will attempt to # generate correct paths for the sysroot. # # When using a sysroot, you must also specify the architecture via # "-a " where arch is either "x86" or "x64". # # Additionally, you can specify the option --atleast-version. This will skip # the normal outputting of a dictionary and instead print true or false, # depending on the return value of pkg-config for the given package. # If this is run on non-Linux platforms, just return nothing and indicate # success. This allows us to "kind of emulate" a Linux build from other # platforms. if sys.platform.find("linux") == -1: print "[[],[],[],[],[]]" sys.exit(0) def SetConfigPath(options): """Set the PKG_CONFIG_PATH environment variable. This takes into account any sysroot and architecture specification from the options on the given command line.""" sysroot = options.sysroot if not sysroot: sysroot = "" # Compute the library path name based on the architecture. arch = options.arch if sysroot and not arch: print "You must specify an architecture via -a if using a sysroot." sys.exit(1) # In the gyp world this is configurable via the 'system_libdir' variable, # which doesn't seem to have an equivelent in gn yet. # TOOD(sbc): Make this configurable like it is under gyp. libpath = 'lib' # Add the sysroot path to the environment's PKG_CONFIG_PATH config_path = sysroot + '/usr/' + libpath + '/pkgconfig' config_path += ':' + sysroot + '/usr/share/pkgconfig' if 'PKG_CONFIG_PATH' in os.environ: os.environ['PKG_CONFIG_PATH'] += ':' + config_path else: os.environ['PKG_CONFIG_PATH'] = config_path def GetPkgConfigPrefixToStrip(args): """Returns the prefix from pkg-config where packages are installed. This returned prefix is the one that should be stripped from the beginning of directory names to take into account sysroots.""" # Some sysroots, like the Chromium OS ones, may generate paths that are not # relative to the sysroot. For example, # /path/to/chroot/build/x86-generic/usr/lib/pkgconfig/pkg.pc may have all # paths relative to /path/to/chroot (i.e. prefix=/build/x86-generic/usr) # instead of relative to /path/to/chroot/build/x86-generic (i.e prefix=/usr). # To support this correctly, it's necessary to extract the prefix to strip # from pkg-config's |prefix| variable. prefix = subprocess.check_output(["pkg-config", "--variable=prefix"] + args, env=os.environ) if prefix[-4] == '/usr': return prefix[4:] return prefix def MatchesAnyRegexp(flag, list_of_regexps): """Returns true if the first argument matches any regular expression in the given list.""" for regexp in list_of_regexps: if regexp.search(flag) != None: return True return False def RewritePath(path, strip_prefix, sysroot): """Rewrites a path by stripping the prefix and prepending the sysroot.""" if os.path.isabs(path) and not path.startswith(sysroot): if path.startswith(strip_prefix): path = path[len(strip_prefix):] path = path.lstrip('/') return os.path.join(sysroot, path) else: return path parser = OptionParser() parser.add_option('-p', action='store', dest='pkg_config', type='string', default='pkg-config') parser.add_option('-v', action='append', dest='strip_out', type='string') parser.add_option('-s', action='store', dest='sysroot', type='string') parser.add_option('-a', action='store', dest='arch', type='string') parser.add_option('--atleast-version', action='store', dest='atleast_version', type='string') parser.add_option('--libdir', action='store_true', dest='libdir') (options, args) = parser.parse_args() # Make a list of regular expressions to strip out. strip_out = [] if options.strip_out != None: for regexp in options.strip_out: strip_out.append(re.compile(regexp)) SetConfigPath(options) if options.sysroot: prefix = GetPkgConfigPrefixToStrip(args) else: prefix = '' if options.atleast_version: # When asking for the return value, just run pkg-config and print the return # value, no need to do other work. if not subprocess.call([options.pkg_config, "--atleast-version=" + options.atleast_version] + args, env=os.environ): print "true" else: print "false" sys.exit(0) if options.libdir: try: libdir = subprocess.check_output([options.pkg_config, "--variable=libdir"] + args, env=os.environ) except: print "Error from pkg-config." sys.exit(1) sys.stdout.write(libdir.strip()) sys.exit(0) try: flag_string = subprocess.check_output( [ options.pkg_config, "--cflags", "--libs" ] + args, env=os.environ) # For now just split on spaces to get the args out. This will break if # pkgconfig returns quoted things with spaces in them, but that doesn't seem # to happen in practice. all_flags = flag_string.strip().split(' ') except: print "Could not run pkg-config." sys.exit(1) sysroot = options.sysroot if not sysroot: sysroot = '' includes = [] cflags = [] libs = [] lib_dirs = [] ldflags = [] for flag in all_flags[:]: if len(flag) == 0 or MatchesAnyRegexp(flag, strip_out): continue; if flag[:2] == '-l': libs.append(RewritePath(flag[2:], prefix, sysroot)) elif flag[:2] == '-L': lib_dirs.append(RewritePath(flag[2:], prefix, sysroot)) elif flag[:2] == '-I': includes.append(RewritePath(flag[2:], prefix, sysroot)) elif flag[:3] == '-Wl': ldflags.append(flag) elif flag == '-pthread': # Many libs specify "-pthread" which we don't need since we always include # this anyway. Removing it here prevents a bunch of duplicate inclusions on # the command line. pass else: cflags.append(flag) # Output a GN array, the first one is the cflags, the second are the libs. The # JSON formatter prints GN compatible lists when everything is a list of # strings. print json.dumps([includes, cflags, libs, lib_dirs, ldflags])