xref: /aosp_15_r20/build/soong/cc/llndk_library.go (revision 333d2b3687b3a337dbcca9d65000bca186795e39)
1*333d2b36SAndroid Build Coastguard Worker// Copyright 2017 Google Inc. All rights reserved.
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 cc
16*333d2b36SAndroid Build Coastguard Worker
17*333d2b36SAndroid Build Coastguard Workerimport (
18*333d2b36SAndroid Build Coastguard Worker	"fmt"
19*333d2b36SAndroid Build Coastguard Worker	"strings"
20*333d2b36SAndroid Build Coastguard Worker
21*333d2b36SAndroid Build Coastguard Worker	"android/soong/android"
22*333d2b36SAndroid Build Coastguard Worker	"android/soong/etc"
23*333d2b36SAndroid Build Coastguard Worker)
24*333d2b36SAndroid Build Coastguard Worker
25*333d2b36SAndroid Build Coastguard Workervar (
26*333d2b36SAndroid Build Coastguard Worker	llndkLibrarySuffix = ".llndk"
27*333d2b36SAndroid Build Coastguard Worker)
28*333d2b36SAndroid Build Coastguard Worker
29*333d2b36SAndroid Build Coastguard Worker// Holds properties to describe a stub shared library based on the provided version file.
30*333d2b36SAndroid Build Coastguard Workertype llndkLibraryProperties struct {
31*333d2b36SAndroid Build Coastguard Worker	// Relative path to the symbol map.
32*333d2b36SAndroid Build Coastguard Worker	// An example file can be seen here: TODO(danalbert): Make an example.
33*333d2b36SAndroid Build Coastguard Worker	Symbol_file *string `android:"path,arch_variant"`
34*333d2b36SAndroid Build Coastguard Worker
35*333d2b36SAndroid Build Coastguard Worker	// Whether to export any headers as -isystem instead of -I. Mainly for use by
36*333d2b36SAndroid Build Coastguard Worker	// bionic/libc.
37*333d2b36SAndroid Build Coastguard Worker	Export_headers_as_system *bool
38*333d2b36SAndroid Build Coastguard Worker
39*333d2b36SAndroid Build Coastguard Worker	// Whether the system library uses symbol versions.
40*333d2b36SAndroid Build Coastguard Worker	Unversioned *bool
41*333d2b36SAndroid Build Coastguard Worker
42*333d2b36SAndroid Build Coastguard Worker	// list of llndk headers to re-export include directories from.
43*333d2b36SAndroid Build Coastguard Worker	Export_llndk_headers []string
44*333d2b36SAndroid Build Coastguard Worker
45*333d2b36SAndroid Build Coastguard Worker	// list of directories relative to the Blueprints file that willbe added to the include path
46*333d2b36SAndroid Build Coastguard Worker	// (using -I) for any module that links against the LLNDK variant of this module, replacing
47*333d2b36SAndroid Build Coastguard Worker	// any that were listed outside the llndk clause.
48*333d2b36SAndroid Build Coastguard Worker	Override_export_include_dirs []string
49*333d2b36SAndroid Build Coastguard Worker
50*333d2b36SAndroid Build Coastguard Worker	// whether this module can be directly depended upon by libs that are installed
51*333d2b36SAndroid Build Coastguard Worker	// to /vendor and /product.
52*333d2b36SAndroid Build Coastguard Worker	// When set to true, this module can only be depended on by VNDK libraries, not
53*333d2b36SAndroid Build Coastguard Worker	// vendor nor product libraries. This effectively hides this module from
54*333d2b36SAndroid Build Coastguard Worker	// non-system modules. Default value is false.
55*333d2b36SAndroid Build Coastguard Worker	Private *bool
56*333d2b36SAndroid Build Coastguard Worker
57*333d2b36SAndroid Build Coastguard Worker	// if true, make this module available to provide headers to other modules that set
58*333d2b36SAndroid Build Coastguard Worker	// llndk.symbol_file.
59*333d2b36SAndroid Build Coastguard Worker	Llndk_headers *bool
60*333d2b36SAndroid Build Coastguard Worker
61*333d2b36SAndroid Build Coastguard Worker	// moved_to_apex marks this module has having been distributed through an apex module.
62*333d2b36SAndroid Build Coastguard Worker	Moved_to_apex *bool
63*333d2b36SAndroid Build Coastguard Worker}
64*333d2b36SAndroid Build Coastguard Worker
65*333d2b36SAndroid Build Coastguard Workerfunc makeLlndkVars(ctx android.MakeVarsContext) {
66*333d2b36SAndroid Build Coastguard Worker	// Make uses LLNDK_MOVED_TO_APEX_LIBRARIES to generate the linker config.
67*333d2b36SAndroid Build Coastguard Worker	movedToApexLlndkLibraries := make(map[string]bool)
68*333d2b36SAndroid Build Coastguard Worker	ctx.VisitAllModules(func(module android.Module) {
69*333d2b36SAndroid Build Coastguard Worker		if library := moduleLibraryInterface(module); library != nil && library.hasLLNDKStubs() {
70*333d2b36SAndroid Build Coastguard Worker			if library.isLLNDKMovedToApex() {
71*333d2b36SAndroid Build Coastguard Worker				name := library.implementationModuleName(module.(*Module).BaseModuleName())
72*333d2b36SAndroid Build Coastguard Worker				movedToApexLlndkLibraries[name] = true
73*333d2b36SAndroid Build Coastguard Worker			}
74*333d2b36SAndroid Build Coastguard Worker		}
75*333d2b36SAndroid Build Coastguard Worker	})
76*333d2b36SAndroid Build Coastguard Worker
77*333d2b36SAndroid Build Coastguard Worker	ctx.Strict("LLNDK_MOVED_TO_APEX_LIBRARIES",
78*333d2b36SAndroid Build Coastguard Worker		strings.Join(android.SortedKeys(movedToApexLlndkLibraries), " "))
79*333d2b36SAndroid Build Coastguard Worker}
80*333d2b36SAndroid Build Coastguard Worker
81*333d2b36SAndroid Build Coastguard Workerfunc init() {
82*333d2b36SAndroid Build Coastguard Worker	RegisterLlndkLibraryTxtType(android.InitRegistrationContext)
83*333d2b36SAndroid Build Coastguard Worker}
84*333d2b36SAndroid Build Coastguard Worker
85*333d2b36SAndroid Build Coastguard Workerfunc RegisterLlndkLibraryTxtType(ctx android.RegistrationContext) {
86*333d2b36SAndroid Build Coastguard Worker	ctx.RegisterParallelSingletonModuleType("llndk_libraries_txt", llndkLibrariesTxtFactory)
87*333d2b36SAndroid Build Coastguard Worker}
88*333d2b36SAndroid Build Coastguard Worker
89*333d2b36SAndroid Build Coastguard Workertype llndkLibrariesTxtModule struct {
90*333d2b36SAndroid Build Coastguard Worker	android.SingletonModuleBase
91*333d2b36SAndroid Build Coastguard Worker
92*333d2b36SAndroid Build Coastguard Worker	outputFile  android.OutputPath
93*333d2b36SAndroid Build Coastguard Worker	moduleNames []string
94*333d2b36SAndroid Build Coastguard Worker	fileNames   []string
95*333d2b36SAndroid Build Coastguard Worker}
96*333d2b36SAndroid Build Coastguard Worker
97*333d2b36SAndroid Build Coastguard Workervar _ etc.PrebuiltEtcModule = &llndkLibrariesTxtModule{}
98*333d2b36SAndroid Build Coastguard Worker
99*333d2b36SAndroid Build Coastguard Worker// llndk_libraries_txt is a singleton module whose content is a list of LLNDK libraries
100*333d2b36SAndroid Build Coastguard Worker// generated by Soong but can be referenced by other modules.
101*333d2b36SAndroid Build Coastguard Worker// For example, apex_vndk can depend on these files as prebuilt.
102*333d2b36SAndroid Build Coastguard Worker// Make uses LLNDK_LIBRARIES to determine which libraries to install.
103*333d2b36SAndroid Build Coastguard Worker// HWASAN is only part of the LL-NDK in builds in which libc depends on HWASAN.
104*333d2b36SAndroid Build Coastguard Worker// Therefore, by removing the library here, we cause it to only be installed if libc
105*333d2b36SAndroid Build Coastguard Worker// depends on it.
106*333d2b36SAndroid Build Coastguard Workerfunc llndkLibrariesTxtFactory() android.SingletonModule {
107*333d2b36SAndroid Build Coastguard Worker	m := &llndkLibrariesTxtModule{}
108*333d2b36SAndroid Build Coastguard Worker	android.InitAndroidArchModule(m, android.DeviceSupported, android.MultilibCommon)
109*333d2b36SAndroid Build Coastguard Worker	return m
110*333d2b36SAndroid Build Coastguard Worker}
111*333d2b36SAndroid Build Coastguard Worker
112*333d2b36SAndroid Build Coastguard Workerfunc (txt *llndkLibrariesTxtModule) GenerateAndroidBuildActions(ctx android.ModuleContext) {
113*333d2b36SAndroid Build Coastguard Worker	filename := txt.Name()
114*333d2b36SAndroid Build Coastguard Worker
115*333d2b36SAndroid Build Coastguard Worker	txt.outputFile = android.PathForModuleOut(ctx, filename).OutputPath
116*333d2b36SAndroid Build Coastguard Worker
117*333d2b36SAndroid Build Coastguard Worker	installPath := android.PathForModuleInstall(ctx, "etc")
118*333d2b36SAndroid Build Coastguard Worker	ctx.InstallFile(installPath, filename, txt.outputFile)
119*333d2b36SAndroid Build Coastguard Worker
120*333d2b36SAndroid Build Coastguard Worker	ctx.SetOutputFiles(android.Paths{txt.outputFile}, "")
121*333d2b36SAndroid Build Coastguard Worker}
122*333d2b36SAndroid Build Coastguard Worker
123*333d2b36SAndroid Build Coastguard Workerfunc getVndkFileName(m *Module) (string, error) {
124*333d2b36SAndroid Build Coastguard Worker	if library, ok := m.linker.(*libraryDecorator); ok {
125*333d2b36SAndroid Build Coastguard Worker		return library.getLibNameHelper(m.BaseModuleName(), true, false) + ".so", nil
126*333d2b36SAndroid Build Coastguard Worker	}
127*333d2b36SAndroid Build Coastguard Worker	if prebuilt, ok := m.linker.(*prebuiltLibraryLinker); ok {
128*333d2b36SAndroid Build Coastguard Worker		return prebuilt.libraryDecorator.getLibNameHelper(m.BaseModuleName(), true, false) + ".so", nil
129*333d2b36SAndroid Build Coastguard Worker	}
130*333d2b36SAndroid Build Coastguard Worker	return "", fmt.Errorf("VNDK library should have libraryDecorator or prebuiltLibraryLinker as linker: %T", m.linker)
131*333d2b36SAndroid Build Coastguard Worker}
132*333d2b36SAndroid Build Coastguard Worker
133*333d2b36SAndroid Build Coastguard Workerfunc (txt *llndkLibrariesTxtModule) GenerateSingletonBuildActions(ctx android.SingletonContext) {
134*333d2b36SAndroid Build Coastguard Worker	if txt.outputFile.String() == "" {
135*333d2b36SAndroid Build Coastguard Worker		// Skip if target file path is empty
136*333d2b36SAndroid Build Coastguard Worker		return
137*333d2b36SAndroid Build Coastguard Worker	}
138*333d2b36SAndroid Build Coastguard Worker
139*333d2b36SAndroid Build Coastguard Worker	ctx.VisitAllModules(func(m android.Module) {
140*333d2b36SAndroid Build Coastguard Worker		if c, ok := m.(*Module); ok && c.VendorProperties.IsLLNDK && !c.Header() && !c.IsVndkPrebuiltLibrary() {
141*333d2b36SAndroid Build Coastguard Worker			filename, err := getVndkFileName(c)
142*333d2b36SAndroid Build Coastguard Worker			if err != nil {
143*333d2b36SAndroid Build Coastguard Worker				ctx.ModuleErrorf(m, "%s", err)
144*333d2b36SAndroid Build Coastguard Worker			}
145*333d2b36SAndroid Build Coastguard Worker
146*333d2b36SAndroid Build Coastguard Worker			if !strings.HasPrefix(ctx.ModuleName(m), "libclang_rt.hwasan") {
147*333d2b36SAndroid Build Coastguard Worker				txt.moduleNames = append(txt.moduleNames, ctx.ModuleName(m))
148*333d2b36SAndroid Build Coastguard Worker			}
149*333d2b36SAndroid Build Coastguard Worker			txt.fileNames = append(txt.fileNames, filename)
150*333d2b36SAndroid Build Coastguard Worker		}
151*333d2b36SAndroid Build Coastguard Worker	})
152*333d2b36SAndroid Build Coastguard Worker	txt.moduleNames = android.SortedUniqueStrings(txt.moduleNames)
153*333d2b36SAndroid Build Coastguard Worker	txt.fileNames = android.SortedUniqueStrings(txt.fileNames)
154*333d2b36SAndroid Build Coastguard Worker
155*333d2b36SAndroid Build Coastguard Worker	android.WriteFileRule(ctx, txt.outputFile, strings.Join(txt.fileNames, "\n"))
156*333d2b36SAndroid Build Coastguard Worker}
157*333d2b36SAndroid Build Coastguard Worker
158*333d2b36SAndroid Build Coastguard Workerfunc (txt *llndkLibrariesTxtModule) AndroidMkEntries() []android.AndroidMkEntries {
159*333d2b36SAndroid Build Coastguard Worker	return []android.AndroidMkEntries{{
160*333d2b36SAndroid Build Coastguard Worker		Class:      "ETC",
161*333d2b36SAndroid Build Coastguard Worker		OutputFile: android.OptionalPathForPath(txt.outputFile),
162*333d2b36SAndroid Build Coastguard Worker		ExtraEntries: []android.AndroidMkExtraEntriesFunc{
163*333d2b36SAndroid Build Coastguard Worker			func(ctx android.AndroidMkExtraEntriesContext, entries *android.AndroidMkEntries) {
164*333d2b36SAndroid Build Coastguard Worker				entries.SetString("LOCAL_MODULE_STEM", txt.outputFile.Base())
165*333d2b36SAndroid Build Coastguard Worker			},
166*333d2b36SAndroid Build Coastguard Worker		},
167*333d2b36SAndroid Build Coastguard Worker	}}
168*333d2b36SAndroid Build Coastguard Worker}
169*333d2b36SAndroid Build Coastguard Worker
170*333d2b36SAndroid Build Coastguard Workerfunc (txt *llndkLibrariesTxtModule) MakeVars(ctx android.MakeVarsContext) {
171*333d2b36SAndroid Build Coastguard Worker	ctx.Strict("LLNDK_LIBRARIES", strings.Join(txt.moduleNames, " "))
172*333d2b36SAndroid Build Coastguard Worker}
173*333d2b36SAndroid Build Coastguard Worker
174*333d2b36SAndroid Build Coastguard Worker// PrebuiltEtcModule interface
175*333d2b36SAndroid Build Coastguard Workerfunc (txt *llndkLibrariesTxtModule) BaseDir() string {
176*333d2b36SAndroid Build Coastguard Worker	return "etc"
177*333d2b36SAndroid Build Coastguard Worker}
178*333d2b36SAndroid Build Coastguard Worker
179*333d2b36SAndroid Build Coastguard Worker// PrebuiltEtcModule interface
180*333d2b36SAndroid Build Coastguard Workerfunc (txt *llndkLibrariesTxtModule) SubDir() string {
181*333d2b36SAndroid Build Coastguard Worker	return ""
182*333d2b36SAndroid Build Coastguard Worker}
183*333d2b36SAndroid Build Coastguard Worker
184*333d2b36SAndroid Build Coastguard Workerfunc llndkMutator(mctx android.BottomUpMutatorContext) {
185*333d2b36SAndroid Build Coastguard Worker	m, ok := mctx.Module().(*Module)
186*333d2b36SAndroid Build Coastguard Worker	if !ok {
187*333d2b36SAndroid Build Coastguard Worker		return
188*333d2b36SAndroid Build Coastguard Worker	}
189*333d2b36SAndroid Build Coastguard Worker
190*333d2b36SAndroid Build Coastguard Worker	if shouldSkipLlndkMutator(mctx, m) {
191*333d2b36SAndroid Build Coastguard Worker		return
192*333d2b36SAndroid Build Coastguard Worker	}
193*333d2b36SAndroid Build Coastguard Worker
194*333d2b36SAndroid Build Coastguard Worker	lib, isLib := m.linker.(*libraryDecorator)
195*333d2b36SAndroid Build Coastguard Worker	prebuiltLib, isPrebuiltLib := m.linker.(*prebuiltLibraryLinker)
196*333d2b36SAndroid Build Coastguard Worker
197*333d2b36SAndroid Build Coastguard Worker	if m.InVendorOrProduct() && isLib && lib.hasLLNDKStubs() {
198*333d2b36SAndroid Build Coastguard Worker		m.VendorProperties.IsLLNDK = true
199*333d2b36SAndroid Build Coastguard Worker	}
200*333d2b36SAndroid Build Coastguard Worker	if m.InVendorOrProduct() && isPrebuiltLib && prebuiltLib.hasLLNDKStubs() {
201*333d2b36SAndroid Build Coastguard Worker		m.VendorProperties.IsLLNDK = true
202*333d2b36SAndroid Build Coastguard Worker	}
203*333d2b36SAndroid Build Coastguard Worker
204*333d2b36SAndroid Build Coastguard Worker	if vndkprebuilt, ok := m.linker.(*vndkPrebuiltLibraryDecorator); ok {
205*333d2b36SAndroid Build Coastguard Worker		if !Bool(vndkprebuilt.properties.Vndk.Enabled) {
206*333d2b36SAndroid Build Coastguard Worker			m.VendorProperties.IsLLNDK = true
207*333d2b36SAndroid Build Coastguard Worker		}
208*333d2b36SAndroid Build Coastguard Worker	}
209*333d2b36SAndroid Build Coastguard Worker}
210*333d2b36SAndroid Build Coastguard Worker
211*333d2b36SAndroid Build Coastguard Worker// Check for modules that mustn't be LLNDK
212*333d2b36SAndroid Build Coastguard Workerfunc shouldSkipLlndkMutator(mctx android.BottomUpMutatorContext, m *Module) bool {
213*333d2b36SAndroid Build Coastguard Worker	if !m.Enabled(mctx) {
214*333d2b36SAndroid Build Coastguard Worker		return true
215*333d2b36SAndroid Build Coastguard Worker	}
216*333d2b36SAndroid Build Coastguard Worker	if !m.Device() {
217*333d2b36SAndroid Build Coastguard Worker		return true
218*333d2b36SAndroid Build Coastguard Worker	}
219*333d2b36SAndroid Build Coastguard Worker	if m.Target().NativeBridge == android.NativeBridgeEnabled {
220*333d2b36SAndroid Build Coastguard Worker		return true
221*333d2b36SAndroid Build Coastguard Worker	}
222*333d2b36SAndroid Build Coastguard Worker	return false
223*333d2b36SAndroid Build Coastguard Worker}
224