| // Copyright 2021 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 rust | 
 |  | 
 | import ( | 
 | 	"fmt" | 
 |  | 
 | 	"android/soong/android" | 
 | 	"android/soong/cc" | 
 |  | 
 | 	"github.com/google/blueprint/proptools" | 
 | ) | 
 |  | 
 | type snapshotLibraryDecorator struct { | 
 | 	cc.BaseSnapshotDecorator | 
 | 	*libraryDecorator | 
 | 	properties          cc.SnapshotLibraryProperties | 
 | 	sanitizerProperties struct { | 
 | 		SanitizerVariation cc.SanitizerType `blueprint:"mutated"` | 
 |  | 
 | 		//TODO: Library flags for cfi variant when CFI is supported. | 
 | 		//Cfi cc.SnapshotLibraryProperties `android:"arch_variant"` | 
 |  | 
 | 		// Library flags for hwasan variant. | 
 | 		Hwasan cc.SnapshotLibraryProperties `android:"arch_variant"` | 
 | 	} | 
 | } | 
 |  | 
 | var _ cc.SnapshotSanitizer = (*snapshotLibraryDecorator)(nil) | 
 |  | 
 | func (library *snapshotLibraryDecorator) IsSanitizerAvailable(t cc.SanitizerType) bool { | 
 | 	switch t { | 
 | 	//TODO: When CFI is supported, add a check here as well | 
 | 	case cc.Hwasan: | 
 | 		return library.sanitizerProperties.Hwasan.Src != nil | 
 | 	default: | 
 | 		return false | 
 | 	} | 
 | } | 
 |  | 
 | func (library *snapshotLibraryDecorator) SetSanitizerVariation(t cc.SanitizerType, enabled bool) { | 
 | 	if !enabled || library.IsSanitizerEnabled(t) { | 
 | 		return | 
 | 	} | 
 | 	if !library.IsUnsanitizedVariant() { | 
 | 		panic(fmt.Errorf("snapshot Sanitizer must be one of Cfi or Hwasan but not both")) | 
 | 	} | 
 | 	library.sanitizerProperties.SanitizerVariation = t | 
 | } | 
 |  | 
 | func (library *snapshotLibraryDecorator) IsSanitizerEnabled(t cc.SanitizerType) bool { | 
 | 	return library.sanitizerProperties.SanitizerVariation == t | 
 | } | 
 |  | 
 | func (library *snapshotLibraryDecorator) IsUnsanitizedVariant() bool { | 
 | 	//TODO: When CFI is supported, add a check here as well | 
 | 	return !library.IsSanitizerEnabled(cc.Hwasan) | 
 | } | 
 |  | 
 | func init() { | 
 | 	registerRustSnapshotModules(android.InitRegistrationContext) | 
 | } | 
 |  | 
 | func (mod *Module) IsSnapshotSanitizerAvailable(t cc.SanitizerType) bool { | 
 | 	if ss, ok := mod.compiler.(cc.SnapshotSanitizer); ok { | 
 | 		return ss.IsSanitizerAvailable(t) | 
 | 	} | 
 | 	return false | 
 | } | 
 |  | 
 | func (mod *Module) SetSnapshotSanitizerVariation(t cc.SanitizerType, enabled bool) { | 
 | 	if ss, ok := mod.compiler.(cc.SnapshotSanitizer); ok { | 
 | 		ss.SetSanitizerVariation(t, enabled) | 
 | 	} else { | 
 | 		panic(fmt.Errorf("Calling SetSnapshotSanitizerVariation on a non-snapshotLibraryDecorator: %s", mod.Name())) | 
 | 	} | 
 | } | 
 |  | 
 | func (mod *Module) IsSnapshotUnsanitizedVariant() bool { | 
 | 	if ss, ok := mod.compiler.(cc.SnapshotSanitizer); ok { | 
 | 		return ss.IsUnsanitizedVariant() | 
 | 	} | 
 | 	return false | 
 | } | 
 |  | 
 | func (mod *Module) IsSnapshotSanitizer() bool { | 
 | 	if _, ok := mod.compiler.(cc.SnapshotSanitizer); ok { | 
 | 		return true | 
 | 	} | 
 | 	return false | 
 | } | 
 |  | 
 | func registerRustSnapshotModules(ctx android.RegistrationContext) { | 
 | 	cc.VendorSnapshotImageSingleton.RegisterAdditionalModule(ctx, | 
 | 		"vendor_snapshot_rlib", VendorSnapshotRlibFactory) | 
 | 	cc.VendorSnapshotImageSingleton.RegisterAdditionalModule(ctx, | 
 | 		"vendor_snapshot_dylib", VendorSnapshotDylibFactory) | 
 | 	cc.RecoverySnapshotImageSingleton.RegisterAdditionalModule(ctx, | 
 | 		"recovery_snapshot_rlib", RecoverySnapshotRlibFactory) | 
 | } | 
 |  | 
 | func snapshotLibraryFactory(image cc.SnapshotImage, moduleSuffix string) (*Module, *snapshotLibraryDecorator) { | 
 | 	module, library := NewRustLibrary(android.DeviceSupported) | 
 |  | 
 | 	module.sanitize = nil | 
 | 	library.stripper.StripProperties.Strip.None = proptools.BoolPtr(true) | 
 |  | 
 | 	prebuilt := &snapshotLibraryDecorator{ | 
 | 		libraryDecorator: library, | 
 | 	} | 
 |  | 
 | 	module.compiler = prebuilt | 
 |  | 
 | 	prebuilt.Init(module, image, moduleSuffix) | 
 | 	module.AddProperties( | 
 | 		&prebuilt.properties, | 
 | 		&prebuilt.sanitizerProperties, | 
 | 	) | 
 |  | 
 | 	return module, prebuilt | 
 | } | 
 |  | 
 | func (library *snapshotLibraryDecorator) compile(ctx ModuleContext, flags Flags, deps PathDeps) buildOutput { | 
 | 	var variant string | 
 | 	if library.static() { | 
 | 		variant = cc.SnapshotStaticSuffix | 
 | 	} else if library.shared() { | 
 | 		variant = cc.SnapshotSharedSuffix | 
 | 	} else if library.rlib() { | 
 | 		variant = cc.SnapshotRlibSuffix | 
 | 	} else if library.dylib() { | 
 | 		variant = cc.SnapshotDylibSuffix | 
 | 	} | 
 |  | 
 | 	library.SetSnapshotAndroidMkSuffix(ctx, variant) | 
 |  | 
 | 	if library.IsSanitizerEnabled(cc.Hwasan) { | 
 | 		library.properties = library.sanitizerProperties.Hwasan | 
 | 	} | 
 | 	if !library.MatchesWithDevice(ctx.DeviceConfig()) { | 
 | 		return buildOutput{} | 
 | 	} | 
 | 	outputFile := android.PathForModuleSrc(ctx, *library.properties.Src) | 
 | 	library.unstrippedOutputFile = outputFile | 
 | 	return buildOutput{outputFile: outputFile} | 
 | } | 
 |  | 
 | func (library *snapshotLibraryDecorator) rustdoc(ctx ModuleContext, flags Flags, deps PathDeps) android.OptionalPath { | 
 | 	return android.OptionalPath{} | 
 | } | 
 |  | 
 | // vendor_snapshot_rlib is a special prebuilt rlib library which is auto-generated by | 
 | // development/vendor_snapshot/update.py. As a part of vendor snapshot, vendor_snapshot_rlib | 
 | // overrides the vendor variant of the rust rlib library with the same name, if BOARD_VNDK_VERSION | 
 | // is set. | 
 | func VendorSnapshotRlibFactory() android.Module { | 
 | 	module, prebuilt := snapshotLibraryFactory(cc.VendorSnapshotImageSingleton, cc.SnapshotRlibSuffix) | 
 | 	prebuilt.libraryDecorator.BuildOnlyRlib() | 
 | 	prebuilt.libraryDecorator.setNoStdlibs() | 
 | 	return module.Init() | 
 | } | 
 |  | 
 | // vendor_snapshot_dylib is a special prebuilt dylib library which is auto-generated by | 
 | // development/vendor_snapshot/update.py. As a part of vendor snapshot, vendor_snapshot_dylib | 
 | // overrides the vendor variant of the rust dylib library with the same name, if BOARD_VNDK_VERSION | 
 | // is set. | 
 | func VendorSnapshotDylibFactory() android.Module { | 
 | 	module, prebuilt := snapshotLibraryFactory(cc.VendorSnapshotImageSingleton, cc.SnapshotDylibSuffix) | 
 | 	prebuilt.libraryDecorator.BuildOnlyDylib() | 
 | 	prebuilt.libraryDecorator.setNoStdlibs() | 
 | 	return module.Init() | 
 | } | 
 |  | 
 | func RecoverySnapshotRlibFactory() android.Module { | 
 | 	module, prebuilt := snapshotLibraryFactory(cc.RecoverySnapshotImageSingleton, cc.SnapshotRlibSuffix) | 
 | 	prebuilt.libraryDecorator.BuildOnlyRlib() | 
 | 	prebuilt.libraryDecorator.setNoStdlibs() | 
 | 	return module.Init() | 
 | } | 
 |  | 
 | func (library *snapshotLibraryDecorator) MatchesWithDevice(config android.DeviceConfig) bool { | 
 | 	arches := config.Arches() | 
 | 	if len(arches) == 0 || arches[0].ArchType.String() != library.Arch() { | 
 | 		return false | 
 | 	} | 
 | 	if library.properties.Src == nil { | 
 | 		return false | 
 | 	} | 
 | 	return true | 
 | } | 
 |  | 
 | func (library *snapshotLibraryDecorator) IsSnapshotPrebuilt() bool { | 
 | 	return true | 
 | } | 
 |  | 
 | var _ cc.SnapshotInterface = (*snapshotLibraryDecorator)(nil) |