| // 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) |