xref: /aosp_15_r20/build/soong/scripts/construct_context.py (revision 333d2b3687b3a337dbcca9d65000bca186795e39)
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