1*333d2b36SAndroid Build Coastguard Worker#!/usr/bin/env python 2*333d2b36SAndroid Build Coastguard Worker# 3*333d2b36SAndroid Build Coastguard Worker# Copyright (C) 2020 The Android Open Source Project 4*333d2b36SAndroid Build Coastguard Worker# 5*333d2b36SAndroid Build Coastguard Worker# Licensed under the Apache License, Version 2.0 (the "License"); 6*333d2b36SAndroid Build Coastguard Worker# you may not use this file except in compliance with the License. 7*333d2b36SAndroid Build Coastguard Worker# You may obtain a copy of the License at 8*333d2b36SAndroid Build Coastguard Worker# 9*333d2b36SAndroid Build Coastguard Worker# http://www.apache.org/licenses/LICENSE-2.0 10*333d2b36SAndroid Build Coastguard Worker# 11*333d2b36SAndroid Build Coastguard Worker# Unless required by applicable law or agreed to in writing, software 12*333d2b36SAndroid Build Coastguard Worker# distributed under the License is distributed on an "AS IS" BASIS, 13*333d2b36SAndroid Build Coastguard Worker# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 14*333d2b36SAndroid Build Coastguard Worker# See the License for the specific language governing permissions and 15*333d2b36SAndroid Build Coastguard Worker# limitations under the License. 16*333d2b36SAndroid Build Coastguard Worker# 17*333d2b36SAndroid Build Coastguard Worker"""A tool for constructing class loader context.""" 18*333d2b36SAndroid Build Coastguard Worker 19*333d2b36SAndroid Build Coastguard Workerimport argparse 20*333d2b36SAndroid Build Coastguard Workerimport json 21*333d2b36SAndroid Build Coastguard Workerimport sys 22*333d2b36SAndroid Build Coastguard Worker 23*333d2b36SAndroid Build Coastguard Workerfrom manifest import compare_version_gt 24*333d2b36SAndroid Build Coastguard Worker 25*333d2b36SAndroid Build Coastguard Worker 26*333d2b36SAndroid Build Coastguard Workerdef parse_args(args): 27*333d2b36SAndroid Build Coastguard Worker """Parse commandline arguments.""" 28*333d2b36SAndroid Build Coastguard Worker parser = argparse.ArgumentParser() 29*333d2b36SAndroid Build Coastguard Worker parser.add_argument( 30*333d2b36SAndroid Build Coastguard Worker '--target-sdk-version', 31*333d2b36SAndroid Build Coastguard Worker default='', 32*333d2b36SAndroid Build Coastguard Worker dest='sdk', 33*333d2b36SAndroid Build Coastguard Worker help='specify target SDK version (as it appears in the manifest)') 34*333d2b36SAndroid Build Coastguard Worker parser.add_argument( 35*333d2b36SAndroid Build Coastguard Worker '--context-json', 36*333d2b36SAndroid Build Coastguard Worker default='', 37*333d2b36SAndroid Build Coastguard Worker dest='context_json', 38*333d2b36SAndroid Build Coastguard Worker ) 39*333d2b36SAndroid Build Coastguard Worker parser.add_argument( 40*333d2b36SAndroid Build Coastguard Worker '--product-packages', 41*333d2b36SAndroid Build Coastguard Worker default='', 42*333d2b36SAndroid Build Coastguard Worker dest='product_packages_file', 43*333d2b36SAndroid Build Coastguard Worker ) 44*333d2b36SAndroid Build Coastguard Worker return parser.parse_args(args) 45*333d2b36SAndroid Build Coastguard Worker 46*333d2b36SAndroid Build Coastguard Worker 47*333d2b36SAndroid Build Coastguard Worker# Special keyword that means that the context should be added to class loader 48*333d2b36SAndroid Build Coastguard Worker# context regardless of the target SDK version. 49*333d2b36SAndroid Build Coastguard Workerany_sdk = 'any' 50*333d2b36SAndroid Build Coastguard Worker 51*333d2b36SAndroid Build Coastguard Workercontext_sep = '#' 52*333d2b36SAndroid Build Coastguard Worker 53*333d2b36SAndroid Build Coastguard Worker 54*333d2b36SAndroid Build Coastguard Workerdef encode_class_loader(context, product_packages): 55*333d2b36SAndroid Build Coastguard Worker host_sub_contexts, target_sub_contexts = encode_class_loaders( 56*333d2b36SAndroid Build Coastguard Worker context['Subcontexts'], product_packages) 57*333d2b36SAndroid Build Coastguard Worker 58*333d2b36SAndroid Build Coastguard Worker return ('PCL[%s]%s' % (context['Host'], host_sub_contexts), 59*333d2b36SAndroid Build Coastguard Worker 'PCL[%s]%s' % (context['Device'], target_sub_contexts)) 60*333d2b36SAndroid Build Coastguard Worker 61*333d2b36SAndroid Build Coastguard Worker 62*333d2b36SAndroid Build Coastguard Workerdef encode_class_loaders(contexts, product_packages): 63*333d2b36SAndroid Build Coastguard Worker host_contexts = [] 64*333d2b36SAndroid Build Coastguard Worker target_contexts = [] 65*333d2b36SAndroid Build Coastguard Worker 66*333d2b36SAndroid Build Coastguard Worker for context in contexts: 67*333d2b36SAndroid Build Coastguard Worker if not context['Optional'] or context['Name'] in product_packages: 68*333d2b36SAndroid Build Coastguard Worker host_context, target_context = encode_class_loader( 69*333d2b36SAndroid Build Coastguard Worker context, product_packages) 70*333d2b36SAndroid Build Coastguard Worker host_contexts.append(host_context) 71*333d2b36SAndroid Build Coastguard Worker target_contexts.append(target_context) 72*333d2b36SAndroid Build Coastguard Worker 73*333d2b36SAndroid Build Coastguard Worker if host_contexts: 74*333d2b36SAndroid Build Coastguard Worker return ('{%s}' % context_sep.join(host_contexts), 75*333d2b36SAndroid Build Coastguard Worker '{%s}' % context_sep.join(target_contexts)) 76*333d2b36SAndroid Build Coastguard Worker else: 77*333d2b36SAndroid Build Coastguard Worker return '', '' 78*333d2b36SAndroid Build Coastguard Worker 79*333d2b36SAndroid Build Coastguard Worker 80*333d2b36SAndroid Build Coastguard Workerdef construct_context_args(target_sdk, context_json, product_packages): 81*333d2b36SAndroid Build Coastguard Worker all_contexts = [] 82*333d2b36SAndroid Build Coastguard Worker 83*333d2b36SAndroid Build Coastguard Worker # CLC for different SDK versions should come in specific order that agrees 84*333d2b36SAndroid Build Coastguard Worker # with PackageManager. Since PackageManager processes SDK versions in 85*333d2b36SAndroid Build Coastguard Worker # ascending order and prepends compatibility libraries at the front, the 86*333d2b36SAndroid Build Coastguard Worker # required order is descending, except for any_sdk that has numerically 87*333d2b36SAndroid Build Coastguard Worker # the largest order, but must be the last one. Example of correct order: 88*333d2b36SAndroid Build Coastguard Worker # [30, 29, 28, any_sdk]. There are Python tests to ensure that someone 89*333d2b36SAndroid Build Coastguard Worker # doesn't change this by accident, but there is no way to guard against 90*333d2b36SAndroid Build Coastguard Worker # changes in the PackageManager, except for grepping logcat on the first 91*333d2b36SAndroid Build Coastguard Worker # boot for absence of the following messages: 92*333d2b36SAndroid Build Coastguard Worker # 93*333d2b36SAndroid Build Coastguard Worker # `logcat | grep -E 'ClassLoaderContext [a-z ]+ mismatch` 94*333d2b36SAndroid Build Coastguard Worker 95*333d2b36SAndroid Build Coastguard Worker for sdk, contexts in sorted( 96*333d2b36SAndroid Build Coastguard Worker ((sdk, contexts) 97*333d2b36SAndroid Build Coastguard Worker for sdk, contexts in context_json.items() 98*333d2b36SAndroid Build Coastguard Worker if sdk != any_sdk and compare_version_gt(sdk, target_sdk)), 99*333d2b36SAndroid Build Coastguard Worker key=lambda item: int(item[0]), reverse=True): 100*333d2b36SAndroid Build Coastguard Worker all_contexts += contexts 101*333d2b36SAndroid Build Coastguard Worker 102*333d2b36SAndroid Build Coastguard Worker if any_sdk in context_json: 103*333d2b36SAndroid Build Coastguard Worker all_contexts += context_json[any_sdk] 104*333d2b36SAndroid Build Coastguard Worker 105*333d2b36SAndroid Build Coastguard Worker host_contexts, target_contexts = encode_class_loaders( 106*333d2b36SAndroid Build Coastguard Worker all_contexts, product_packages) 107*333d2b36SAndroid Build Coastguard Worker 108*333d2b36SAndroid Build Coastguard Worker return ( 109*333d2b36SAndroid Build Coastguard Worker 'class_loader_context_arg=--class-loader-context=PCL[]%s ; ' % 110*333d2b36SAndroid Build Coastguard Worker host_contexts + 111*333d2b36SAndroid Build Coastguard Worker 'stored_class_loader_context_arg=' 112*333d2b36SAndroid Build Coastguard Worker '--stored-class-loader-context=PCL[]%s' 113*333d2b36SAndroid Build Coastguard Worker % target_contexts) 114*333d2b36SAndroid Build Coastguard Worker 115*333d2b36SAndroid Build Coastguard Worker 116*333d2b36SAndroid Build Coastguard Workerdef main(): 117*333d2b36SAndroid Build Coastguard Worker """Program entry point.""" 118*333d2b36SAndroid Build Coastguard Worker try: 119*333d2b36SAndroid Build Coastguard Worker args = parse_args(sys.argv[1:]) 120*333d2b36SAndroid Build Coastguard Worker if not args.sdk: 121*333d2b36SAndroid Build Coastguard Worker raise SystemExit('target sdk version is not set') 122*333d2b36SAndroid Build Coastguard Worker 123*333d2b36SAndroid Build Coastguard Worker context_json = json.loads(args.context_json) 124*333d2b36SAndroid Build Coastguard Worker with open(args.product_packages_file, 'r') as f: 125*333d2b36SAndroid Build Coastguard Worker product_packages = set(line.strip() for line in f if line.strip()) 126*333d2b36SAndroid Build Coastguard Worker 127*333d2b36SAndroid Build Coastguard Worker print(construct_context_args(args.sdk, context_json, product_packages)) 128*333d2b36SAndroid Build Coastguard Worker 129*333d2b36SAndroid Build Coastguard Worker # pylint: disable=broad-except 130*333d2b36SAndroid Build Coastguard Worker except Exception as err: 131*333d2b36SAndroid Build Coastguard Worker print('error: ' + str(err), file=sys.stderr) 132*333d2b36SAndroid Build Coastguard Worker sys.exit(-1) 133*333d2b36SAndroid Build Coastguard Worker 134*333d2b36SAndroid Build Coastguard Worker 135*333d2b36SAndroid Build Coastguard Workerif __name__ == '__main__': 136*333d2b36SAndroid Build Coastguard Worker main() 137