| // Copyright 2017 Google Inc. All rights reserved. |
| // |
| // 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 cc |
| |
| import ( |
| "encoding/json" |
| "errors" |
| "fmt" |
| "path/filepath" |
| "sort" |
| "strings" |
| "sync" |
| |
| "android/soong/android" |
| "android/soong/cc/config" |
| "android/soong/etc" |
| |
| "github.com/google/blueprint" |
| ) |
| |
| const ( |
| llndkLibrariesTxt = "llndk.libraries.txt" |
| vndkCoreLibrariesTxt = "vndkcore.libraries.txt" |
| vndkSpLibrariesTxt = "vndksp.libraries.txt" |
| vndkPrivateLibrariesTxt = "vndkprivate.libraries.txt" |
| vndkUsingCoreVariantLibrariesTxt = "vndkcorevariant.libraries.txt" |
| ) |
| |
| func VndkLibrariesTxtModules(vndkVersion string) []string { |
| if vndkVersion == "current" { |
| return []string{ |
| llndkLibrariesTxt, |
| vndkCoreLibrariesTxt, |
| vndkSpLibrariesTxt, |
| vndkPrivateLibrariesTxt, |
| } |
| } |
| // Snapshot vndks have their own *.libraries.VER.txt files. |
| // Note that snapshots don't have "vndkcorevariant.libraries.VER.txt" |
| return []string{ |
| insertVndkVersion(llndkLibrariesTxt, vndkVersion), |
| insertVndkVersion(vndkCoreLibrariesTxt, vndkVersion), |
| insertVndkVersion(vndkSpLibrariesTxt, vndkVersion), |
| insertVndkVersion(vndkPrivateLibrariesTxt, vndkVersion), |
| } |
| } |
| |
| type VndkProperties struct { |
| Vndk struct { |
| // declared as a VNDK or VNDK-SP module. The vendor variant |
| // will be installed in /system instead of /vendor partition. |
| // |
| // `vendor_available` must be explicitly set to either true or |
| // false together with `vndk: {enabled: true}`. |
| Enabled *bool |
| |
| // declared as a VNDK-SP module, which is a subset of VNDK. |
| // |
| // `vndk: { enabled: true }` must set together. |
| // |
| // All these modules are allowed to link to VNDK-SP or LL-NDK |
| // modules only. Other dependency will cause link-type errors. |
| // |
| // If `support_system_process` is not set or set to false, |
| // the module is VNDK-core and can link to other VNDK-core, |
| // VNDK-SP or LL-NDK modules only. |
| Support_system_process *bool |
| |
| // Extending another module |
| Extends *string |
| } |
| } |
| |
| type vndkdep struct { |
| Properties VndkProperties |
| } |
| |
| func (vndk *vndkdep) props() []interface{} { |
| return []interface{}{&vndk.Properties} |
| } |
| |
| func (vndk *vndkdep) begin(ctx BaseModuleContext) {} |
| |
| func (vndk *vndkdep) deps(ctx BaseModuleContext, deps Deps) Deps { |
| return deps |
| } |
| |
| func (vndk *vndkdep) isVndk() bool { |
| return Bool(vndk.Properties.Vndk.Enabled) |
| } |
| |
| func (vndk *vndkdep) isVndkSp() bool { |
| return Bool(vndk.Properties.Vndk.Support_system_process) |
| } |
| |
| func (vndk *vndkdep) isVndkExt() bool { |
| return vndk.Properties.Vndk.Extends != nil |
| } |
| |
| func (vndk *vndkdep) getVndkExtendsModuleName() string { |
| return String(vndk.Properties.Vndk.Extends) |
| } |
| |
| func (vndk *vndkdep) typeName() string { |
| if !vndk.isVndk() { |
| return "native:vendor" |
| } |
| if !vndk.isVndkExt() { |
| if !vndk.isVndkSp() { |
| return "native:vendor:vndk" |
| } |
| return "native:vendor:vndksp" |
| } |
| if !vndk.isVndkSp() { |
| return "native:vendor:vndkext" |
| } |
| return "native:vendor:vndkspext" |
| } |
| |
| func (vndk *vndkdep) vndkCheckLinkType(ctx android.ModuleContext, to *Module, tag blueprint.DependencyTag) { |
| if to.linker == nil { |
| return |
| } |
| if !vndk.isVndk() { |
| // Non-VNDK modules (those installed to /vendor, /product, or /system/product) can't depend |
| // on modules marked with vendor_available: false. |
| violation := false |
| if lib, ok := to.linker.(*llndkStubDecorator); ok && !Bool(lib.Properties.Vendor_available) { |
| violation = true |
| } else { |
| if _, ok := to.linker.(libraryInterface); ok && to.VendorProperties.Vendor_available != nil && !Bool(to.VendorProperties.Vendor_available) { |
| // Vendor_available == nil && !Bool(Vendor_available) should be okay since |
| // it means a vendor-only, or product-only library which is a valid dependency |
| // for non-VNDK modules. |
| violation = true |
| } |
| } |
| if violation { |
| ctx.ModuleErrorf("Vendor module that is not VNDK should not link to %q which is marked as `vendor_available: false`", to.Name()) |
| } |
| } |
| if lib, ok := to.linker.(*libraryDecorator); !ok || !lib.shared() { |
| // Check only shared libraries. |
| // Other (static and LL-NDK) libraries are allowed to link. |
| return |
| } |
| if !to.UseVndk() { |
| ctx.ModuleErrorf("(%s) should not link to %q which is not a vendor-available library", |
| vndk.typeName(), to.Name()) |
| return |
| } |
| if tag == vndkExtDepTag { |
| // Ensure `extends: "name"` property refers a vndk module that has vendor_available |
| // and has identical vndk properties. |
| if to.vndkdep == nil || !to.vndkdep.isVndk() { |
| ctx.ModuleErrorf("`extends` refers a non-vndk module %q", to.Name()) |
| return |
| } |
| if vndk.isVndkSp() != to.vndkdep.isVndkSp() { |
| ctx.ModuleErrorf( |
| "`extends` refers a module %q with mismatched support_system_process", |
| to.Name()) |
| return |
| } |
| if !Bool(to.VendorProperties.Vendor_available) { |
| ctx.ModuleErrorf( |
| "`extends` refers module %q which does not have `vendor_available: true`", |
| to.Name()) |
| return |
| } |
| } |
| if to.vndkdep == nil { |
| return |
| } |
| |
| // Check the dependencies of VNDK shared libraries. |
| if err := vndkIsVndkDepAllowed(vndk, to.vndkdep); err != nil { |
| ctx.ModuleErrorf("(%s) should not link to %q (%s): %v", |
| vndk.typeName(), to.Name(), to.vndkdep.typeName(), err) |
| return |
| } |
| } |
| |
| func vndkIsVndkDepAllowed(from *vndkdep, to *vndkdep) error { |
| // Check the dependencies of VNDK, VNDK-Ext, VNDK-SP, VNDK-SP-Ext and vendor modules. |
| if from.isVndkExt() { |
| if from.isVndkSp() { |
| if to.isVndk() && !to.isVndkSp() { |
| return errors.New("VNDK-SP extensions must not depend on VNDK or VNDK extensions") |
| } |
| return nil |
| } |
| // VNDK-Ext may depend on VNDK, VNDK-Ext, VNDK-SP, VNDK-SP-Ext, or vendor libs. |
| return nil |
| } |
| if from.isVndk() { |
| if to.isVndkExt() { |
| return errors.New("VNDK-core and VNDK-SP must not depend on VNDK extensions") |
| } |
| if from.isVndkSp() { |
| if !to.isVndkSp() { |
| return errors.New("VNDK-SP must only depend on VNDK-SP") |
| } |
| return nil |
| } |
| if !to.isVndk() { |
| return errors.New("VNDK-core must only depend on VNDK-core or VNDK-SP") |
| } |
| return nil |
| } |
| // Vendor modules may depend on VNDK, VNDK-Ext, VNDK-SP, VNDK-SP-Ext, or vendor libs. |
| return nil |
| } |
| |
| var ( |
| vndkCoreLibrariesKey = android.NewOnceKey("vndkCoreLibrarires") |
| vndkSpLibrariesKey = android.NewOnceKey("vndkSpLibrarires") |
| llndkLibrariesKey = android.NewOnceKey("llndkLibrarires") |
| vndkPrivateLibrariesKey = android.NewOnceKey("vndkPrivateLibrarires") |
| vndkUsingCoreVariantLibrariesKey = android.NewOnceKey("vndkUsingCoreVariantLibraries") |
| vndkMustUseVendorVariantListKey = android.NewOnceKey("vndkMustUseVendorVariantListKey") |
| vndkLibrariesLock sync.Mutex |
| ) |
| |
| func vndkCoreLibraries(config android.Config) map[string]string { |
| return config.Once(vndkCoreLibrariesKey, func() interface{} { |
| return make(map[string]string) |
| }).(map[string]string) |
| } |
| |
| func vndkSpLibraries(config android.Config) map[string]string { |
| return config.Once(vndkSpLibrariesKey, func() interface{} { |
| return make(map[string]string) |
| }).(map[string]string) |
| } |
| |
| func isLlndkLibrary(baseModuleName string, config android.Config) bool { |
| _, ok := llndkLibraries(config)[baseModuleName] |
| return ok |
| } |
| |
| func llndkLibraries(config android.Config) map[string]string { |
| return config.Once(llndkLibrariesKey, func() interface{} { |
| return make(map[string]string) |
| }).(map[string]string) |
| } |
| |
| func isVndkPrivateLibrary(baseModuleName string, config android.Config) bool { |
| _, ok := vndkPrivateLibraries(config)[baseModuleName] |
| return ok |
| } |
| |
| func vndkPrivateLibraries(config android.Config) map[string]string { |
| return config.Once(vndkPrivateLibrariesKey, func() interface{} { |
| return make(map[string]string) |
| }).(map[string]string) |
| } |
| |
| func vndkUsingCoreVariantLibraries(config android.Config) map[string]string { |
| return config.Once(vndkUsingCoreVariantLibrariesKey, func() interface{} { |
| return make(map[string]string) |
| }).(map[string]string) |
| } |
| |
| func vndkMustUseVendorVariantList(cfg android.Config) []string { |
| return cfg.Once(vndkMustUseVendorVariantListKey, func() interface{} { |
| return config.VndkMustUseVendorVariantList |
| }).([]string) |
| } |
| |
| // test may call this to override global configuration(config.VndkMustUseVendorVariantList) |
| // when it is called, it must be before the first call to vndkMustUseVendorVariantList() |
| func setVndkMustUseVendorVariantListForTest(config android.Config, mustUseVendorVariantList []string) { |
| config.Once(vndkMustUseVendorVariantListKey, func() interface{} { |
| return mustUseVendorVariantList |
| }) |
| } |
| |
| func processLlndkLibrary(mctx android.BottomUpMutatorContext, m *Module) { |
| lib := m.linker.(*llndkStubDecorator) |
| name := m.BaseModuleName() |
| filename := m.BaseModuleName() + ".so" |
| |
| vndkLibrariesLock.Lock() |
| defer vndkLibrariesLock.Unlock() |
| |
| llndkLibraries(mctx.Config())[name] = filename |
| if !Bool(lib.Properties.Vendor_available) { |
| vndkPrivateLibraries(mctx.Config())[name] = filename |
| } |
| |
| if mctx.OtherModuleExists(name) { |
| mctx.AddFarVariationDependencies(m.Target().Variations(), llndkImplDep, name) |
| } |
| } |
| |
| func processVndkLibrary(mctx android.BottomUpMutatorContext, m *Module) { |
| name := m.BaseModuleName() |
| filename, err := getVndkFileName(m) |
| if err != nil { |
| panic(err) |
| } |
| |
| if m.HasStubsVariants() && name != "libz" { |
| // b/155456180 libz is the ONLY exception here. We don't want to make |
| // libz an LLNDK library because we in general can't guarantee that |
| // libz will behave consistently especially about the compression. |
| // i.e. the compressed output might be different across releases. |
| // As the library is an external one, it's risky to keep the compatibility |
| // promise if it becomes an LLNDK. |
| mctx.PropertyErrorf("vndk.enabled", "This library provides stubs. Shouldn't be VNDK. Consider making it as LLNDK") |
| } |
| |
| vndkLibrariesLock.Lock() |
| defer vndkLibrariesLock.Unlock() |
| |
| if inList(name, vndkMustUseVendorVariantList(mctx.Config())) { |
| m.Properties.MustUseVendorVariant = true |
| } |
| if mctx.DeviceConfig().VndkUseCoreVariant() && !m.Properties.MustUseVendorVariant { |
| vndkUsingCoreVariantLibraries(mctx.Config())[name] = filename |
| } |
| |
| if m.vndkdep.isVndkSp() { |
| vndkSpLibraries(mctx.Config())[name] = filename |
| } else { |
| vndkCoreLibraries(mctx.Config())[name] = filename |
| } |
| if !Bool(m.VendorProperties.Vendor_available) { |
| vndkPrivateLibraries(mctx.Config())[name] = filename |
| } |
| } |
| |
| // Check for modules that mustn't be VNDK |
| func shouldSkipVndkMutator(m *Module) bool { |
| if !m.Enabled() { |
| return true |
| } |
| if !m.Device() { |
| // Skip non-device modules |
| return true |
| } |
| if m.Target().NativeBridge == android.NativeBridgeEnabled { |
| // Skip native_bridge modules |
| return true |
| } |
| return false |
| } |
| |
| func IsForVndkApex(mctx android.BottomUpMutatorContext, m *Module) bool { |
| if shouldSkipVndkMutator(m) { |
| return false |
| } |
| |
| // prebuilt vndk modules should match with device |
| // TODO(b/142675459): Use enabled: to select target device in vndk_prebuilt_shared |
| // When b/142675459 is landed, remove following check |
| if p, ok := m.linker.(*vndkPrebuiltLibraryDecorator); ok && !p.matchesWithDevice(mctx.DeviceConfig()) { |
| return false |
| } |
| |
| if lib, ok := m.linker.(libraryInterface); ok { |
| // VNDK APEX for VNDK-Lite devices will have VNDK-SP libraries from core variants |
| if mctx.DeviceConfig().VndkVersion() == "" { |
| // b/73296261: filter out libz.so because it is considered as LLNDK for VNDK-lite devices |
| if mctx.ModuleName() == "libz" { |
| return false |
| } |
| return m.ImageVariation().Variation == android.CoreVariation && lib.shared() && m.isVndkSp() |
| } |
| |
| useCoreVariant := m.VndkVersion() == mctx.DeviceConfig().PlatformVndkVersion() && |
| mctx.DeviceConfig().VndkUseCoreVariant() && !m.MustUseVendorVariant() |
| return lib.shared() && m.inVendor() && m.IsVndk() && !m.isVndkExt() && !useCoreVariant |
| } |
| return false |
| } |
| |
| // gather list of vndk-core, vndk-sp, and ll-ndk libs |
| func VndkMutator(mctx android.BottomUpMutatorContext) { |
| m, ok := mctx.Module().(*Module) |
| if !ok { |
| return |
| } |
| |
| if shouldSkipVndkMutator(m) { |
| return |
| } |
| |
| if _, ok := m.linker.(*llndkStubDecorator); ok { |
| processLlndkLibrary(mctx, m) |
| return |
| } |
| |
| lib, is_lib := m.linker.(*libraryDecorator) |
| prebuilt_lib, is_prebuilt_lib := m.linker.(*prebuiltLibraryLinker) |
| |
| if (is_lib && lib.buildShared()) || (is_prebuilt_lib && prebuilt_lib.buildShared()) { |
| if m.vndkdep != nil && m.vndkdep.isVndk() && !m.vndkdep.isVndkExt() { |
| processVndkLibrary(mctx, m) |
| return |
| } |
| } |
| } |
| |
| func init() { |
| android.RegisterModuleType("vndk_libraries_txt", VndkLibrariesTxtFactory) |
| android.RegisterSingletonType("vndk-snapshot", VndkSnapshotSingleton) |
| } |
| |
| type vndkLibrariesTxt struct { |
| android.ModuleBase |
| outputFile android.OutputPath |
| } |
| |
| var _ etc.PrebuiltEtcModule = &vndkLibrariesTxt{} |
| var _ android.OutputFileProducer = &vndkLibrariesTxt{} |
| |
| // vndk_libraries_txt is a special kind of module type in that it name is one of |
| // - llndk.libraries.txt |
| // - vndkcore.libraries.txt |
| // - vndksp.libraries.txt |
| // - vndkprivate.libraries.txt |
| // - vndkcorevariant.libraries.txt |
| // A module behaves like a prebuilt_etc but its content is generated by soong. |
| // By being a soong module, these files can be referenced by other soong modules. |
| // For example, apex_vndk can depend on these files as prebuilt. |
| func VndkLibrariesTxtFactory() android.Module { |
| m := &vndkLibrariesTxt{} |
| android.InitAndroidModule(m) |
| return m |
| } |
| |
| func insertVndkVersion(filename string, vndkVersion string) string { |
| if index := strings.LastIndex(filename, "."); index != -1 { |
| return filename[:index] + "." + vndkVersion + filename[index:] |
| } |
| return filename |
| } |
| |
| func (txt *vndkLibrariesTxt) GenerateAndroidBuildActions(ctx android.ModuleContext) { |
| var list []string |
| switch txt.Name() { |
| case llndkLibrariesTxt: |
| for _, filename := range android.SortedStringMapValues(llndkLibraries(ctx.Config())) { |
| if strings.HasPrefix(filename, "libclang_rt.hwasan-") { |
| continue |
| } |
| list = append(list, filename) |
| } |
| case vndkCoreLibrariesTxt: |
| list = android.SortedStringMapValues(vndkCoreLibraries(ctx.Config())) |
| case vndkSpLibrariesTxt: |
| list = android.SortedStringMapValues(vndkSpLibraries(ctx.Config())) |
| case vndkPrivateLibrariesTxt: |
| list = android.SortedStringMapValues(vndkPrivateLibraries(ctx.Config())) |
| case vndkUsingCoreVariantLibrariesTxt: |
| list = android.SortedStringMapValues(vndkUsingCoreVariantLibraries(ctx.Config())) |
| default: |
| ctx.ModuleErrorf("name(%s) is unknown.", txt.Name()) |
| return |
| } |
| |
| var filename string |
| if txt.Name() != vndkUsingCoreVariantLibrariesTxt { |
| filename = insertVndkVersion(txt.Name(), ctx.DeviceConfig().PlatformVndkVersion()) |
| } else { |
| filename = txt.Name() |
| } |
| |
| txt.outputFile = android.PathForModuleOut(ctx, filename).OutputPath |
| ctx.Build(pctx, android.BuildParams{ |
| Rule: android.WriteFile, |
| Output: txt.outputFile, |
| Description: "Writing " + txt.outputFile.String(), |
| Args: map[string]string{ |
| "content": strings.Join(list, "\\n"), |
| }, |
| }) |
| |
| installPath := android.PathForModuleInstall(ctx, "etc") |
| ctx.InstallFile(installPath, filename, txt.outputFile) |
| } |
| |
| func (txt *vndkLibrariesTxt) AndroidMkEntries() []android.AndroidMkEntries { |
| return []android.AndroidMkEntries{android.AndroidMkEntries{ |
| Class: "ETC", |
| OutputFile: android.OptionalPathForPath(txt.outputFile), |
| ExtraEntries: []android.AndroidMkExtraEntriesFunc{ |
| func(entries *android.AndroidMkEntries) { |
| entries.SetString("LOCAL_MODULE_STEM", txt.outputFile.Base()) |
| }, |
| }, |
| }} |
| } |
| |
| // PrebuiltEtcModule interface |
| func (txt *vndkLibrariesTxt) OutputFile() android.OutputPath { |
| return txt.outputFile |
| } |
| |
| // PrebuiltEtcModule interface |
| func (txt *vndkLibrariesTxt) BaseDir() string { |
| return "etc" |
| } |
| |
| // PrebuiltEtcModule interface |
| func (txt *vndkLibrariesTxt) SubDir() string { |
| return "" |
| } |
| |
| func (txt *vndkLibrariesTxt) OutputFiles(tag string) (android.Paths, error) { |
| return android.Paths{txt.outputFile}, nil |
| } |
| |
| func VndkSnapshotSingleton() android.Singleton { |
| return &vndkSnapshotSingleton{} |
| } |
| |
| type vndkSnapshotSingleton struct { |
| vndkLibrariesFile android.OutputPath |
| vndkSnapshotZipFile android.OptionalPath |
| } |
| |
| func isVndkSnapshotLibrary(config android.DeviceConfig, m *Module, |
| apexInfo android.ApexInfo) (i snapshotLibraryInterface, vndkType string, isVndkSnapshotLib bool) { |
| |
| if m.Target().NativeBridge == android.NativeBridgeEnabled { |
| return nil, "", false |
| } |
| // !inVendor: There's product/vendor variants for VNDK libs. We only care about vendor variants. |
| // !installable: Snapshot only cares about "installable" modules. |
| // isSnapshotPrebuilt: Snapshotting a snapshot doesn't make sense. |
| if !m.inVendor() || !m.installable(apexInfo) || m.isSnapshotPrebuilt() { |
| return nil, "", false |
| } |
| l, ok := m.linker.(snapshotLibraryInterface) |
| if !ok || !l.shared() { |
| return nil, "", false |
| } |
| if m.VndkVersion() == config.PlatformVndkVersion() && m.IsVndk() && !m.isVndkExt() { |
| if m.isVndkSp() { |
| return l, "vndk-sp", true |
| } else { |
| return l, "vndk-core", true |
| } |
| } |
| |
| return nil, "", false |
| } |
| |
| func (c *vndkSnapshotSingleton) GenerateBuildActions(ctx android.SingletonContext) { |
| // build these files even if PlatformVndkVersion or BoardVndkVersion is not set |
| c.buildVndkLibrariesTxtFiles(ctx) |
| |
| // BOARD_VNDK_VERSION must be set to 'current' in order to generate a VNDK snapshot. |
| if ctx.DeviceConfig().VndkVersion() != "current" { |
| return |
| } |
| |
| if ctx.DeviceConfig().PlatformVndkVersion() == "" { |
| return |
| } |
| |
| var snapshotOutputs android.Paths |
| |
| /* |
| VNDK snapshot zipped artifacts directory structure: |
| {SNAPSHOT_ARCH}/ |
| arch-{TARGET_ARCH}-{TARGET_ARCH_VARIANT}/ |
| shared/ |
| vndk-core/ |
| (VNDK-core libraries, e.g. libbinder.so) |
| vndk-sp/ |
| (VNDK-SP libraries, e.g. libc++.so) |
| arch-{TARGET_2ND_ARCH}-{TARGET_2ND_ARCH_VARIANT}/ |
| shared/ |
| vndk-core/ |
| (VNDK-core libraries, e.g. libbinder.so) |
| vndk-sp/ |
| (VNDK-SP libraries, e.g. libc++.so) |
| binder32/ |
| (This directory is newly introduced in v28 (Android P) to hold |
| prebuilts built for 32-bit binder interface.) |
| arch-{TARGET_ARCH}-{TARGE_ARCH_VARIANT}/ |
| ... |
| configs/ |
| (various *.txt configuration files) |
| include/ |
| (header files of same directory structure with source tree) |
| NOTICE_FILES/ |
| (notice files of libraries, e.g. libcutils.so.txt) |
| */ |
| |
| snapshotDir := "vndk-snapshot" |
| snapshotArchDir := filepath.Join(snapshotDir, ctx.DeviceConfig().DeviceArch()) |
| |
| configsDir := filepath.Join(snapshotArchDir, "configs") |
| noticeDir := filepath.Join(snapshotArchDir, "NOTICE_FILES") |
| includeDir := filepath.Join(snapshotArchDir, "include") |
| |
| // set of notice files copied. |
| noticeBuilt := make(map[string]bool) |
| |
| // paths of VNDK modules for GPL license checking |
| modulePaths := make(map[string]string) |
| |
| // actual module names of .so files |
| // e.g. moduleNames["libprotobuf-cpp-full-3.9.1.so"] = "libprotobuf-cpp-full" |
| moduleNames := make(map[string]string) |
| |
| var headers android.Paths |
| |
| installVndkSnapshotLib := func(m *Module, vndkType string) (android.Paths, bool) { |
| var ret android.Paths |
| |
| targetArch := "arch-" + m.Target().Arch.ArchType.String() |
| if m.Target().Arch.ArchVariant != "" { |
| targetArch += "-" + m.Target().Arch.ArchVariant |
| } |
| |
| libPath := m.outputFile.Path() |
| snapshotLibOut := filepath.Join(snapshotArchDir, targetArch, "shared", vndkType, libPath.Base()) |
| ret = append(ret, copyFile(ctx, libPath, snapshotLibOut)) |
| |
| if ctx.Config().VndkSnapshotBuildArtifacts() { |
| prop := struct { |
| ExportedDirs []string `json:",omitempty"` |
| ExportedSystemDirs []string `json:",omitempty"` |
| ExportedFlags []string `json:",omitempty"` |
| RelativeInstallPath string `json:",omitempty"` |
| }{} |
| exportedInfo := ctx.ModuleProvider(m, FlagExporterInfoProvider).(FlagExporterInfo) |
| prop.ExportedFlags = exportedInfo.Flags |
| prop.ExportedDirs = exportedInfo.IncludeDirs.Strings() |
| prop.ExportedSystemDirs = exportedInfo.SystemIncludeDirs.Strings() |
| prop.RelativeInstallPath = m.RelativeInstallPath() |
| |
| propOut := snapshotLibOut + ".json" |
| |
| j, err := json.Marshal(prop) |
| if err != nil { |
| ctx.Errorf("json marshal to %q failed: %#v", propOut, err) |
| return nil, false |
| } |
| ret = append(ret, writeStringToFile(ctx, string(j), propOut)) |
| } |
| return ret, true |
| } |
| |
| ctx.VisitAllModules(func(module android.Module) { |
| m, ok := module.(*Module) |
| if !ok || !m.Enabled() { |
| return |
| } |
| |
| apexInfo := ctx.ModuleProvider(module, android.ApexInfoProvider).(android.ApexInfo) |
| |
| l, vndkType, ok := isVndkSnapshotLibrary(ctx.DeviceConfig(), m, apexInfo) |
| if !ok { |
| return |
| } |
| |
| // install .so files for appropriate modules. |
| // Also install .json files if VNDK_SNAPSHOT_BUILD_ARTIFACTS |
| libs, ok := installVndkSnapshotLib(m, vndkType) |
| if !ok { |
| return |
| } |
| snapshotOutputs = append(snapshotOutputs, libs...) |
| |
| // These are for generating module_names.txt and module_paths.txt |
| stem := m.outputFile.Path().Base() |
| moduleNames[stem] = ctx.ModuleName(m) |
| modulePaths[stem] = ctx.ModuleDir(m) |
| |
| if len(m.NoticeFiles()) > 0 { |
| noticeName := stem + ".txt" |
| // skip already copied notice file |
| if _, ok := noticeBuilt[noticeName]; !ok { |
| noticeBuilt[noticeName] = true |
| snapshotOutputs = append(snapshotOutputs, combineNotices( |
| ctx, m.NoticeFiles(), filepath.Join(noticeDir, noticeName))) |
| } |
| } |
| |
| if ctx.Config().VndkSnapshotBuildArtifacts() { |
| headers = append(headers, l.snapshotHeaders()...) |
| } |
| }) |
| |
| // install all headers after removing duplicates |
| for _, header := range android.FirstUniquePaths(headers) { |
| snapshotOutputs = append(snapshotOutputs, copyFile( |
| ctx, header, filepath.Join(includeDir, header.String()))) |
| } |
| |
| // install *.libraries.txt except vndkcorevariant.libraries.txt |
| ctx.VisitAllModules(func(module android.Module) { |
| m, ok := module.(*vndkLibrariesTxt) |
| if !ok || !m.Enabled() || m.Name() == vndkUsingCoreVariantLibrariesTxt { |
| return |
| } |
| snapshotOutputs = append(snapshotOutputs, copyFile( |
| ctx, m.OutputFile(), filepath.Join(configsDir, m.Name()))) |
| }) |
| |
| /* |
| Dump a map to a list file as: |
| |
| {key1} {value1} |
| {key2} {value2} |
| ... |
| */ |
| installMapListFile := func(m map[string]string, path string) android.OutputPath { |
| var txtBuilder strings.Builder |
| for idx, k := range android.SortedStringKeys(m) { |
| if idx > 0 { |
| txtBuilder.WriteString("\\n") |
| } |
| txtBuilder.WriteString(k) |
| txtBuilder.WriteString(" ") |
| txtBuilder.WriteString(m[k]) |
| } |
| return writeStringToFile(ctx, txtBuilder.String(), path) |
| } |
| |
| /* |
| module_paths.txt contains paths on which VNDK modules are defined. |
| e.g., |
| libbase.so system/core/base |
| libc.so bionic/libc |
| ... |
| */ |
| snapshotOutputs = append(snapshotOutputs, installMapListFile(modulePaths, filepath.Join(configsDir, "module_paths.txt"))) |
| |
| /* |
| module_names.txt contains names as which VNDK modules are defined, |
| because output filename and module name can be different with stem and suffix properties. |
| |
| e.g., |
| libcutils.so libcutils |
| libprotobuf-cpp-full-3.9.2.so libprotobuf-cpp-full |
| ... |
| */ |
| snapshotOutputs = append(snapshotOutputs, installMapListFile(moduleNames, filepath.Join(configsDir, "module_names.txt"))) |
| |
| // All artifacts are ready. Sort them to normalize ninja and then zip. |
| sort.Slice(snapshotOutputs, func(i, j int) bool { |
| return snapshotOutputs[i].String() < snapshotOutputs[j].String() |
| }) |
| |
| zipPath := android.PathForOutput(ctx, snapshotDir, "android-vndk-"+ctx.DeviceConfig().DeviceArch()+".zip") |
| zipRule := android.NewRuleBuilder() |
| |
| // filenames in rspfile from FlagWithRspFileInputList might be single-quoted. Remove it with xargs |
| snapshotOutputList := android.PathForOutput(ctx, snapshotDir, "android-vndk-"+ctx.DeviceConfig().DeviceArch()+"_list") |
| zipRule.Command(). |
| Text("tr"). |
| FlagWithArg("-d ", "\\'"). |
| FlagWithRspFileInputList("< ", snapshotOutputs). |
| FlagWithOutput("> ", snapshotOutputList) |
| |
| zipRule.Temporary(snapshotOutputList) |
| |
| zipRule.Command(). |
| BuiltTool(ctx, "soong_zip"). |
| FlagWithOutput("-o ", zipPath). |
| FlagWithArg("-C ", android.PathForOutput(ctx, snapshotDir).String()). |
| FlagWithInput("-l ", snapshotOutputList) |
| |
| zipRule.Build(pctx, ctx, zipPath.String(), "vndk snapshot "+zipPath.String()) |
| zipRule.DeleteTemporaryFiles() |
| c.vndkSnapshotZipFile = android.OptionalPathForPath(zipPath) |
| } |
| |
| func getVndkFileName(m *Module) (string, error) { |
| if library, ok := m.linker.(*libraryDecorator); ok { |
| return library.getLibNameHelper(m.BaseModuleName(), true) + ".so", nil |
| } |
| if prebuilt, ok := m.linker.(*prebuiltLibraryLinker); ok { |
| return prebuilt.libraryDecorator.getLibNameHelper(m.BaseModuleName(), true) + ".so", nil |
| } |
| return "", fmt.Errorf("VNDK library should have libraryDecorator or prebuiltLibraryLinker as linker: %T", m.linker) |
| } |
| |
| func (c *vndkSnapshotSingleton) buildVndkLibrariesTxtFiles(ctx android.SingletonContext) { |
| llndk := android.SortedStringMapValues(llndkLibraries(ctx.Config())) |
| vndkcore := android.SortedStringMapValues(vndkCoreLibraries(ctx.Config())) |
| vndksp := android.SortedStringMapValues(vndkSpLibraries(ctx.Config())) |
| vndkprivate := android.SortedStringMapValues(vndkPrivateLibraries(ctx.Config())) |
| |
| // Build list of vndk libs as merged & tagged & filter-out(libclang_rt): |
| // Since each target have different set of libclang_rt.* files, |
| // keep the common set of files in vndk.libraries.txt |
| var merged []string |
| filterOutLibClangRt := func(libList []string) (filtered []string) { |
| for _, lib := range libList { |
| if !strings.HasPrefix(lib, "libclang_rt.") { |
| filtered = append(filtered, lib) |
| } |
| } |
| return |
| } |
| merged = append(merged, addPrefix(filterOutLibClangRt(llndk), "LLNDK: ")...) |
| merged = append(merged, addPrefix(vndksp, "VNDK-SP: ")...) |
| merged = append(merged, addPrefix(filterOutLibClangRt(vndkcore), "VNDK-core: ")...) |
| merged = append(merged, addPrefix(vndkprivate, "VNDK-private: ")...) |
| c.vndkLibrariesFile = android.PathForOutput(ctx, "vndk", "vndk.libraries.txt") |
| ctx.Build(pctx, android.BuildParams{ |
| Rule: android.WriteFile, |
| Output: c.vndkLibrariesFile, |
| Description: "Writing " + c.vndkLibrariesFile.String(), |
| Args: map[string]string{ |
| "content": strings.Join(merged, "\\n"), |
| }, |
| }) |
| } |
| |
| func (c *vndkSnapshotSingleton) MakeVars(ctx android.MakeVarsContext) { |
| // Make uses LLNDK_MOVED_TO_APEX_LIBRARIES to avoid installing libraries on /system if |
| // they been moved to an apex. |
| movedToApexLlndkLibraries := make(map[string]bool) |
| ctx.VisitAllModules(func(module android.Module) { |
| if m, ok := module.(*Module); ok { |
| if llndk, ok := m.linker.(*llndkStubDecorator); ok { |
| // Skip bionic libs, they are handled in different manner |
| name := m.BaseModuleName() |
| if llndk.movedToApex && !isBionic(m.BaseModuleName()) { |
| movedToApexLlndkLibraries[name] = true |
| } |
| } |
| } |
| }) |
| |
| ctx.Strict("LLNDK_MOVED_TO_APEX_LIBRARIES", |
| strings.Join(android.SortedStringKeys(movedToApexLlndkLibraries), " ")) |
| |
| // Make uses LLNDK_LIBRARIES to determine which libraries to install. |
| // HWASAN is only part of the LL-NDK in builds in which libc depends on HWASAN. |
| // Therefore, by removing the library here, we cause it to only be installed if libc |
| // depends on it. |
| installedLlndkLibraries := []string{} |
| for lib := range llndkLibraries(ctx.Config()) { |
| if strings.HasPrefix(lib, "libclang_rt.hwasan-") { |
| continue |
| } |
| installedLlndkLibraries = append(installedLlndkLibraries, lib) |
| } |
| sort.Strings(installedLlndkLibraries) |
| ctx.Strict("LLNDK_LIBRARIES", strings.Join(installedLlndkLibraries, " ")) |
| |
| ctx.Strict("VNDK_CORE_LIBRARIES", strings.Join(android.SortedStringKeys(vndkCoreLibraries(ctx.Config())), " ")) |
| ctx.Strict("VNDK_SAMEPROCESS_LIBRARIES", strings.Join(android.SortedStringKeys(vndkSpLibraries(ctx.Config())), " ")) |
| ctx.Strict("VNDK_PRIVATE_LIBRARIES", strings.Join(android.SortedStringKeys(vndkPrivateLibraries(ctx.Config())), " ")) |
| ctx.Strict("VNDK_USING_CORE_VARIANT_LIBRARIES", strings.Join(android.SortedStringKeys(vndkUsingCoreVariantLibraries(ctx.Config())), " ")) |
| |
| ctx.Strict("VNDK_LIBRARIES_FILE", c.vndkLibrariesFile.String()) |
| ctx.Strict("SOONG_VNDK_SNAPSHOT_ZIP", c.vndkSnapshotZipFile.String()) |
| } |