135 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			135 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
| #!/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])
 |