// Copyright 2021 Google Inc. All rights reserved. // // 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 java import ( "path/filepath" "sort" "strings" "android/soong/android" "android/soong/cc" "android/soong/fuzz" "github.com/google/blueprint" "github.com/google/blueprint/proptools" ) const ( hostString = "host" targetString = "target" deviceString = "device" ) // Any shared libs for these deps will also be packaged var artDeps = []string{"libdl_android"} func init() { RegisterJavaFuzzBuildComponents(android.InitRegistrationContext) } func RegisterJavaFuzzBuildComponents(ctx android.RegistrationContext) { ctx.RegisterModuleType("java_fuzz", JavaFuzzFactory) ctx.RegisterParallelSingletonType("java_fuzz_packaging", javaFuzzPackagingFactory) } type JavaFuzzTest struct { Test fuzzPackagedModule fuzz.FuzzPackagedModule jniFilePaths android.Paths } type JavaFuzzTestInfo struct { JniFilePaths android.Paths } var JavaFuzzTestInfoProvider = blueprint.NewProvider[JavaFuzzTestInfo]() // java_fuzz builds and links sources into a `.jar` file for the device. // This generates .class files in a jar which can then be instrumented before // fuzzing in Android Runtime (ART: Android OS on emulator or device) func JavaFuzzFactory() android.Module { module := &JavaFuzzTest{} module.addHostAndDeviceProperties() module.AddProperties(&module.testProperties) module.AddProperties(&module.fuzzPackagedModule.FuzzProperties) module.Module.properties.Installable = proptools.BoolPtr(true) module.Module.dexpreopter.isTest = true module.Module.linter.properties.Lint.Test_module_type = proptools.BoolPtr(true) module.Module.sourceProperties.Test_only = proptools.BoolPtr(true) module.Module.sourceProperties.Top_level_test_target = true android.AddLoadHook(module, func(ctx android.LoadHookContext) { disableLinuxBionic := struct { Target struct { Linux_bionic struct { Enabled *bool } } }{} disableLinuxBionic.Target.Linux_bionic.Enabled = proptools.BoolPtr(false) ctx.AppendProperties(&disableLinuxBionic) }) InitJavaModuleMultiTargets(module, android.HostAndDeviceSupported) return module } func (j *JavaFuzzTest) DepsMutator(ctx android.BottomUpMutatorContext) { if j.Os().Class.String() == deviceString { j.testProperties.Jni_libs.AppendSimpleValue(artDeps) } jniLibs := j.testProperties.Jni_libs.GetOrDefault(ctx, nil) if len(jniLibs) > 0 { if j.fuzzPackagedModule.FuzzProperties.Fuzz_config == nil { config := &fuzz.FuzzConfig{} j.fuzzPackagedModule.FuzzProperties.Fuzz_config = config } // this will be used by the ingestion pipeline to determine the version // of jazzer to add to the fuzzer package j.fuzzPackagedModule.FuzzProperties.Fuzz_config.IsJni = proptools.BoolPtr(true) for _, target := range ctx.MultiTargets() { sharedLibVariations := append(target.Variations(), blueprint.Variation{Mutator: "link", Variation: "shared"}) ctx.AddFarVariationDependencies(sharedLibVariations, jniLibTag, jniLibs...) } } j.deps(ctx) } func (j *JavaFuzzTest) GenerateAndroidBuildActions(ctx android.ModuleContext) { j.fuzzPackagedModule = cc.PackageFuzzModule(ctx, j.fuzzPackagedModule) _, sharedDeps := cc.CollectAllSharedDependencies(ctx) for _, dep := range sharedDeps { sharedLibInfo, _ := android.OtherModuleProvider(ctx, dep, cc.SharedLibraryInfoProvider) if sharedLibInfo.SharedLibrary != nil { arch := "lib" if sharedLibInfo.Target.Arch.ArchType.Multilib == "lib64" { arch = "lib64" } libPath := android.PathForModuleOut(ctx, filepath.Join(arch, sharedLibInfo.SharedLibrary.Base())) ctx.Build(pctx, android.BuildParams{ Rule: android.Cp, Input: sharedLibInfo.SharedLibrary, Output: libPath, }) j.jniFilePaths = append(j.jniFilePaths, libPath) } else { ctx.PropertyErrorf("jni_libs", "%q of type %q is not supported", dep.Name(), ctx.OtherModuleType(dep)) } } checkMinSdkVersionMts(ctx, j.MinSdkVersion(ctx)) j.Test.generateAndroidBuildActionsWithConfig(ctx, nil) var compatibilitySupportFiles android.Paths compatibilitySupportFiles = append(compatibilitySupportFiles, j.implementationJarFile) compatibilitySupportFiles = append(compatibilitySupportFiles, j.jniFilePaths...) compatibilitySupportFiles = append(compatibilitySupportFiles, j.fuzzPackagedModule.Corpus...) if j.fuzzPackagedModule.Dictionary != nil { compatibilitySupportFiles = append(compatibilitySupportFiles, j.fuzzPackagedModule.Dictionary) } ctx.SetTestSuiteInfo(android.TestSuiteInfo{ TestSuites: j.testProperties.Test_suites, MainFile: j.outputFile, MainFileStem: j.Stem(), MainFileExt: ".jar", ConfigFile: j.testConfig, ExtraConfigs: j.extraTestConfigs, NeedsArchFolder: ctx.Device(), CompatibilitySupportFiles: compatibilitySupportFiles, PerTestcaseDirectory: proptools.Bool(j.testProperties.Per_testcase_directory), }) fuzzModuleValidator := fuzz.FuzzModule{ j.ModuleBase, j.DefaultableModuleBase, j.ApexModuleBase, } if fuzz.IsValid(ctx, fuzzModuleValidator) { fuzz.SetFuzzPackagedModuleInfo(ctx, &j.fuzzPackagedModule) android.SetProvider(ctx, JavaFuzzTestInfoProvider, JavaFuzzTestInfo{ JniFilePaths: j.jniFilePaths, }) } } type javaFuzzPackager struct { fuzz.FuzzPackager } func javaFuzzPackagingFactory() android.Singleton { return &javaFuzzPackager{} } func (s *javaFuzzPackager) GenerateBuildActions(ctx android.SingletonContext) { // Map between each architecture + host/device combination. archDirs := make(map[fuzz.ArchOs][]fuzz.FileToZip) s.FuzzTargets = make(map[string]bool) ctx.VisitAllModuleProxies(func(module android.ModuleProxy) { // Discard non-fuzz targets. javaInfo, ok := android.OtherModuleProvider(ctx, module, JavaInfoProvider) if !ok { return } fuzzInfo, ok := android.OtherModuleProvider(ctx, module, fuzz.FuzzPackagedModuleInfoProvider) if !ok { return } javaFuzzTestInfo, ok := android.OtherModuleProvider(ctx, module, JavaFuzzTestInfoProvider) if !ok { return } commonInfo := android.OtherModuleProviderOrDefault(ctx, module, android.CommonModuleInfoProvider) hostOrTargetString := "target" if commonInfo.Target.HostCross { hostOrTargetString = "host_cross" } else if commonInfo.Host { hostOrTargetString = "host" } archString := commonInfo.Target.Arch.ArchType.String() archDir := android.PathForIntermediates(ctx, "fuzz", hostOrTargetString, archString) archOs := fuzz.ArchOs{HostOrTarget: hostOrTargetString, Arch: archString, Dir: archDir.String()} var files []fuzz.FileToZip builder := android.NewRuleBuilder(pctx, ctx) // Package the artifacts (data, corpus, config and dictionary) into a zipfile. files = s.PackageArtifacts(ctx, module, &fuzzInfo, archDir, builder) // Add .jar if !commonInfo.Host { for _, jar := range javaInfo.ImplementationJars { files = append(files, fuzz.FileToZip{SourceFilePath: jar, DestinationPathPrefix: "classes"}) } } outputFile := android.OutputFileForModule(ctx, module, "") files = append(files, fuzz.FileToZip{SourceFilePath: outputFile}) // Add jni .so files for _, fPath := range javaFuzzTestInfo.JniFilePaths { files = append(files, fuzz.FileToZip{SourceFilePath: fPath}) } archDirs[archOs], ok = s.BuildZipFile(ctx, module, &fuzzInfo, files, builder, archDir, archString, hostOrTargetString, archOs, archDirs) if !ok { return } }) s.CreateFuzzPackage(ctx, archDirs, fuzz.Java, pctx) // Create the phony and dist rules ctx.Phony("haiku-java", s.Packages...) ctx.DistForGoals([]string{"haiku-java"}, s.Packages...) for _, target := range android.SortedKeys(s.FuzzTargets) { ctx.Phony("haiku-java", android.PathForPhony(ctx, target)) } } func (s *javaFuzzPackager) MakeVars(ctx android.MakeVarsContext) { packages := s.Packages.Strings() sort.Strings(packages) ctx.Strict("SOONG_JAVA_FUZZ_PACKAGING_ARCH_MODULES", strings.Join(packages, " ")) // Preallocate the slice of fuzz targets to minimize memory allocations. s.PreallocateSlice(ctx, "ALL_JAVA_FUZZ_TARGETS") }