|  | #!/usr/bin/env python | 
|  | # | 
|  | # Copyright (C) 2016 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. | 
|  |  | 
|  | """Tool to prioritize which modules to convert to Soong. | 
|  |  | 
|  | Generally, you'd use this through the make integration, which automatically | 
|  | generates the CSV input file that this tool expects: | 
|  |  | 
|  | $ m $OUT/soong_to_convert.txt | 
|  | $ less $OUT/soong_to_convert.txt | 
|  |  | 
|  | The output is a list of modules that are probably ready to convert to Soong: | 
|  |  | 
|  | # Blocked on Module (potential problems) | 
|  | 283 libEGL (srcs_dotarm) | 
|  | 246 libicuuc (dotdot_incs dotdot_srcs) | 
|  | 221 libspeexresampler | 
|  | 215 libcamera_metadata | 
|  | ... | 
|  | 0 zram-perf (dotdot_incs) | 
|  |  | 
|  | The number at the beginning of the line shows how many native modules depend | 
|  | on that module. | 
|  |  | 
|  | All of their dependencies have been satisfied, and any potential problems | 
|  | that Make can detect are listed in parenthesis after the module: | 
|  |  | 
|  | dotdot_srcs: LOCAL_SRC_FILES contains paths outside $(LOCAL_PATH) | 
|  | dotdot_incs: LOCAL_C_INCLUDES contains paths include '..' | 
|  | srcs_dotarm: LOCAL_SRC_FILES contains source files like <...>.c.arm | 
|  | aidl: LOCAL_SRC_FILES contains .aidl sources | 
|  | objc: LOCAL_SRC_FILES contains Objective-C sources | 
|  | proto: LOCAL_SRC_FILES contains .proto sources | 
|  | rs: LOCAL_SRC_FILES contains renderscript sources | 
|  | vts: LOCAL_SRC_FILES contains .vts sources | 
|  |  | 
|  | Not all problems can be discovered, but this is a starting point. | 
|  |  | 
|  | """ | 
|  |  | 
|  | from __future__ import print_function | 
|  |  | 
|  | import csv | 
|  | import sys | 
|  |  | 
|  | def count_deps(depsdb, module, seen): | 
|  | """Based on the depsdb, count the number of transitive dependencies. | 
|  |  | 
|  | You can pass in an reversed dependency graph to conut the number of | 
|  | modules that depend on the module.""" | 
|  | count = 0 | 
|  | seen.append(module) | 
|  | if module in depsdb: | 
|  | for dep in depsdb[module]: | 
|  | if dep in seen: | 
|  | continue | 
|  | count += 1 + count_deps(depsdb, dep, seen) | 
|  | return count | 
|  |  | 
|  | def process(reader): | 
|  | """Read the input file and produce a list of modules ready to move to Soong | 
|  | """ | 
|  | problems = dict() | 
|  | deps = dict() | 
|  | reverse_deps = dict() | 
|  | module_types = dict() | 
|  |  | 
|  | for (module, module_type, problem, dependencies) in reader: | 
|  | module_types[module] = module_type | 
|  | problems[module] = problem | 
|  | deps[module] = [d for d in dependencies.strip().split(' ') if d != ""] | 
|  | for dep in deps[module]: | 
|  | if not dep in reverse_deps: | 
|  | reverse_deps[dep] = [] | 
|  | reverse_deps[dep].append(module) | 
|  |  | 
|  | results = [] | 
|  | for module in problems: | 
|  | # Only display actionable conversions, ones without missing dependencies | 
|  | if len(deps[module]) != 0: | 
|  | continue | 
|  |  | 
|  | extra = "" | 
|  | if len(problems[module]) > 0: | 
|  | extra = " ({})".format(problems[module]) | 
|  | results.append((count_deps(reverse_deps, module, []), module + extra, module_types[module])) | 
|  |  | 
|  | return sorted(results, key=lambda result: (-result[0], result[1])) | 
|  |  | 
|  | def filter(results, module_type): | 
|  | return [x for x in results if x[2] == module_type] | 
|  |  | 
|  | def display(results): | 
|  | """Displays the results""" | 
|  | count_header = "# Blocked on" | 
|  | count_width = len(count_header) | 
|  | print("{} Module (potential problems)".format(count_header)) | 
|  | for (count, module, module_type) in results: | 
|  | print("{:>{}} {}".format(count, count_width, module)) | 
|  |  | 
|  | def main(filename): | 
|  | """Read the CSV file, print the results""" | 
|  | with open(filename, 'rb') as csvfile: | 
|  | results = process(csv.reader(csvfile)) | 
|  |  | 
|  | native_results = filter(results, "native") | 
|  | java_results = filter(results, "java") | 
|  |  | 
|  | print("native modules ready to convert") | 
|  | display(native_results) | 
|  |  | 
|  | print("") | 
|  | print("java modules ready to convert") | 
|  | display(java_results) | 
|  |  | 
|  | if __name__ == "__main__": | 
|  | if len(sys.argv) != 2: | 
|  | print("usage: soong_conversion.py <file>", file=sys.stderr) | 
|  | sys.exit(1) | 
|  |  | 
|  | main(sys.argv[1]) |