111 lines
3.1 KiB
Python
111 lines
3.1 KiB
Python
|
#!/usr/bin/env python
|
||
|
#
|
||
|
# Copyright 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.
|
||
|
|
||
|
"""Creates a TOC file from a Java jar.
|
||
|
|
||
|
The TOC file contains the non-package API of the jar. This includes all
|
||
|
public/protected classes/functions/members and the values of static final
|
||
|
variables. Some other information (major/minor javac version) is also included.
|
||
|
|
||
|
This TOC file then can be used to determine if a dependent library should be
|
||
|
rebuilt when this jar changes. I.e. any change to the jar that would require a
|
||
|
rebuild, will have a corresponding change in the TOC file.
|
||
|
"""
|
||
|
|
||
|
import optparse
|
||
|
import os
|
||
|
import re
|
||
|
import sys
|
||
|
import zipfile
|
||
|
|
||
|
from util import build_utils
|
||
|
from util import md5_check
|
||
|
|
||
|
|
||
|
def GetClassesInZipFile(zip_file):
|
||
|
classes = []
|
||
|
files = zip_file.namelist()
|
||
|
for f in files:
|
||
|
if f.endswith('.class'):
|
||
|
# f is of the form org/chromium/base/Class$Inner.class
|
||
|
classes.append(f.replace('/', '.')[:-6])
|
||
|
return classes
|
||
|
|
||
|
|
||
|
def CallJavap(classpath, classes):
|
||
|
javap_cmd = [
|
||
|
'javap',
|
||
|
'-protected', # In reality both public & protected.
|
||
|
# -verbose is required to get constant values (which can be inlined in
|
||
|
# dependents).
|
||
|
'-verbose',
|
||
|
'-classpath', classpath
|
||
|
] + classes
|
||
|
return build_utils.CheckCallDie(javap_cmd, suppress_output=True)
|
||
|
|
||
|
|
||
|
def ExtractToc(disassembled_classes):
|
||
|
# javap output is structured by indent (2-space) levels.
|
||
|
good_patterns = [
|
||
|
'^[^ ]', # This includes all class/function/member signatures.
|
||
|
'^ SourceFile:',
|
||
|
'^ minor version:',
|
||
|
'^ major version:',
|
||
|
'^ Constant value:',
|
||
|
]
|
||
|
bad_patterns = [
|
||
|
'^const #', # Matches the constant pool (i.e. literals used in the class).
|
||
|
]
|
||
|
|
||
|
def JavapFilter(line):
|
||
|
return (re.match('|'.join(good_patterns), line) and
|
||
|
not re.match('|'.join(bad_patterns), line))
|
||
|
toc = filter(JavapFilter, disassembled_classes.split('\n'))
|
||
|
|
||
|
return '\n'.join(toc)
|
||
|
|
||
|
|
||
|
def UpdateToc(jar_path, toc_path):
|
||
|
classes = GetClassesInZipFile(zipfile.ZipFile(jar_path))
|
||
|
javap_output = CallJavap(classpath=jar_path, classes=classes)
|
||
|
toc = ExtractToc(javap_output)
|
||
|
|
||
|
with open(toc_path, 'w') as tocfile:
|
||
|
tocfile.write(toc)
|
||
|
|
||
|
|
||
|
def DoJarToc(options):
|
||
|
jar_path = options.jar_path
|
||
|
toc_path = options.toc_path
|
||
|
record_path = '%s.md5.stamp' % toc_path
|
||
|
md5_check.CallAndRecordIfStale(
|
||
|
lambda: UpdateToc(jar_path, toc_path),
|
||
|
record_path=record_path,
|
||
|
input_paths=[jar_path],
|
||
|
)
|
||
|
build_utils.Touch(toc_path)
|
||
|
|
||
|
|
||
|
def main(argv):
|
||
|
parser = optparse.OptionParser()
|
||
|
parser.add_option('--jar-path', help='Input .jar path.')
|
||
|
parser.add_option('--toc-path', help='Output .jar.TOC path.')
|
||
|
parser.add_option('--stamp', help='Path to touch on success.')
|
||
|
|
||
|
# TODO(newt): remove this once http://crbug.com/177552 is fixed in ninja.
|
||
|
parser.add_option('--ignore', help='Ignored.')
|
||
|
|
||
|
options, _ = parser.parse_args()
|
||
|
|
||
|
DoJarToc(options)
|
||
|
|
||
|
if options.stamp:
|
||
|
build_utils.Touch(options.stamp)
|
||
|
|
||
|
|
||
|
if __name__ == '__main__':
|
||
|
sys.exit(main(sys.argv))
|