// Copyright (C) 2020 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 linkerconfig import ( "fmt" "sort" "strings" "github.com/google/blueprint/proptools" "android/soong/android" "android/soong/cc" "android/soong/etc" ) var ( pctx = android.NewPackageContext("android/soong/linkerconfig") ) func init() { pctx.HostBinToolVariable("conv_linker_config", "conv_linker_config") registerLinkerConfigBuildComponent(android.InitRegistrationContext) } func registerLinkerConfigBuildComponent(ctx android.RegistrationContext) { ctx.RegisterModuleType("linker_config", LinkerConfigFactory) } type linkerConfigProperties struct { // source linker configuration property file Src *string `android:"path"` // If set to true, allow module to be installed to one of the partitions. // Default value is true. // Installable should be marked as false for APEX configuration to avoid // conflicts of configuration on /system/etc directory. Installable *bool } type linkerConfig struct { android.ModuleBase properties linkerConfigProperties outputFilePath android.OutputPath installDirPath android.InstallPath } // Implement PrebuiltEtcModule interface to fit in APEX prebuilt list. var _ etc.PrebuiltEtcModule = (*linkerConfig)(nil) func (l *linkerConfig) BaseDir() string { return "etc" } func (l *linkerConfig) SubDir() string { return "" } func (l *linkerConfig) OutputFile() android.OutputPath { return l.outputFilePath } var _ android.OutputFileProducer = (*linkerConfig)(nil) func (l *linkerConfig) OutputFiles(tag string) (android.Paths, error) { switch tag { case "": return android.Paths{l.outputFilePath}, nil default: return nil, fmt.Errorf("unsupported module reference tag %q", tag) } } func (l *linkerConfig) GenerateAndroidBuildActions(ctx android.ModuleContext) { input := android.PathForModuleSrc(ctx, android.String(l.properties.Src)) output := android.PathForModuleOut(ctx, "linker.config.pb").OutputPath builder := android.NewRuleBuilder(pctx, ctx) BuildLinkerConfig(ctx, builder, input, nil, nil, output) builder.Build("conv_linker_config", "Generate linker config protobuf "+output.String()) l.outputFilePath = output l.installDirPath = android.PathForModuleInstall(ctx, "etc") if !proptools.BoolDefault(l.properties.Installable, true) { l.SkipInstall() } ctx.InstallFile(l.installDirPath, l.outputFilePath.Base(), l.outputFilePath) } func BuildLinkerConfig(ctx android.ModuleContext, builder *android.RuleBuilder, input android.Path, provideModules []android.Module, requireModules []android.Module, output android.OutputPath) { // First, convert the input json to protobuf format interimOutput := android.PathForModuleOut(ctx, "temp.pb") builder.Command(). BuiltTool("conv_linker_config"). Flag("proto"). FlagWithInput("-s ", input). FlagWithOutput("-o ", interimOutput) // Secondly, if there's provideLibs gathered from provideModules, append them var provideLibs []string for _, m := range provideModules { if c, ok := m.(*cc.Module); ok && cc.IsStubTarget(c) { for _, ps := range c.PackagingSpecs() { provideLibs = append(provideLibs, ps.FileName()) } } } provideLibs = android.FirstUniqueStrings(provideLibs) sort.Strings(provideLibs) var requireLibs []string for _, m := range requireModules { if c, ok := m.(*cc.Module); ok && c.HasStubsVariants() && !c.Host() { requireLibs = append(requireLibs, c.ImplementationModuleName(ctx)+".so") } } requireLibs = android.FirstUniqueStrings(requireLibs) sort.Strings(requireLibs) if len(provideLibs) > 0 { prevOutput := interimOutput interimOutput = android.PathForModuleOut(ctx, "temp_provideLibs.pb") builder.Command(). BuiltTool("conv_linker_config"). Flag("append"). FlagWithInput("-s ", prevOutput). FlagWithOutput("-o ", interimOutput). FlagWithArg("--key ", "provideLibs"). FlagWithArg("--value ", proptools.ShellEscapeIncludingSpaces(strings.Join(provideLibs, " "))) builder.Temporary(prevOutput) } if len(requireLibs) > 0 { prevOutput := interimOutput interimOutput = android.PathForModuleOut(ctx, "temp_requireLibs.pb") builder.Command(). BuiltTool("conv_linker_config"). Flag("append"). FlagWithInput("-s ", prevOutput). FlagWithOutput("-o ", interimOutput). FlagWithArg("--key ", "requireLibs"). FlagWithArg("--value ", proptools.ShellEscapeIncludingSpaces(strings.Join(requireLibs, " "))) builder.Temporary(prevOutput) } // cp to the final output builder.Command().Text("cp").Input(interimOutput).Output(output) builder.Temporary(interimOutput) builder.DeleteTemporaryFiles() } // linker_config generates protobuf file from json file. This protobuf file will be used from // linkerconfig while generating ld.config.txt. Format of this file can be found from // https://android.googlesource.com/platform/system/linkerconfig/+/main/README.md func LinkerConfigFactory() android.Module { m := &linkerConfig{} m.AddProperties(&m.properties) android.InitAndroidArchModule(m, android.HostAndDeviceSupported, android.MultilibFirst) return m } func (l *linkerConfig) AndroidMkEntries() []android.AndroidMkEntries { installable := proptools.BoolDefault(l.properties.Installable, true) return []android.AndroidMkEntries{android.AndroidMkEntries{ Class: "ETC", OutputFile: android.OptionalPathForPath(l.outputFilePath), ExtraEntries: []android.AndroidMkExtraEntriesFunc{ func(ctx android.AndroidMkExtraEntriesContext, entries *android.AndroidMkEntries) { entries.SetString("LOCAL_MODULE_PATH", l.installDirPath.String()) entries.SetString("LOCAL_INSTALLED_MODULE_STEM", l.outputFilePath.Base()) entries.SetBoolIfTrue("LOCAL_UNINSTALLABLE_MODULE", !installable) }, }, }} }