xref: /aosp_15_r20/build/soong/apex/vndk.go (revision 333d2b3687b3a337dbcca9d65000bca186795e39)
1*333d2b36SAndroid Build Coastguard Worker// Copyright (C) 2019 The Android Open Source Project
2*333d2b36SAndroid Build Coastguard Worker//
3*333d2b36SAndroid Build Coastguard Worker// Licensed under the Apache License, Version 2.0 (the "License");
4*333d2b36SAndroid Build Coastguard Worker// you may not use this file except in compliance with the License.
5*333d2b36SAndroid Build Coastguard Worker// You may obtain a copy of the License at
6*333d2b36SAndroid Build Coastguard Worker//
7*333d2b36SAndroid Build Coastguard Worker//     http://www.apache.org/licenses/LICENSE-2.0
8*333d2b36SAndroid Build Coastguard Worker//
9*333d2b36SAndroid Build Coastguard Worker// Unless required by applicable law or agreed to in writing, software
10*333d2b36SAndroid Build Coastguard Worker// distributed under the License is distributed on an "AS IS" BASIS,
11*333d2b36SAndroid Build Coastguard Worker// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12*333d2b36SAndroid Build Coastguard Worker// See the License for the specific language governing permissions and
13*333d2b36SAndroid Build Coastguard Worker// limitations under the License.
14*333d2b36SAndroid Build Coastguard Worker
15*333d2b36SAndroid Build Coastguard Workerpackage apex
16*333d2b36SAndroid Build Coastguard Worker
17*333d2b36SAndroid Build Coastguard Workerimport (
18*333d2b36SAndroid Build Coastguard Worker	"strings"
19*333d2b36SAndroid Build Coastguard Worker
20*333d2b36SAndroid Build Coastguard Worker	"android/soong/android"
21*333d2b36SAndroid Build Coastguard Worker	"android/soong/cc"
22*333d2b36SAndroid Build Coastguard Worker
23*333d2b36SAndroid Build Coastguard Worker	"github.com/google/blueprint"
24*333d2b36SAndroid Build Coastguard Worker	"github.com/google/blueprint/proptools"
25*333d2b36SAndroid Build Coastguard Worker)
26*333d2b36SAndroid Build Coastguard Worker
27*333d2b36SAndroid Build Coastguard Workerconst (
28*333d2b36SAndroid Build Coastguard Worker	vndkApexName       = "com.android.vndk"
29*333d2b36SAndroid Build Coastguard Worker	vndkApexNamePrefix = vndkApexName + ".v"
30*333d2b36SAndroid Build Coastguard Worker)
31*333d2b36SAndroid Build Coastguard Worker
32*333d2b36SAndroid Build Coastguard Worker// apex_vndk creates a special variant of apex modules which contains only VNDK libraries.
33*333d2b36SAndroid Build Coastguard Worker// If `vndk_version` is specified, the VNDK libraries of the specified VNDK version are gathered automatically.
34*333d2b36SAndroid Build Coastguard Worker// If not specified, then the "current" versions are gathered.
35*333d2b36SAndroid Build Coastguard Workerfunc vndkApexBundleFactory() android.Module {
36*333d2b36SAndroid Build Coastguard Worker	bundle := newApexBundle()
37*333d2b36SAndroid Build Coastguard Worker	bundle.vndkApex = true
38*333d2b36SAndroid Build Coastguard Worker	bundle.AddProperties(&bundle.vndkProperties)
39*333d2b36SAndroid Build Coastguard Worker	android.AddLoadHook(bundle, func(ctx android.LoadHookContext) {
40*333d2b36SAndroid Build Coastguard Worker		ctx.AppendProperties(&struct {
41*333d2b36SAndroid Build Coastguard Worker			Compile_multilib *string
42*333d2b36SAndroid Build Coastguard Worker		}{
43*333d2b36SAndroid Build Coastguard Worker			proptools.StringPtr("both"),
44*333d2b36SAndroid Build Coastguard Worker		})
45*333d2b36SAndroid Build Coastguard Worker	})
46*333d2b36SAndroid Build Coastguard Worker	return bundle
47*333d2b36SAndroid Build Coastguard Worker}
48*333d2b36SAndroid Build Coastguard Worker
49*333d2b36SAndroid Build Coastguard Workerfunc (a *apexBundle) vndkVersion() string {
50*333d2b36SAndroid Build Coastguard Worker	return proptools.StringDefault(a.vndkProperties.Vndk_version, "current")
51*333d2b36SAndroid Build Coastguard Worker}
52*333d2b36SAndroid Build Coastguard Worker
53*333d2b36SAndroid Build Coastguard Workertype apexVndkProperties struct {
54*333d2b36SAndroid Build Coastguard Worker	// Indicates VNDK version of which this VNDK APEX bundles VNDK libs.
55*333d2b36SAndroid Build Coastguard Worker	Vndk_version *string
56*333d2b36SAndroid Build Coastguard Worker}
57*333d2b36SAndroid Build Coastguard Worker
58*333d2b36SAndroid Build Coastguard Workerfunc apexVndkDepsMutator(mctx android.BottomUpMutatorContext) {
59*333d2b36SAndroid Build Coastguard Worker	if m, ok := mctx.Module().(*cc.Module); ok && cc.IsForVndkApex(mctx, m) {
60*333d2b36SAndroid Build Coastguard Worker		vndkVersion := m.VndkVersion()
61*333d2b36SAndroid Build Coastguard Worker
62*333d2b36SAndroid Build Coastguard Worker		if vndkVersion == "" {
63*333d2b36SAndroid Build Coastguard Worker			return
64*333d2b36SAndroid Build Coastguard Worker		}
65*333d2b36SAndroid Build Coastguard Worker		vndkVersion = "v" + vndkVersion
66*333d2b36SAndroid Build Coastguard Worker
67*333d2b36SAndroid Build Coastguard Worker		vndkApexName := "com.android.vndk." + vndkVersion
68*333d2b36SAndroid Build Coastguard Worker
69*333d2b36SAndroid Build Coastguard Worker		if mctx.OtherModuleExists(vndkApexName) {
70*333d2b36SAndroid Build Coastguard Worker			// Reverse dependencies must exactly specify the variant they want, starting from the
71*333d2b36SAndroid Build Coastguard Worker			// current module's variant. But unlike cc modules, the vndk apex doesn't have
72*333d2b36SAndroid Build Coastguard Worker			// arch/image/link variations, so we explicitly remove them here.
73*333d2b36SAndroid Build Coastguard Worker			mctx.AddReverseVariationDependency([]blueprint.Variation{
74*333d2b36SAndroid Build Coastguard Worker				{Mutator: "arch", Variation: "common"},
75*333d2b36SAndroid Build Coastguard Worker				{Mutator: "image", Variation: ""},
76*333d2b36SAndroid Build Coastguard Worker				{Mutator: "link", Variation: ""},
77*333d2b36SAndroid Build Coastguard Worker			}, sharedLibTag, vndkApexName)
78*333d2b36SAndroid Build Coastguard Worker		}
79*333d2b36SAndroid Build Coastguard Worker	} else if a, ok := mctx.Module().(*apexBundle); ok && a.vndkApex {
80*333d2b36SAndroid Build Coastguard Worker		if a.IsNativeBridgeSupported() {
81*333d2b36SAndroid Build Coastguard Worker			mctx.PropertyErrorf("native_bridge_supported", "%q doesn't support native bridge binary.", mctx.ModuleType())
82*333d2b36SAndroid Build Coastguard Worker		}
83*333d2b36SAndroid Build Coastguard Worker
84*333d2b36SAndroid Build Coastguard Worker		vndkVersion := a.vndkVersion()
85*333d2b36SAndroid Build Coastguard Worker		if vndkVersion != "" {
86*333d2b36SAndroid Build Coastguard Worker			apiLevel, err := android.ApiLevelFromUser(mctx, vndkVersion)
87*333d2b36SAndroid Build Coastguard Worker			if err != nil {
88*333d2b36SAndroid Build Coastguard Worker				mctx.PropertyErrorf("vndk_version", "%s", err.Error())
89*333d2b36SAndroid Build Coastguard Worker				return
90*333d2b36SAndroid Build Coastguard Worker			}
91*333d2b36SAndroid Build Coastguard Worker
92*333d2b36SAndroid Build Coastguard Worker			targets := mctx.MultiTargets()
93*333d2b36SAndroid Build Coastguard Worker			if len(targets) > 0 && apiLevel.LessThan(cc.MinApiForArch(mctx, targets[0].Arch.ArchType)) {
94*333d2b36SAndroid Build Coastguard Worker				// Disable VNDK APEXes for VNDK versions less than the minimum supported API
95*333d2b36SAndroid Build Coastguard Worker				// level for the primary architecture.
96*333d2b36SAndroid Build Coastguard Worker				a.Disable()
97*333d2b36SAndroid Build Coastguard Worker			} else {
98*333d2b36SAndroid Build Coastguard Worker				mctx.AddVariationDependencies(
99*333d2b36SAndroid Build Coastguard Worker					mctx.Config().AndroidFirstDeviceTarget.Variations(),
100*333d2b36SAndroid Build Coastguard Worker					prebuiltTag,
101*333d2b36SAndroid Build Coastguard Worker					cc.VndkLibrariesTxtModules(vndkVersion, mctx)...,
102*333d2b36SAndroid Build Coastguard Worker				)
103*333d2b36SAndroid Build Coastguard Worker			}
104*333d2b36SAndroid Build Coastguard Worker		}
105*333d2b36SAndroid Build Coastguard Worker	}
106*333d2b36SAndroid Build Coastguard Worker}
107*333d2b36SAndroid Build Coastguard Worker
108*333d2b36SAndroid Build Coastguard Worker// name is module.BaseModuleName() which is used as LOCAL_MODULE_NAME and also LOCAL_OVERRIDES_*
109*333d2b36SAndroid Build Coastguard Workerfunc makeCompatSymlinks(name string, ctx android.ModuleContext) (symlinks android.InstallPaths) {
110*333d2b36SAndroid Build Coastguard Worker	// small helper to add symlink commands
111*333d2b36SAndroid Build Coastguard Worker	addSymlink := func(target string, dir android.InstallPath, linkName string) {
112*333d2b36SAndroid Build Coastguard Worker		symlinks = append(symlinks, ctx.InstallAbsoluteSymlink(dir, linkName, target))
113*333d2b36SAndroid Build Coastguard Worker	}
114*333d2b36SAndroid Build Coastguard Worker
115*333d2b36SAndroid Build Coastguard Worker	// TODO(b/142911355): [VNDK APEX] Fix hard-coded references to /system/lib/vndk
116*333d2b36SAndroid Build Coastguard Worker	// When all hard-coded references are fixed, remove symbolic links
117*333d2b36SAndroid Build Coastguard Worker	// Note that  we should keep following symlinks for older VNDKs (<=29)
118*333d2b36SAndroid Build Coastguard Worker	// Since prebuilt vndk libs still depend on system/lib/vndk path
119*333d2b36SAndroid Build Coastguard Worker	if strings.HasPrefix(name, vndkApexNamePrefix) {
120*333d2b36SAndroid Build Coastguard Worker		vndkVersion := strings.TrimPrefix(name, vndkApexNamePrefix)
121*333d2b36SAndroid Build Coastguard Worker		if ver, err := android.ApiLevelFromUser(ctx, vndkVersion); err != nil {
122*333d2b36SAndroid Build Coastguard Worker			ctx.ModuleErrorf("apex_vndk should be named as %v<ver:number>: %s", vndkApexNamePrefix, name)
123*333d2b36SAndroid Build Coastguard Worker			return
124*333d2b36SAndroid Build Coastguard Worker		} else if ver.GreaterThan(android.SdkVersion_Android10) {
125*333d2b36SAndroid Build Coastguard Worker			return
126*333d2b36SAndroid Build Coastguard Worker		}
127*333d2b36SAndroid Build Coastguard Worker		// the name of vndk apex is formatted "com.android.vndk.v" + version
128*333d2b36SAndroid Build Coastguard Worker		apexName := vndkApexNamePrefix + vndkVersion
129*333d2b36SAndroid Build Coastguard Worker		if ctx.Config().Android64() {
130*333d2b36SAndroid Build Coastguard Worker			dir := android.PathForModuleInPartitionInstall(ctx, "system", "lib64")
131*333d2b36SAndroid Build Coastguard Worker			addSymlink("/apex/"+apexName+"/lib64", dir, "vndk-sp-"+vndkVersion)
132*333d2b36SAndroid Build Coastguard Worker			addSymlink("/apex/"+apexName+"/lib64", dir, "vndk-"+vndkVersion)
133*333d2b36SAndroid Build Coastguard Worker		}
134*333d2b36SAndroid Build Coastguard Worker		if !ctx.Config().Android64() || ctx.DeviceConfig().DeviceSecondaryArch() != "" {
135*333d2b36SAndroid Build Coastguard Worker			dir := android.PathForModuleInPartitionInstall(ctx, "system", "lib")
136*333d2b36SAndroid Build Coastguard Worker			addSymlink("/apex/"+apexName+"/lib", dir, "vndk-sp-"+vndkVersion)
137*333d2b36SAndroid Build Coastguard Worker			addSymlink("/apex/"+apexName+"/lib", dir, "vndk-"+vndkVersion)
138*333d2b36SAndroid Build Coastguard Worker		}
139*333d2b36SAndroid Build Coastguard Worker	}
140*333d2b36SAndroid Build Coastguard Worker
141*333d2b36SAndroid Build Coastguard Worker	// http://b/121248172 - create a link from /system/usr/icu to
142*333d2b36SAndroid Build Coastguard Worker	// /apex/com.android.i18n/etc/icu so that apps can find the ICU .dat file.
143*333d2b36SAndroid Build Coastguard Worker	// A symlink can't overwrite a directory and the /system/usr/icu directory once
144*333d2b36SAndroid Build Coastguard Worker	// existed so the required structure must be created whatever we find.
145*333d2b36SAndroid Build Coastguard Worker	if name == "com.android.i18n" {
146*333d2b36SAndroid Build Coastguard Worker		dir := android.PathForModuleInPartitionInstall(ctx, "system", "usr")
147*333d2b36SAndroid Build Coastguard Worker		addSymlink("/apex/com.android.i18n/etc/icu", dir, "icu")
148*333d2b36SAndroid Build Coastguard Worker	}
149*333d2b36SAndroid Build Coastguard Worker
150*333d2b36SAndroid Build Coastguard Worker	return symlinks
151*333d2b36SAndroid Build Coastguard Worker}
152