#!/usr/bin/env python3
import argparse
import csv
import glob
import json
import os
import sys
HELP_MSG = '''
This script computes the differences between two system images (system1 -
system2), and lists the files grouped by package. The difference is just based
on the existence of the file, not on its contents.
'''
VENDOR_PATH_MAP = {
'vendor/google' : 'Google',
'vendor/unbundled_google': 'Google',
'vendor/verizon' : 'Verizon',
'vendor/qcom' : 'Qualcomm',
'vendor/tmobile' : 'TMobile',
'vendor/mediatek' : 'Mediatek',
'vendor/htc' : 'HTC',
'vendor/realtek' : 'Realtek'
}
def _get_relative_out_path_from_root(out_path):
"""Given a path to a target out directory, get the relative path from the
Android root.
The module-info.json file paths are relative to the root source folder
ie. one directory before out."""
system_path = os.path.normpath(os.path.join(out_path, 'system'))
system_path_dirs = system_path.split(os.sep)
out_index = system_path_dirs.index("out")
return os.path.join(*system_path_dirs[out_index:])
def system_files(path):
"""Returns an array of the files under /system, recursively, and ignoring
symbolic-links"""
system_files = []
system_prefix = os.path.join(path, 'system')
# Skip trailing '/'
system_prefix_len = len(system_prefix) + 1
for root, dirs, files in os.walk(system_prefix, topdown=True):
for file in files:
# Ignore symbolic links.
if not os.path.islink(os.path.join(root, file)):
system_files.append(os.path.join(root[system_prefix_len:], file))
return system_files
def system_files_to_package_map(path):
"""Returns a dictionary mapping from each file in the /system partition to its
package, according to modules-info.json."""
system_files_to_package_map = {}
system_prefix = _get_relative_out_path_from_root(path)
# Skip trailing '/'
system_prefix_len = len(system_prefix) + 1
with open(os.path.join(path, 'module-info.json')) as module_info_json:
module_info = json.load(module_info_json)
for module in module_info:
installs = module_info[module]['installed']
for install in installs:
if install.startswith(system_prefix):
system_file = install[system_prefix_len:]
# Not clear if collisions can ever happen in modules-info.json (e.g.
# the same file installed by multiple packages), but it doesn't hurt
# to check.
if system_file in system_files_to_package_map:
system_files_to_package_map[system_file] = "--multiple--"
else:
system_files_to_package_map[system_file] = module
return system_files_to_package_map
def package_to_vendor_map(path):
"""Returns a dictionary mapping from each package in modules-info.json to its
vendor. If a vendor cannot be found, it maps to "--unknown--". Those cases
are:
1. The package maps to multiple modules (e.g., one in APPS and one in
SHARED_LIBRARIES.
2. The path to the module is not one of the recognized vendor paths in
VENDOR_PATH_MAP."""
package_vendor_map = {}
system_prefix = os.path.join(path, 'system')
# Skip trailing '/'
system_prefix_len = len(system_prefix) + 1
vendor_prefixes = VENDOR_PATH_MAP.keys()
with open(os.path.join(path, 'module-info.json')) as module_info_json:
module_info = json.load(module_info_json)
for module in module_info:
paths = module_info[module]['path']
vendor = ""
if len(paths) == 1:
path = paths[0]
for prefix in vendor_prefixes:
if path.startswith(prefix):
vendor = VENDOR_PATH_MAP[prefix]
break
if vendor == "":
vendor = "--unknown--"
else:
vendor = "--multiple--"
package_vendor_map[module] = vendor
return package_vendor_map
def main():
parser = argparse.ArgumentParser(description=HELP_MSG)
parser.add_argument("out1", help="First $OUT directory")
parser.add_argument("out2", help="Second $OUT directory")
args = parser.parse_args()
system_files1 = system_files(args.out1)
system_files2 = system_files(args.out2)
system_files_diff = set(system_files1) - set(system_files2)
system_files_map = system_files_to_package_map(args.out1)
package_vendor_map = package_to_vendor_map(args.out1)
packages = {}
for file in system_files_diff:
if file in system_files_map:
package = system_files_map[file]
else:
package = "--unknown--"
if package in packages:
packages[package].append(file)
else:
packages[package] = [file]
with open(os.path.join(args.out1, 'module-info.json')) as module_info_json:
module_info = json.load(module_info_json)
writer = csv.writer(sys.stdout, quoting = csv.QUOTE_NONNUMERIC,
delimiter = ',', lineterminator = '\n')
for package, files in packages.iteritems():
for file in files:
# Group sources of the deltas.
if package in package_vendor_map:
vendor = package_vendor_map[package]
else:
vendor = "--unknown--"
# Get file size.
full_path = os.path.join(args.out1, 'system', file)
size = os.stat(full_path).st_size
if package in module_info.keys():
module_path = module_info[package]['path']
else:
module_path = ''
writer.writerow([
# File that exists in out1 but not out2.
file,
# Module name that the file came from.
package,
# Path to the module.
module_path,
# File size.
size,
# Vendor owner.
vendor])
if __name__ == '__main__':
main()