| // Copyright (C) 2019 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. | 
 |  | 
 | package apex | 
 |  | 
 | import ( | 
 | 	"strings" | 
 |  | 
 | 	"android/soong/android" | 
 | 	"android/soong/cc" | 
 |  | 
 | 	"github.com/google/blueprint/proptools" | 
 | ) | 
 |  | 
 | const ( | 
 | 	vndkApexName       = "com.android.vndk" | 
 | 	vndkApexNamePrefix = vndkApexName + ".v" | 
 | ) | 
 |  | 
 | // apex_vndk creates a special variant of apex modules which contains only VNDK libraries. | 
 | // If `vndk_version` is specified, the VNDK libraries of the specified VNDK version are gathered automatically. | 
 | // If not specified, then the "current" versions are gathered. | 
 | func vndkApexBundleFactory() android.Module { | 
 | 	bundle := newApexBundle() | 
 | 	bundle.vndkApex = true | 
 | 	bundle.AddProperties(&bundle.vndkProperties) | 
 | 	android.AddLoadHook(bundle, func(ctx android.LoadHookContext) { | 
 | 		ctx.AppendProperties(&struct { | 
 | 			Compile_multilib *string | 
 | 		}{ | 
 | 			proptools.StringPtr("both"), | 
 | 		}) | 
 | 	}) | 
 | 	return bundle | 
 | } | 
 |  | 
 | func (a *apexBundle) vndkVersion(config android.DeviceConfig) string { | 
 | 	vndkVersion := proptools.StringDefault(a.vndkProperties.Vndk_version, "current") | 
 | 	if vndkVersion == "current" { | 
 | 		vndkVersion = config.PlatformVndkVersion() | 
 | 	} | 
 | 	return vndkVersion | 
 | } | 
 |  | 
 | type apexVndkProperties struct { | 
 | 	// Indicates VNDK version of which this VNDK APEX bundles VNDK libs. Default is Platform VNDK Version. | 
 | 	Vndk_version *string | 
 | } | 
 |  | 
 | func apexVndkMutator(mctx android.TopDownMutatorContext) { | 
 | 	if ab, ok := mctx.Module().(*apexBundle); ok && ab.vndkApex { | 
 | 		if ab.IsNativeBridgeSupported() { | 
 | 			mctx.PropertyErrorf("native_bridge_supported", "%q doesn't support native bridge binary.", mctx.ModuleType()) | 
 | 		} | 
 |  | 
 | 		vndkVersion := ab.vndkVersion(mctx.DeviceConfig()) | 
 | 		apiLevel, err := android.ApiLevelFromUser(mctx, vndkVersion) | 
 | 		if err != nil { | 
 | 			mctx.PropertyErrorf("vndk_version", "%s", err.Error()) | 
 | 			return | 
 | 		} | 
 |  | 
 | 		targets := mctx.MultiTargets() | 
 | 		if len(targets) > 0 && apiLevel.LessThan(cc.MinApiForArch(mctx, targets[0].Arch.ArchType)) && | 
 | 			vndkVersion != mctx.DeviceConfig().PlatformVndkVersion() { | 
 | 			// Disable VNDK APEXes for VNDK versions less than the minimum supported API | 
 | 			// level for the primary architecture. This validation is skipped if the VNDK | 
 | 			// version matches the platform VNDK version, which can occur when the device | 
 | 			// config targets the 'current' VNDK (see `vndkVersion`). | 
 | 			ab.Disable() | 
 | 		} | 
 | 		if proptools.String(ab.vndkProperties.Vndk_version) != "" && | 
 | 			apiLevel.GreaterThanOrEqualTo(android.ApiLevelOrPanic(mctx, mctx.DeviceConfig().PlatformVndkVersion())) { | 
 | 			ab.Disable() | 
 | 		} | 
 | 	} | 
 | } | 
 |  | 
 | func apexVndkDepsMutator(mctx android.BottomUpMutatorContext) { | 
 | 	if m, ok := mctx.Module().(*cc.Module); ok && cc.IsForVndkApex(mctx, m) { | 
 | 		vndkVersion := m.VndkVersion() | 
 | 		// For VNDK-Lite device, we gather core-variants of VNDK-Sp libraries, which doesn't have VNDK version defined | 
 | 		if vndkVersion == "" { | 
 | 			vndkVersion = mctx.DeviceConfig().PlatformVndkVersion() | 
 | 		} | 
 | 		if vndkVersion == mctx.DeviceConfig().PlatformVndkVersion() { | 
 | 			vndkVersion = "current" | 
 | 		} else { | 
 | 			vndkVersion = "v" + vndkVersion | 
 | 		} | 
 |  | 
 | 		vndkApexName := "com.android.vndk." + vndkVersion | 
 |  | 
 | 		if mctx.OtherModuleExists(vndkApexName) { | 
 | 			mctx.AddReverseDependency(mctx.Module(), sharedLibTag, vndkApexName) | 
 | 		} | 
 | 	} else if a, ok := mctx.Module().(*apexBundle); ok && a.vndkApex { | 
 | 		vndkVersion := proptools.StringDefault(a.vndkProperties.Vndk_version, "current") | 
 | 		mctx.AddDependency(mctx.Module(), prebuiltTag, cc.VndkLibrariesTxtModules(vndkVersion, mctx)...) | 
 | 	} | 
 | } | 
 |  | 
 | // name is module.BaseModuleName() which is used as LOCAL_MODULE_NAME and also LOCAL_OVERRIDES_* | 
 | func makeCompatSymlinks(name string, ctx android.ModuleContext) (symlinks android.InstallPaths) { | 
 | 	// small helper to add symlink commands | 
 | 	addSymlink := func(target string, dir android.InstallPath, linkName string) { | 
 | 		symlinks = append(symlinks, ctx.InstallAbsoluteSymlink(dir, linkName, target)) | 
 | 	} | 
 |  | 
 | 	// TODO(b/142911355): [VNDK APEX] Fix hard-coded references to /system/lib/vndk | 
 | 	// When all hard-coded references are fixed, remove symbolic links | 
 | 	// Note that  we should keep following symlinks for older VNDKs (<=29) | 
 | 	// Since prebuilt vndk libs still depend on system/lib/vndk path | 
 | 	if strings.HasPrefix(name, vndkApexNamePrefix) { | 
 | 		vndkVersion := strings.TrimPrefix(name, vndkApexNamePrefix) | 
 | 		if ver, err := android.ApiLevelFromUser(ctx, vndkVersion); err != nil { | 
 | 			ctx.ModuleErrorf("apex_vndk should be named as %v<ver:number>: %s", vndkApexNamePrefix, name) | 
 | 			return | 
 | 		} else if ver.GreaterThan(android.SdkVersion_Android10) { | 
 | 			return | 
 | 		} | 
 | 		// the name of vndk apex is formatted "com.android.vndk.v" + version | 
 | 		apexName := vndkApexNamePrefix + vndkVersion | 
 | 		if ctx.Config().Android64() { | 
 | 			dir := android.PathForModuleInPartitionInstall(ctx, "system", "lib64") | 
 | 			addSymlink("/apex/"+apexName+"/lib64", dir, "vndk-sp-"+vndkVersion) | 
 | 			addSymlink("/apex/"+apexName+"/lib64", dir, "vndk-"+vndkVersion) | 
 | 		} | 
 | 		if !ctx.Config().Android64() || ctx.DeviceConfig().DeviceSecondaryArch() != "" { | 
 | 			dir := android.PathForModuleInPartitionInstall(ctx, "system", "lib") | 
 | 			addSymlink("/apex/"+apexName+"/lib", dir, "vndk-sp-"+vndkVersion) | 
 | 			addSymlink("/apex/"+apexName+"/lib", dir, "vndk-"+vndkVersion) | 
 | 		} | 
 | 	} | 
 |  | 
 | 	// http://b/121248172 - create a link from /system/usr/icu to | 
 | 	// /apex/com.android.i18n/etc/icu so that apps can find the ICU .dat file. | 
 | 	// A symlink can't overwrite a directory and the /system/usr/icu directory once | 
 | 	// existed so the required structure must be created whatever we find. | 
 | 	if name == "com.android.i18n" { | 
 | 		dir := android.PathForModuleInPartitionInstall(ctx, "system", "usr") | 
 | 		addSymlink("/apex/com.android.i18n/etc/icu", dir, "icu") | 
 | 	} | 
 |  | 
 | 	return symlinks | 
 | } |