Ivan Lozano | 6cd99e6 | 2020-02-11 08:24:25 -0500 | [diff] [blame] | 1 | // Copyright 2020 The Android Open Source Project |
| 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 | |
| 15 | package rust |
| 16 | |
| 17 | import ( |
hamzeh | c651b52 | 2021-04-29 12:50:47 -0700 | [diff] [blame] | 18 | "path/filepath" |
| 19 | "sort" |
| 20 | "strings" |
| 21 | |
Ivan Lozano | 6cd99e6 | 2020-02-11 08:24:25 -0500 | [diff] [blame] | 22 | "android/soong/android" |
| 23 | "android/soong/cc" |
hamzeh | c0a671f | 2021-07-22 12:05:08 -0700 | [diff] [blame] | 24 | "android/soong/fuzz" |
Ivan Lozano | 6cd99e6 | 2020-02-11 08:24:25 -0500 | [diff] [blame] | 25 | "android/soong/rust/config" |
| 26 | ) |
| 27 | |
| 28 | func init() { |
| 29 | android.RegisterModuleType("rust_fuzz", RustFuzzFactory) |
hamzeh | c651b52 | 2021-04-29 12:50:47 -0700 | [diff] [blame] | 30 | android.RegisterSingletonType("rust_fuzz_packaging", rustFuzzPackagingFactory) |
Ivan Lozano | 6cd99e6 | 2020-02-11 08:24:25 -0500 | [diff] [blame] | 31 | } |
| 32 | |
| 33 | type fuzzDecorator struct { |
| 34 | *binaryDecorator |
| 35 | |
hamzeh | c0a671f | 2021-07-22 12:05:08 -0700 | [diff] [blame] | 36 | fuzzPackagedModule fuzz.FuzzPackagedModule |
Ivan Lozano | 6cd99e6 | 2020-02-11 08:24:25 -0500 | [diff] [blame] | 37 | } |
| 38 | |
Ivan Lozano | 5482d6a | 2021-11-01 10:13:25 -0400 | [diff] [blame] | 39 | var _ compiler = (*fuzzDecorator)(nil) |
Ivan Lozano | 6cd99e6 | 2020-02-11 08:24:25 -0500 | [diff] [blame] | 40 | |
| 41 | // rust_binary produces a binary that is runnable on a device. |
| 42 | func RustFuzzFactory() android.Module { |
| 43 | module, _ := NewRustFuzz(android.HostAndDeviceSupported) |
| 44 | return module.Init() |
| 45 | } |
| 46 | |
| 47 | func NewRustFuzz(hod android.HostOrDeviceSupported) (*Module, *fuzzDecorator) { |
| 48 | module, binary := NewRustBinary(hod) |
| 49 | fuzz := &fuzzDecorator{ |
| 50 | binaryDecorator: binary, |
| 51 | } |
| 52 | |
| 53 | // Change the defaults for the binaryDecorator's baseCompiler |
| 54 | fuzz.binaryDecorator.baseCompiler.dir = "fuzz" |
| 55 | fuzz.binaryDecorator.baseCompiler.dir64 = "fuzz" |
| 56 | fuzz.binaryDecorator.baseCompiler.location = InstallInData |
| 57 | module.sanitize.SetSanitizer(cc.Fuzzer, true) |
| 58 | module.compiler = fuzz |
| 59 | return module, fuzz |
| 60 | } |
| 61 | |
| 62 | func (fuzzer *fuzzDecorator) compilerFlags(ctx ModuleContext, flags Flags) Flags { |
| 63 | flags = fuzzer.binaryDecorator.compilerFlags(ctx, flags) |
| 64 | |
| 65 | // `../lib` for installed fuzz targets (both host and device), and `./lib` for fuzz target packages. |
| 66 | flags.LinkFlags = append(flags.LinkFlags, `-Wl,-rpath,\$$ORIGIN/../lib`) |
| 67 | flags.LinkFlags = append(flags.LinkFlags, `-Wl,-rpath,\$$ORIGIN/lib`) |
| 68 | |
| 69 | return flags |
| 70 | } |
| 71 | |
| 72 | func (fuzzer *fuzzDecorator) compilerDeps(ctx DepsContext, deps Deps) Deps { |
Liz Kammer | 9c21086 | 2021-04-12 18:52:29 -0400 | [diff] [blame] | 73 | if libFuzzerRuntimeLibrary := config.LibFuzzerRuntimeLibrary(ctx.toolchain()); libFuzzerRuntimeLibrary != "" { |
| 74 | deps.StaticLibs = append(deps.StaticLibs, libFuzzerRuntimeLibrary) |
| 75 | } |
Ivan Lozano | 6cd99e6 | 2020-02-11 08:24:25 -0500 | [diff] [blame] | 76 | deps.SharedLibs = append(deps.SharedLibs, "libc++") |
| 77 | deps.Rlibs = append(deps.Rlibs, "liblibfuzzer_sys") |
| 78 | |
| 79 | deps = fuzzer.binaryDecorator.compilerDeps(ctx, deps) |
| 80 | |
| 81 | return deps |
| 82 | } |
| 83 | |
| 84 | func (fuzzer *fuzzDecorator) compilerProps() []interface{} { |
| 85 | return append(fuzzer.binaryDecorator.compilerProps(), |
hamzeh | 41ad881 | 2021-07-07 14:00:07 -0700 | [diff] [blame] | 86 | &fuzzer.fuzzPackagedModule.FuzzProperties) |
Ivan Lozano | 6cd99e6 | 2020-02-11 08:24:25 -0500 | [diff] [blame] | 87 | } |
| 88 | |
| 89 | func (fuzzer *fuzzDecorator) stdLinkage(ctx *depsContext) RustLinkage { |
| 90 | return RlibLinkage |
| 91 | } |
| 92 | |
Liz Kammer | 356f7d4 | 2021-01-26 09:18:53 -0500 | [diff] [blame] | 93 | func (fuzzer *fuzzDecorator) autoDep(ctx android.BottomUpMutatorContext) autoDep { |
Ivan Lozano | 6cd99e6 | 2020-02-11 08:24:25 -0500 | [diff] [blame] | 94 | return rlibAutoDep |
| 95 | } |
hamzeh | c651b52 | 2021-04-29 12:50:47 -0700 | [diff] [blame] | 96 | |
| 97 | // Responsible for generating GNU Make rules that package fuzz targets into |
| 98 | // their architecture & target/host specific zip file. |
| 99 | type rustFuzzPackager struct { |
hamzeh | c0a671f | 2021-07-22 12:05:08 -0700 | [diff] [blame] | 100 | fuzz.FuzzPackager |
hamzeh | c651b52 | 2021-04-29 12:50:47 -0700 | [diff] [blame] | 101 | } |
| 102 | |
| 103 | func rustFuzzPackagingFactory() android.Singleton { |
| 104 | return &rustFuzzPackager{} |
| 105 | } |
| 106 | |
hamzeh | c651b52 | 2021-04-29 12:50:47 -0700 | [diff] [blame] | 107 | func (s *rustFuzzPackager) GenerateBuildActions(ctx android.SingletonContext) { |
hamzeh | c651b52 | 2021-04-29 12:50:47 -0700 | [diff] [blame] | 108 | // Map between each architecture + host/device combination. |
hamzeh | c0a671f | 2021-07-22 12:05:08 -0700 | [diff] [blame] | 109 | archDirs := make(map[fuzz.ArchOs][]fuzz.FileToZip) |
hamzeh | c651b52 | 2021-04-29 12:50:47 -0700 | [diff] [blame] | 110 | |
| 111 | // List of individual fuzz targets. |
hamzeh | 41ad881 | 2021-07-07 14:00:07 -0700 | [diff] [blame] | 112 | s.FuzzTargets = make(map[string]bool) |
hamzeh | c651b52 | 2021-04-29 12:50:47 -0700 | [diff] [blame] | 113 | |
Ivan Lozano | 39b0bf0 | 2021-10-14 12:22:09 -0400 | [diff] [blame] | 114 | // Map tracking whether each shared library has an install rule to avoid duplicate install rules from |
| 115 | // multiple fuzzers that depend on the same shared library. |
| 116 | sharedLibraryInstalled := make(map[string]bool) |
| 117 | |
hamzeh | c651b52 | 2021-04-29 12:50:47 -0700 | [diff] [blame] | 118 | ctx.VisitAllModules(func(module android.Module) { |
| 119 | // Discard non-fuzz targets. |
| 120 | rustModule, ok := module.(*Module) |
| 121 | if !ok { |
| 122 | return |
| 123 | } |
| 124 | |
hamzeh | c0a671f | 2021-07-22 12:05:08 -0700 | [diff] [blame] | 125 | if ok := fuzz.IsValid(rustModule.FuzzModule); !ok || rustModule.Properties.PreventInstall { |
hamzeh | 41ad881 | 2021-07-07 14:00:07 -0700 | [diff] [blame] | 126 | return |
| 127 | } |
| 128 | |
hamzeh | c651b52 | 2021-04-29 12:50:47 -0700 | [diff] [blame] | 129 | fuzzModule, ok := rustModule.compiler.(*fuzzDecorator) |
| 130 | if !ok { |
| 131 | return |
| 132 | } |
| 133 | |
hamzeh | c651b52 | 2021-04-29 12:50:47 -0700 | [diff] [blame] | 134 | hostOrTargetString := "target" |
| 135 | if rustModule.Host() { |
| 136 | hostOrTargetString = "host" |
| 137 | } |
| 138 | |
| 139 | archString := rustModule.Arch().ArchType.String() |
| 140 | archDir := android.PathForIntermediates(ctx, "fuzz", hostOrTargetString, archString) |
hamzeh | c0a671f | 2021-07-22 12:05:08 -0700 | [diff] [blame] | 141 | archOs := fuzz.ArchOs{HostOrTarget: hostOrTargetString, Arch: archString, Dir: archDir.String()} |
hamzeh | c651b52 | 2021-04-29 12:50:47 -0700 | [diff] [blame] | 142 | |
hamzeh | c0a671f | 2021-07-22 12:05:08 -0700 | [diff] [blame] | 143 | var files []fuzz.FileToZip |
hamzeh | c651b52 | 2021-04-29 12:50:47 -0700 | [diff] [blame] | 144 | builder := android.NewRuleBuilder(pctx, ctx) |
| 145 | |
hamzeh | 41ad881 | 2021-07-07 14:00:07 -0700 | [diff] [blame] | 146 | // Package the artifacts (data, corpus, config and dictionary into a zipfile. |
| 147 | files = s.PackageArtifacts(ctx, module, fuzzModule.fuzzPackagedModule, archDir, builder) |
hamzeh | c651b52 | 2021-04-29 12:50:47 -0700 | [diff] [blame] | 148 | |
| 149 | // The executable. |
Ivan Lozano | 8d10fc3 | 2021-11-05 16:36:47 -0400 | [diff] [blame] | 150 | files = append(files, fuzz.FileToZip{rustModule.UnstrippedOutputFile(), ""}) |
hamzeh | c651b52 | 2021-04-29 12:50:47 -0700 | [diff] [blame] | 151 | |
Ivan Lozano | 39b0bf0 | 2021-10-14 12:22:09 -0400 | [diff] [blame] | 152 | // Grab the list of required shared libraries. |
| 153 | sharedLibraries := fuzz.CollectAllSharedDependencies(ctx, module, cc.UnstrippedOutputFile, cc.IsValidSharedDependency) |
| 154 | |
| 155 | // Package shared libraries |
| 156 | files = append(files, cc.GetSharedLibsToZip(sharedLibraries, rustModule, &s.FuzzPackager, archString, &sharedLibraryInstalled)...) |
| 157 | |
hamzeh | 41ad881 | 2021-07-07 14:00:07 -0700 | [diff] [blame] | 158 | archDirs[archOs], ok = s.BuildZipFile(ctx, module, fuzzModule.fuzzPackagedModule, files, builder, archDir, archString, hostOrTargetString, archOs, archDirs) |
| 159 | if !ok { |
| 160 | return |
hamzeh | c651b52 | 2021-04-29 12:50:47 -0700 | [diff] [blame] | 161 | } |
| 162 | |
hamzeh | c651b52 | 2021-04-29 12:50:47 -0700 | [diff] [blame] | 163 | }) |
hamzeh | c0a671f | 2021-07-22 12:05:08 -0700 | [diff] [blame] | 164 | s.CreateFuzzPackage(ctx, archDirs, fuzz.Rust, pctx) |
hamzeh | c651b52 | 2021-04-29 12:50:47 -0700 | [diff] [blame] | 165 | } |
| 166 | |
| 167 | func (s *rustFuzzPackager) MakeVars(ctx android.MakeVarsContext) { |
hamzeh | 41ad881 | 2021-07-07 14:00:07 -0700 | [diff] [blame] | 168 | packages := s.Packages.Strings() |
hamzeh | c651b52 | 2021-04-29 12:50:47 -0700 | [diff] [blame] | 169 | sort.Strings(packages) |
| 170 | |
| 171 | ctx.Strict("SOONG_RUST_FUZZ_PACKAGING_ARCH_MODULES", strings.Join(packages, " ")) |
| 172 | |
hamzeh | 41ad881 | 2021-07-07 14:00:07 -0700 | [diff] [blame] | 173 | // Preallocate the slice of fuzz targets to minimize memory allocations. |
| 174 | s.PreallocateSlice(ctx, "ALL_RUST_FUZZ_TARGETS") |
hamzeh | c651b52 | 2021-04-29 12:50:47 -0700 | [diff] [blame] | 175 | } |
| 176 | |
| 177 | func (fuzz *fuzzDecorator) install(ctx ModuleContext) { |
| 178 | fuzz.binaryDecorator.baseCompiler.dir = filepath.Join( |
| 179 | "fuzz", ctx.Target().Arch.ArchType.String(), ctx.ModuleName()) |
| 180 | fuzz.binaryDecorator.baseCompiler.dir64 = filepath.Join( |
| 181 | "fuzz", ctx.Target().Arch.ArchType.String(), ctx.ModuleName()) |
| 182 | fuzz.binaryDecorator.baseCompiler.install(ctx) |
| 183 | |
hamzeh | 41ad881 | 2021-07-07 14:00:07 -0700 | [diff] [blame] | 184 | if fuzz.fuzzPackagedModule.FuzzProperties.Corpus != nil { |
| 185 | fuzz.fuzzPackagedModule.Corpus = android.PathsForModuleSrc(ctx, fuzz.fuzzPackagedModule.FuzzProperties.Corpus) |
hamzeh | c651b52 | 2021-04-29 12:50:47 -0700 | [diff] [blame] | 186 | } |
hamzeh | 41ad881 | 2021-07-07 14:00:07 -0700 | [diff] [blame] | 187 | if fuzz.fuzzPackagedModule.FuzzProperties.Data != nil { |
| 188 | fuzz.fuzzPackagedModule.Data = android.PathsForModuleSrc(ctx, fuzz.fuzzPackagedModule.FuzzProperties.Data) |
hamzeh | c651b52 | 2021-04-29 12:50:47 -0700 | [diff] [blame] | 189 | } |
hamzeh | 41ad881 | 2021-07-07 14:00:07 -0700 | [diff] [blame] | 190 | if fuzz.fuzzPackagedModule.FuzzProperties.Dictionary != nil { |
| 191 | fuzz.fuzzPackagedModule.Dictionary = android.PathForModuleSrc(ctx, *fuzz.fuzzPackagedModule.FuzzProperties.Dictionary) |
hamzeh | c651b52 | 2021-04-29 12:50:47 -0700 | [diff] [blame] | 192 | } |
| 193 | } |