#!/usr/bin/env python # Generate version information for a program # # Copyright (C) 2015 Kevin O'Connor # # This file may be distributed under the terms of the GNU GPLv3 license. import sys, os, subprocess, shlex, time, socket, optparse, logging, traceback VERSION_FORMAT = """ /* DO NOT EDIT! This is an autogenerated file. See scripts/buildversion.py. */ #define BUILD_VERSION "%s" #define BUILD_TOOLS "%s" """ # Run program and return the specified output def check_output(prog): logging.debug("Running %s" % (repr(prog),)) try: process = subprocess.Popen(shlex.split(prog), stdout=subprocess.PIPE) output = process.communicate()[0] retcode = process.poll() except OSError: logging.debug("Exception on run: %s" % (traceback.format_exc(),)) return "" logging.debug("Got (code=%s): %s" % (retcode, repr(output))) if retcode: return "" try: return output.decode() except UnicodeError: logging.debug("Exception on decode: %s" % (traceback.format_exc(),)) return "" # Obtain version info from "git" program def git_version(): if not os.path.exists('.git'): logging.debug("No '.git' file/directory found") return "" ver = check_output("git describe --tags --long --dirty").strip() logging.debug("Got git version: %s" % (repr(ver),)) return ver # Look for version in a ".version" file. Official release tarballs # have this file (see scripts/tarball.sh). def file_version(): if not os.path.isfile('.version'): logging.debug("No '.version' file found") return "" try: f = open('.version', 'r') ver = f.readline().strip() f.close() except OSError: logging.debug("Exception on read: %s" % (traceback.format_exc(),)) return "" logging.debug("Got .version: %s" % (repr(ver),)) return ver # Generate an output file with the version information def write_version(outfile, version, toolstr): logging.debug("Write file %s and %s" % (repr(version), repr(toolstr))) sys.stdout.write("Version: %s\n" % (version,)) f = open(outfile, 'w') f.write(VERSION_FORMAT % (version, toolstr)) f.close() # Run "tool --version" for each specified tool and extract versions def tool_versions(tools): tools = [t.strip() for t in tools.split(';')] versions = ['', ''] success = 0 for tool in tools: # Extract first line from "tool --version" output verstr = check_output("%s --version" % (tool,)).split('\n')[0] # Check if this tool looks like a binutils program isbinutils = 0 if verstr.startswith('GNU '): isbinutils = 1 verstr = verstr[4:] # Extract version information and exclude program name if ' ' not in verstr: continue prog, ver = verstr.split(' ', 1) if not prog or not ver: continue # Check for any version conflicts if versions[isbinutils] and versions[isbinutils] != ver: logging.debug("Mixed version %s vs %s" % ( repr(versions[isbinutils]), repr(ver))) versions[isbinutils] = "mixed" continue versions[isbinutils] = ver success += 1 cleanbuild = versions[0] and versions[1] and success == len(tools) return cleanbuild, "gcc: %s binutils: %s" % (versions[0], versions[1]) def main(): usage = "%prog [options] " opts = optparse.OptionParser(usage) opts.add_option("-e", "--extra", dest="extra", default="", help="extra version string to append to version") opts.add_option("-t", "--tools", dest="tools", default="", help="list of build programs to extract version from") opts.add_option("-v", action="store_true", dest="verbose", help="enable debug messages") options, args = opts.parse_args() if len(args) != 1: opts.error("Incorrect arguments") outfile = args[0] if options.verbose: logging.basicConfig(level=logging.DEBUG) cleanbuild, toolstr = tool_versions(options.tools) ver = git_version() cleanbuild = cleanbuild and 'dirty' not in ver if not ver: ver = file_version() # We expect the "extra version" to contain information on the # distributor and distribution package version (if # applicable). It is a "clean" build if this is a build from # an official release tarball and the above info is present. cleanbuild = cleanbuild and ver and options.extra != "" if not ver: ver = "?" if not cleanbuild: btime = time.strftime("%Y%m%d_%H%M%S") hostname = socket.gethostname() ver = "%s-%s-%s" % (ver, btime, hostname) write_version(outfile, ver + options.extra, toolstr) if __name__ == '__main__': main()