| // 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 ( |
| "fmt" |
| "strings" |
| |
| "android/soong/android" |
| "android/soong/etc" |
| ) |
| |
| var ( |
| llndkLibrarySuffix = ".llndk" |
| ) |
| |
| // Holds properties to describe a stub shared library based on the provided version file. |
| type llndkLibraryProperties struct { |
| // Relative path to the symbol map. |
| // An example file can be seen here: TODO(danalbert): Make an example. |
| Symbol_file *string |
| |
| // Whether to export any headers as -isystem instead of -I. Mainly for use by |
| // bionic/libc. |
| Export_headers_as_system *bool |
| |
| // Which headers to process with versioner. This really only handles |
| // bionic/libc/include right now. |
| Export_preprocessed_headers []string |
| |
| // Whether the system library uses symbol versions. |
| Unversioned *bool |
| |
| // list of llndk headers to re-export include directories from. |
| Export_llndk_headers []string |
| |
| // list of directories relative to the Blueprints file that willbe added to the include path |
| // (using -I) for any module that links against the LLNDK variant of this module, replacing |
| // any that were listed outside the llndk clause. |
| Override_export_include_dirs []string |
| |
| // whether this module can be directly depended upon by libs that are installed |
| // to /vendor and /product. |
| // When set to true, this module can only be depended on by VNDK libraries, not |
| // vendor nor product libraries. This effectively hides this module from |
| // non-system modules. Default value is false. |
| Private *bool |
| |
| // if true, make this module available to provide headers to other modules that set |
| // llndk.symbol_file. |
| Llndk_headers *bool |
| } |
| |
| func makeLlndkVars(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 library := moduleLibraryInterface(module); library != nil && library.hasLLNDKStubs() { |
| // Skip bionic libs, they are handled in different manner |
| name := library.implementationModuleName(module.(*Module).BaseModuleName()) |
| if module.(android.ApexModule).DirectlyInAnyApex() && !isBionic(name) { |
| movedToApexLlndkLibraries[name] = true |
| } |
| } |
| }) |
| |
| ctx.Strict("LLNDK_MOVED_TO_APEX_LIBRARIES", |
| strings.Join(android.SortedKeys(movedToApexLlndkLibraries), " ")) |
| } |
| |
| func init() { |
| RegisterLlndkLibraryTxtType(android.InitRegistrationContext) |
| } |
| |
| func RegisterLlndkLibraryTxtType(ctx android.RegistrationContext) { |
| ctx.RegisterParallelSingletonModuleType("llndk_libraries_txt", llndkLibrariesTxtFactory) |
| } |
| |
| type llndkLibrariesTxtModule struct { |
| android.SingletonModuleBase |
| |
| outputFile android.OutputPath |
| moduleNames []string |
| fileNames []string |
| } |
| |
| var _ etc.PrebuiltEtcModule = &llndkLibrariesTxtModule{} |
| |
| // llndk_libraries_txt is a singleton module whose content is a list of LLNDK libraries |
| // generated by Soong but can be referenced by other modules. |
| // For example, apex_vndk can depend on these files as prebuilt. |
| // 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. |
| func llndkLibrariesTxtFactory() android.SingletonModule { |
| m := &llndkLibrariesTxtModule{} |
| android.InitAndroidArchModule(m, android.DeviceSupported, android.MultilibCommon) |
| return m |
| } |
| |
| func (txt *llndkLibrariesTxtModule) GenerateAndroidBuildActions(ctx android.ModuleContext) { |
| filename := txt.Name() |
| |
| txt.outputFile = android.PathForModuleOut(ctx, filename).OutputPath |
| |
| installPath := android.PathForModuleInstall(ctx, "etc") |
| ctx.InstallFile(installPath, filename, txt.outputFile) |
| |
| ctx.SetOutputFiles(android.Paths{txt.outputFile}, "") |
| } |
| |
| func getVndkFileName(m *Module) (string, error) { |
| if library, ok := m.linker.(*libraryDecorator); ok { |
| return library.getLibNameHelper(m.BaseModuleName(), true, false) + ".so", nil |
| } |
| if prebuilt, ok := m.linker.(*prebuiltLibraryLinker); ok { |
| return prebuilt.libraryDecorator.getLibNameHelper(m.BaseModuleName(), true, false) + ".so", nil |
| } |
| return "", fmt.Errorf("VNDK library should have libraryDecorator or prebuiltLibraryLinker as linker: %T", m.linker) |
| } |
| |
| func (txt *llndkLibrariesTxtModule) GenerateSingletonBuildActions(ctx android.SingletonContext) { |
| if txt.outputFile.String() == "" { |
| // Skip if target file path is empty |
| return |
| } |
| |
| ctx.VisitAllModules(func(m android.Module) { |
| if c, ok := m.(*Module); ok && c.VendorProperties.IsLLNDK && !c.Header() && !c.IsVndkPrebuiltLibrary() { |
| filename, err := getVndkFileName(c) |
| if err != nil { |
| ctx.ModuleErrorf(m, "%s", err) |
| } |
| |
| if !strings.HasPrefix(ctx.ModuleName(m), "libclang_rt.hwasan") { |
| txt.moduleNames = append(txt.moduleNames, ctx.ModuleName(m)) |
| } |
| txt.fileNames = append(txt.fileNames, filename) |
| } |
| }) |
| txt.moduleNames = android.SortedUniqueStrings(txt.moduleNames) |
| txt.fileNames = android.SortedUniqueStrings(txt.fileNames) |
| |
| android.WriteFileRule(ctx, txt.outputFile, strings.Join(txt.fileNames, "\n")) |
| } |
| |
| func (txt *llndkLibrariesTxtModule) AndroidMkEntries() []android.AndroidMkEntries { |
| return []android.AndroidMkEntries{{ |
| Class: "ETC", |
| OutputFile: android.OptionalPathForPath(txt.outputFile), |
| ExtraEntries: []android.AndroidMkExtraEntriesFunc{ |
| func(ctx android.AndroidMkExtraEntriesContext, entries *android.AndroidMkEntries) { |
| entries.SetString("LOCAL_MODULE_STEM", txt.outputFile.Base()) |
| }, |
| }, |
| }} |
| } |
| |
| func (txt *llndkLibrariesTxtModule) MakeVars(ctx android.MakeVarsContext) { |
| ctx.Strict("LLNDK_LIBRARIES", strings.Join(txt.moduleNames, " ")) |
| } |
| |
| // PrebuiltEtcModule interface |
| func (txt *llndkLibrariesTxtModule) BaseDir() string { |
| return "etc" |
| } |
| |
| // PrebuiltEtcModule interface |
| func (txt *llndkLibrariesTxtModule) SubDir() string { |
| return "" |
| } |
| |
| func (txt *llndkLibrariesTxtModule) OutputFiles(tag string) (android.Paths, error) { |
| return android.Paths{txt.outputFile}, nil |
| } |
| |
| func llndkMutator(mctx android.BottomUpMutatorContext) { |
| m, ok := mctx.Module().(*Module) |
| if !ok { |
| return |
| } |
| |
| if shouldSkipLlndkMutator(mctx, m) { |
| return |
| } |
| |
| lib, isLib := m.linker.(*libraryDecorator) |
| prebuiltLib, isPrebuiltLib := m.linker.(*prebuiltLibraryLinker) |
| |
| if m.InVendorOrProduct() && isLib && lib.hasLLNDKStubs() { |
| m.VendorProperties.IsLLNDK = true |
| } |
| if m.InVendorOrProduct() && isPrebuiltLib && prebuiltLib.hasLLNDKStubs() { |
| m.VendorProperties.IsLLNDK = true |
| } |
| |
| if vndkprebuilt, ok := m.linker.(*vndkPrebuiltLibraryDecorator); ok { |
| if !Bool(vndkprebuilt.properties.Vndk.Enabled) { |
| m.VendorProperties.IsLLNDK = true |
| } |
| } |
| } |
| |
| // Check for modules that mustn't be LLNDK |
| func shouldSkipLlndkMutator(mctx android.BottomUpMutatorContext, m *Module) bool { |
| if !m.Enabled(mctx) { |
| return true |
| } |
| if !m.Device() { |
| return true |
| } |
| if m.Target().NativeBridge == android.NativeBridgeEnabled { |
| return true |
| } |
| return false |
| } |