• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1// Copyright 2019 Google Inc. All rights reserved.
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7//     http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15package cc
16
17import (
18	"path/filepath"
19
20	"android/soong/android"
21
22	"github.com/google/blueprint"
23	"github.com/google/blueprint/proptools"
24)
25
26// This file contains support for using cc library modules within an sdk.
27
28var sharedLibrarySdkMemberType = &librarySdkMemberType{
29	SdkMemberTypeBase: android.SdkMemberTypeBase{
30		PropertyName:          "native_shared_libs",
31		SupportsSdk:           true,
32		HostOsDependent:       true,
33		SupportedLinkageNames: []string{"shared"},
34	},
35	prebuiltModuleType: "cc_prebuilt_library_shared",
36}
37
38var staticLibrarySdkMemberType = &librarySdkMemberType{
39	SdkMemberTypeBase: android.SdkMemberTypeBase{
40		PropertyName:          "native_static_libs",
41		SupportsSdk:           true,
42		HostOsDependent:       true,
43		SupportedLinkageNames: []string{"static"},
44	},
45	prebuiltModuleType: "cc_prebuilt_library_static",
46}
47
48var staticAndSharedLibrarySdkMemberType = &librarySdkMemberType{
49	SdkMemberTypeBase: android.SdkMemberTypeBase{
50		PropertyName:           "native_libs",
51		OverridesPropertyNames: map[string]bool{"native_shared_libs": true, "native_static_libs": true},
52		SupportsSdk:            true,
53		HostOsDependent:        true,
54		SupportedLinkageNames:  []string{"static", "shared"},
55	},
56	prebuiltModuleType: "cc_prebuilt_library",
57}
58
59func init() {
60	// Register sdk member types.
61	android.RegisterSdkMemberType(sharedLibrarySdkMemberType)
62	android.RegisterSdkMemberType(staticLibrarySdkMemberType)
63	android.RegisterSdkMemberType(staticAndSharedLibrarySdkMemberType)
64}
65
66type librarySdkMemberType struct {
67	android.SdkMemberTypeBase
68
69	prebuiltModuleType string
70
71	noOutputFiles bool // True if there are no srcs files.
72
73}
74
75func (mt *librarySdkMemberType) AddDependencies(ctx android.SdkDependencyContext, dependencyTag blueprint.DependencyTag, names []string) {
76	// The base set of targets which does not include native bridge targets.
77	defaultTargets := ctx.MultiTargets()
78
79	// The lazily created list of native bridge targets.
80	var includeNativeBridgeTargets []android.Target
81
82	for _, lib := range names {
83		targets := defaultTargets
84
85		// If native bridge support is required in the sdk snapshot then add native bridge targets to
86		// the basic list of targets that are required.
87		nativeBridgeSupport := ctx.RequiresTrait(lib, nativeBridgeSdkTrait)
88		if nativeBridgeSupport && ctx.Device() {
89			// If not already computed then compute the list of native bridge targets.
90			if includeNativeBridgeTargets == nil {
91				includeNativeBridgeTargets = append([]android.Target{}, defaultTargets...)
92				allAndroidTargets := ctx.Config().Targets[android.Android]
93				for _, possibleNativeBridgeTarget := range allAndroidTargets {
94					if possibleNativeBridgeTarget.NativeBridge == android.NativeBridgeEnabled {
95						includeNativeBridgeTargets = append(includeNativeBridgeTargets, possibleNativeBridgeTarget)
96					}
97				}
98			}
99
100			// Include the native bridge targets as well.
101			targets = includeNativeBridgeTargets
102		}
103
104		// memberDependency encapsulates information about the dependencies to add for this member.
105		type memberDependency struct {
106			// The targets to depend upon.
107			targets []android.Target
108
109			// Additional image variations to depend upon, is either nil for no image variation or
110			// contains a single image variation.
111			imageVariations []blueprint.Variation
112		}
113
114		// Extract the name and version from the module name.
115		name, version := StubsLibNameAndVersion(lib)
116		if version == "" {
117			version = "latest"
118		}
119
120		// Compute the set of dependencies to add.
121		var memberDependencies []memberDependency
122		if ctx.Host() {
123			// Host does not support image variations so add a dependency without any.
124			memberDependencies = append(memberDependencies, memberDependency{
125				targets: targets,
126			})
127		} else {
128			// Otherwise, this is targeting the device so add a dependency on the core image variation
129			// (image:"").
130			memberDependencies = append(memberDependencies, memberDependency{
131				imageVariations: []blueprint.Variation{{Mutator: "image", Variation: android.CoreVariation}},
132				targets:         targets,
133			})
134
135			// If required add additional dependencies on the image:ramdisk variants.
136			if ctx.RequiresTrait(lib, ramdiskImageRequiredSdkTrait) {
137				memberDependencies = append(memberDependencies, memberDependency{
138					imageVariations: []blueprint.Variation{{Mutator: "image", Variation: android.RamdiskVariation}},
139					// Only add a dependency on the first target as that is the only one which will have an
140					// image:ramdisk variant.
141					targets: targets[:1],
142				})
143			}
144
145			// If required add additional dependencies on the image:recovery variants.
146			if ctx.RequiresTrait(lib, recoveryImageRequiredSdkTrait) {
147				memberDependencies = append(memberDependencies, memberDependency{
148					imageVariations: []blueprint.Variation{{Mutator: "image", Variation: android.RecoveryVariation}},
149					// Only add a dependency on the first target as that is the only one which will have an
150					// image:recovery variant.
151					targets: targets[:1],
152				})
153			}
154		}
155
156		// For each dependency in the list add dependencies on the targets with the correct variations.
157		for _, dependency := range memberDependencies {
158			// For each target add a dependency on the target with any additional dependencies.
159			for _, target := range dependency.targets {
160				// Get the variations for the target.
161				variations := target.Variations()
162
163				// Add any additional dependencies needed.
164				variations = append(variations, dependency.imageVariations...)
165
166				if mt.SupportedLinkageNames == nil {
167					// No link types are supported so add a dependency directly.
168					ctx.AddFarVariationDependencies(variations, dependencyTag, name)
169				} else {
170					// Otherwise, add a dependency on each supported link type in turn.
171					for _, linkType := range mt.SupportedLinkageNames {
172						libVariations := append(variations,
173							blueprint.Variation{Mutator: "link", Variation: linkType})
174						// If this is for the device and a shared link type then add a dependency onto the
175						// appropriate version specific variant of the module.
176						if ctx.Device() && linkType == "shared" {
177							libVariations = append(libVariations,
178								blueprint.Variation{Mutator: "version", Variation: version})
179						}
180						ctx.AddFarVariationDependencies(libVariations, dependencyTag, name)
181					}
182				}
183			}
184		}
185	}
186}
187
188func (mt *librarySdkMemberType) IsInstance(module android.Module) bool {
189	// Check the module to see if it can be used with this module type.
190	if m, ok := module.(*Module); ok {
191		for _, allowableMemberType := range m.sdkMemberTypes {
192			if allowableMemberType == mt {
193				return true
194			}
195		}
196	}
197
198	return false
199}
200
201func (mt *librarySdkMemberType) AddPrebuiltModule(ctx android.SdkMemberContext, member android.SdkMember) android.BpModule {
202	pbm := ctx.SnapshotBuilder().AddPrebuiltModule(member, mt.prebuiltModuleType)
203
204	ccModule := member.Variants()[0].(*Module)
205
206	if ctx.RequiresTrait(nativeBridgeSdkTrait) {
207		pbm.AddProperty("native_bridge_supported", true)
208	}
209
210	if ctx.RequiresTrait(ramdiskImageRequiredSdkTrait) {
211		pbm.AddProperty("ramdisk_available", true)
212	}
213
214	if ctx.RequiresTrait(recoveryImageRequiredSdkTrait) {
215		pbm.AddProperty("recovery_available", true)
216	}
217
218	if proptools.Bool(ccModule.VendorProperties.Vendor_available) {
219		pbm.AddProperty("vendor_available", true)
220	}
221
222	if proptools.Bool(ccModule.VendorProperties.Odm_available) {
223		pbm.AddProperty("odm_available", true)
224	}
225
226	if proptools.Bool(ccModule.VendorProperties.Product_available) {
227		pbm.AddProperty("product_available", true)
228	}
229
230	sdkVersion := ccModule.SdkVersion()
231	if sdkVersion != "" {
232		pbm.AddProperty("sdk_version", sdkVersion)
233	}
234
235	stl := ccModule.stl.Properties.Stl
236	if stl != nil {
237		pbm.AddProperty("stl", proptools.String(stl))
238	}
239
240	if lib, ok := ccModule.linker.(*libraryDecorator); ok {
241		uhs := lib.Properties.Unique_host_soname
242		if uhs != nil {
243			pbm.AddProperty("unique_host_soname", proptools.Bool(uhs))
244		}
245	}
246
247	return pbm
248}
249
250func (mt *librarySdkMemberType) CreateVariantPropertiesStruct() android.SdkMemberProperties {
251	return &nativeLibInfoProperties{memberType: mt}
252}
253
254func isBazelOutDirectory(p android.Path) bool {
255	_, bazel := p.(android.BazelOutPath)
256	return bazel
257}
258
259func isGeneratedHeaderDirectory(p android.Path) bool {
260	_, gen := p.(android.WritablePath)
261	// TODO(b/183213331): Here we assume that bazel-based headers are not generated; we need
262	// to support generated headers in mixed builds.
263	return gen && !isBazelOutDirectory(p)
264}
265
266type includeDirsProperty struct {
267	// Accessor to retrieve the paths
268	pathsGetter func(libInfo *nativeLibInfoProperties) android.Paths
269
270	// The name of the property in the prebuilt library, "" means there is no property.
271	propertyName string
272
273	// The directory within the snapshot directory into which items should be copied.
274	snapshotDir string
275
276	// True if the items on the path should be copied.
277	copy bool
278
279	// True if the paths represent directories, files if they represent files.
280	dirs bool
281}
282
283var includeDirProperties = []includeDirsProperty{
284	{
285		// ExportedIncludeDirs lists directories that contains some header files to be
286		// copied into a directory in the snapshot. The snapshot directories must be added to
287		// the export_include_dirs property in the prebuilt module in the snapshot.
288		pathsGetter:  func(libInfo *nativeLibInfoProperties) android.Paths { return libInfo.ExportedIncludeDirs },
289		propertyName: "export_include_dirs",
290		snapshotDir:  nativeIncludeDir,
291		copy:         true,
292		dirs:         true,
293	},
294	{
295		// ExportedSystemIncludeDirs lists directories that contains some system header files to
296		// be copied into a directory in the snapshot. The snapshot directories must be added to
297		// the export_system_include_dirs property in the prebuilt module in the snapshot.
298		pathsGetter:  func(libInfo *nativeLibInfoProperties) android.Paths { return libInfo.ExportedSystemIncludeDirs },
299		propertyName: "export_system_include_dirs",
300		snapshotDir:  nativeIncludeDir,
301		copy:         true,
302		dirs:         true,
303	},
304	{
305		// ExportedGeneratedIncludeDirs lists directories that contains some header files
306		// that are explicitly listed in the ExportedGeneratedHeaders property. So, the contents
307		// of these directories do not need to be copied, but these directories do need adding to
308		// the export_include_dirs property in the prebuilt module in the snapshot.
309		pathsGetter:  func(libInfo *nativeLibInfoProperties) android.Paths { return libInfo.ExportedGeneratedIncludeDirs },
310		propertyName: "export_include_dirs",
311		snapshotDir:  nativeGeneratedIncludeDir,
312		copy:         false,
313		dirs:         true,
314	},
315	{
316		// ExportedGeneratedHeaders lists header files that are in one of the directories
317		// specified in ExportedGeneratedIncludeDirs must be copied into the snapshot.
318		// As they are in a directory in ExportedGeneratedIncludeDirs they do not need adding to a
319		// property in the prebuilt module in the snapshot.
320		pathsGetter:  func(libInfo *nativeLibInfoProperties) android.Paths { return libInfo.ExportedGeneratedHeaders },
321		propertyName: "",
322		snapshotDir:  nativeGeneratedIncludeDir,
323		copy:         true,
324		dirs:         false,
325	},
326}
327
328// Add properties that may, or may not, be arch specific.
329func addPossiblyArchSpecificProperties(sdkModuleContext android.ModuleContext, builder android.SnapshotBuilder, libInfo *nativeLibInfoProperties, outputProperties android.BpPropertySet) {
330
331	outputProperties.AddProperty("sanitize", &libInfo.Sanitize)
332
333	// Copy the generated library to the snapshot and add a reference to it in the .bp module.
334	if libInfo.outputFile != nil {
335		nativeLibraryPath := nativeLibraryPathFor(libInfo)
336		builder.CopyToSnapshot(libInfo.outputFile, nativeLibraryPath)
337		outputProperties.AddProperty("srcs", []string{nativeLibraryPath})
338	}
339
340	if len(libInfo.SharedLibs) > 0 {
341		outputProperties.AddPropertyWithTag("shared_libs", libInfo.SharedLibs, builder.SdkMemberReferencePropertyTag(false))
342	}
343
344	// SystemSharedLibs needs to be propagated if it's a list, even if it's empty,
345	// so check for non-nil instead of nonzero length.
346	if libInfo.SystemSharedLibs != nil {
347		outputProperties.AddPropertyWithTag("system_shared_libs", libInfo.SystemSharedLibs, builder.SdkMemberReferencePropertyTag(false))
348	}
349
350	// Map from property name to the include dirs to add to the prebuilt module in the snapshot.
351	includeDirs := make(map[string][]string)
352
353	// Iterate over each include directory property, copying files and collating property
354	// values where necessary.
355	for _, propertyInfo := range includeDirProperties {
356		// Calculate the base directory in the snapshot into which the files will be copied.
357		// lib.archSubDir is "" for common properties.
358		targetDir := filepath.Join(libInfo.OsPrefix(), libInfo.archSubDir, propertyInfo.snapshotDir)
359
360		propertyName := propertyInfo.propertyName
361
362		// Iterate over each path in one of the include directory properties.
363		for _, path := range propertyInfo.pathsGetter(libInfo) {
364			inputPath := path.String()
365
366			// Map the input path to a snapshot relative path. The mapping is independent of the module
367			// that references them so that if multiple modules within the same snapshot export the same
368			// header files they end up in the same place in the snapshot and so do not get duplicated.
369			targetRelativePath := inputPath
370			if isGeneratedHeaderDirectory(path) {
371				// Remove everything up to the .intermediates/ from the generated output directory to
372				// leave a module relative path.
373				base := android.PathForIntermediates(sdkModuleContext, "")
374				targetRelativePath = android.Rel(sdkModuleContext, base.String(), inputPath)
375			}
376
377			snapshotRelativePath := filepath.Join(targetDir, targetRelativePath)
378
379			// Copy the files/directories when necessary.
380			if propertyInfo.copy {
381				if propertyInfo.dirs {
382					// When copying a directory glob and copy all the headers within it.
383					// TODO(jiyong) copy headers having other suffixes
384					headers, _ := sdkModuleContext.GlobWithDeps(inputPath+"/**/*.h", nil)
385					for _, file := range headers {
386						src := android.PathForSource(sdkModuleContext, file)
387
388						// The destination path in the snapshot is constructed from the snapshot relative path
389						// of the input directory and the input directory relative path of the header file.
390						inputRelativePath := android.Rel(sdkModuleContext, inputPath, file)
391						dest := filepath.Join(snapshotRelativePath, inputRelativePath)
392						builder.CopyToSnapshot(src, dest)
393					}
394				} else {
395					// Otherwise, just copy the file to its snapshot relative path.
396					builder.CopyToSnapshot(path, snapshotRelativePath)
397				}
398			}
399
400			// Only directories are added to a property.
401			if propertyInfo.dirs {
402				includeDirs[propertyName] = append(includeDirs[propertyName], snapshotRelativePath)
403			}
404		}
405	}
406
407	// Add the collated include dir properties to the output.
408	for _, property := range android.SortedKeys(includeDirs) {
409		outputProperties.AddProperty(property, includeDirs[property])
410	}
411
412	if len(libInfo.StubsVersions) > 0 {
413		stubsSet := outputProperties.AddPropertySet("stubs")
414		stubsSet.AddProperty("versions", libInfo.StubsVersions)
415	}
416}
417
418const (
419	nativeIncludeDir          = "include"
420	nativeGeneratedIncludeDir = "include_gen"
421	nativeStubDir             = "lib"
422)
423
424// path to the native library. Relative to <sdk_root>/<api_dir>
425func nativeLibraryPathFor(lib *nativeLibInfoProperties) string {
426	return filepath.Join(lib.OsPrefix(), lib.archSubDir,
427		nativeStubDir, lib.outputFile.Base())
428}
429
430// nativeLibInfoProperties represents properties of a native lib
431//
432// The exported (capitalized) fields will be examined and may be changed during common value extraction.
433// The unexported fields will be left untouched.
434type nativeLibInfoProperties struct {
435	android.SdkMemberPropertiesBase
436
437	memberType *librarySdkMemberType
438
439	// archSubDir is the subdirectory within the OS directory in the sdk snapshot into which arch
440	// specific files will be copied.
441	//
442	// It is not exported since any value other than "" is always going to be arch specific.
443	// This is "" for non-arch specific common properties.
444	archSubDir string
445
446	// The list of possibly common exported include dirs.
447	//
448	// This field is exported as its contents may not be arch specific.
449	ExportedIncludeDirs android.Paths `android:"arch_variant"`
450
451	// The list of arch specific exported generated include dirs.
452	//
453	// This field is exported as its contents may not be arch specific, e.g. protos.
454	ExportedGeneratedIncludeDirs android.Paths `android:"arch_variant"`
455
456	// The list of arch specific exported generated header files.
457	//
458	// This field is exported as its contents may not be arch specific, e.g. protos.
459	ExportedGeneratedHeaders android.Paths `android:"arch_variant"`
460
461	// The list of possibly common exported system include dirs.
462	//
463	// This field is exported as its contents may not be arch specific.
464	ExportedSystemIncludeDirs android.Paths `android:"arch_variant"`
465
466	// The list of possibly common exported flags.
467	//
468	// This field is exported as its contents may not be arch specific.
469	ExportedFlags []string `android:"arch_variant"`
470
471	// The set of shared libraries
472	//
473	// This field is exported as its contents may not be arch specific.
474	SharedLibs []string `android:"arch_variant"`
475
476	// The set of system shared libraries. Note nil and [] are semantically
477	// distinct - see BaseLinkerProperties.System_shared_libs.
478	//
479	// This field is exported as its contents may not be arch specific.
480	SystemSharedLibs []string `android:"arch_variant"`
481
482	// The specific stubs version for the lib variant, or empty string if stubs
483	// are not in use.
484	//
485	// Marked 'ignored-on-host' as the AllStubsVersions() from which this is
486	// initialized is not set on host and the stubs.versions property which this
487	// is written to does not vary by arch so cannot be android specific.
488	StubsVersions []string `sdk:"ignored-on-host"`
489
490	// Value of SanitizeProperties.Sanitize. Several - but not all - of these
491	// affect the expanded variants. All are propagated to avoid entangling the
492	// sanitizer logic with the snapshot generation.
493	Sanitize SanitizeUserProps `android:"arch_variant"`
494
495	// outputFile is not exported as it is always arch specific.
496	outputFile android.Path
497}
498
499func (p *nativeLibInfoProperties) PopulateFromVariant(ctx android.SdkMemberContext, variant android.Module) {
500	addOutputFile := true
501	ccModule := variant.(*Module)
502
503	if s := ccModule.sanitize; s != nil {
504		// We currently do not capture sanitizer flags for libs with sanitizers
505		// enabled, because they may vary among variants that cannot be represented
506		// in the input blueprint files. In particular, sanitizerDepsMutator enables
507		// various sanitizers on dependencies, but in many cases only on static
508		// ones, and we cannot specify sanitizer flags at the link type level (i.e.
509		// in StaticOrSharedProperties).
510		if s.isUnsanitizedVariant() {
511			// This still captures explicitly disabled sanitizers, which may be
512			// necessary to avoid cyclic dependencies.
513			p.Sanitize = s.Properties.Sanitize
514		} else {
515			// Do not add the output file to the snapshot if we don't represent it
516			// properly.
517			addOutputFile = false
518		}
519	}
520
521	exportedInfo := ctx.SdkModuleContext().OtherModuleProvider(variant, FlagExporterInfoProvider).(FlagExporterInfo)
522
523	// Separate out the generated include dirs (which are arch specific) from the
524	// include dirs (which may not be).
525	exportedIncludeDirs, exportedGeneratedIncludeDirs := android.FilterPathListPredicate(
526		exportedInfo.IncludeDirs, isGeneratedHeaderDirectory)
527
528	target := ccModule.Target()
529	p.archSubDir = target.Arch.ArchType.String()
530	if target.NativeBridge == android.NativeBridgeEnabled {
531		p.archSubDir += "_native_bridge"
532	}
533
534	// Make sure that the include directories are unique.
535	p.ExportedIncludeDirs = android.FirstUniquePaths(exportedIncludeDirs)
536	p.ExportedGeneratedIncludeDirs = android.FirstUniquePaths(exportedGeneratedIncludeDirs)
537
538	// Take a copy before filtering out duplicates to avoid changing the slice owned by the
539	// ccModule.
540	dirs := append(android.Paths(nil), exportedInfo.SystemIncludeDirs...)
541	p.ExportedSystemIncludeDirs = android.FirstUniquePaths(dirs)
542
543	p.ExportedFlags = exportedInfo.Flags
544	if ccModule.linker != nil {
545		specifiedDeps := specifiedDeps{}
546		specifiedDeps = ccModule.linker.linkerSpecifiedDeps(specifiedDeps)
547
548		if lib := ccModule.library; lib != nil {
549			if !lib.hasStubsVariants() {
550				// Propagate dynamic dependencies for implementation libs, but not stubs.
551				p.SharedLibs = specifiedDeps.sharedLibs
552			} else {
553				// TODO(b/169373910): 1. Only output the specific version (from
554				// ccModule.StubsVersion()) if the module is versioned. 2. Ensure that all
555				// the versioned stub libs are retained in the prebuilt tree; currently only
556				// the stub corresponding to ccModule.StubsVersion() is.
557				p.StubsVersions = lib.allStubsVersions()
558			}
559		}
560		p.SystemSharedLibs = specifiedDeps.systemSharedLibs
561	}
562	p.ExportedGeneratedHeaders = exportedInfo.GeneratedHeaders
563
564	if !p.memberType.noOutputFiles && addOutputFile {
565		p.outputFile = getRequiredMemberOutputFile(ctx, ccModule)
566	}
567}
568
569func getRequiredMemberOutputFile(ctx android.SdkMemberContext, ccModule *Module) android.Path {
570	var path android.Path
571	outputFile := ccModule.OutputFile()
572	if outputFile.Valid() {
573		path = outputFile.Path()
574	} else {
575		ctx.SdkModuleContext().ModuleErrorf("member variant %s does not have a valid output file", ccModule)
576	}
577	return path
578}
579
580func (p *nativeLibInfoProperties) AddToPropertySet(ctx android.SdkMemberContext, propertySet android.BpPropertySet) {
581	addPossiblyArchSpecificProperties(ctx.SdkModuleContext(), ctx.SnapshotBuilder(), p, propertySet)
582}
583