#!/usr/bin/env python
#
# Copyright (C) 2011-2012 The Android Open Source Project
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
import datetime
import os
import re
import sys
import subprocess
try:
import hashlib
sha1 = hashlib.sha1
except ImportError, e:
import sha
sha1 = sha.sha
def get_repo_revision(repo_dir):
if not os.path.exists(os.path.join(repo_dir, '.git')):
return 'Unknown (not git)'
# Get the HEAD revision
proc = subprocess.Popen(['git', 'log', '-1', '--format=%H'],
stdout=subprocess.PIPE,
stderr=subprocess.PIPE,
cwd=repo_dir)
out, err = proc.communicate()
proc.wait()
rev_sha1 = out.strip()
# Working Directory Modified
proc = subprocess.Popen(['git', 'status'],
stdout=subprocess.PIPE,
stderr=subprocess.PIPE,
cwd=repo_dir)
out, err = proc.communicate()
proc.wait()
if out.find('(working directory clean)') == -1:
mod = ' modified'
else:
mod = ''
return rev_sha1 + mod + ' (git)'
def compute_sha1(path, global_hasher = None):
f = open(path, 'rb')
hasher = sha1()
while True:
buf = f.read(512)
hasher.update(buf)
if global_hasher:
global_hasher.update(buf)
if len(buf) < 512:
break
f.close()
return hasher.hexdigest()
def compute_sha1_list(paths):
hasher = sha1()
sha1sums = []
for path in paths:
sha1sums.append(compute_sha1(path, hasher))
return (hasher.hexdigest(), sha1sums)
def quote_str(s):
result = '"'
for c in s:
if c == '\\':
result += '\\\\'
elif c == '\r':
result += '\\r'
elif c == '\n':
result += '\\n'
elif c == '\t':
result += '\\t'
elif c == '\"':
result += '\\"'
elif c == '\'':
result += '\\\''
else:
result += c
result += '"'
return result
def main():
# Check Argument
if len(sys.argv) < 2:
print >> sys.stderr, 'USAGE:', sys.argv[0], '[REPO] [LIBs]'
sys.exit(1)
# Record Build Time
build_time = datetime.datetime.now().strftime('%Y/%m/%d %H:%M:%S')
# Repository Directory (For build revision)
repo_dir = sys.argv[1]
build_rev = get_repo_revision(repo_dir)
# Compute SHA1
lib_list = list(set(sys.argv[2:]))
lib_list.sort()
build_sha1, sha1sum_list = compute_sha1_list(lib_list)
# Build file list string
lib_list_str = ''
for i, path in enumerate(lib_list):
lib_list_str += ' %s %s\n' % (sha1sum_list[i], path)
# Print the automatically generated code
print """/* Automatically generated file (DON'T MODIFY) */
/* Repository directory: %s */
/* File list:
%s*/
#include "bcc/Config/BuildInfo.h"
using namespace bcc;
const char* BuildInfo::GetBuildTime() {
return %s;
}
const char *BuildInfo::GetBuildRev() {
return %s;
}
const char *BuildInfo::GetBuildSourceBlob() {
return %s;
}
""" % (os.path.abspath(repo_dir),
lib_list_str,
quote_str(build_time),
quote_str(build_rev),
quote_str(build_sha1))
if __name__ == '__main__':
main()