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