Multiple modules (e.g. java, cc, python, rust) define the `test_options` field. Extract the common properties in test_options to share across different test rules. Bug: 240928948 Test: `refreshmod` and diff with original module-info.json Change-Id: I404a7a157b4ccaa53d800ee2217559ff695bd825
		
			
				
	
	
		
			260 lines
		
	
	
		
			8.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			260 lines
		
	
	
		
			8.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2019 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 (
 | |
| 	"path/filepath"
 | |
| 
 | |
| 	"github.com/google/blueprint/proptools"
 | |
| 
 | |
| 	"android/soong/android"
 | |
| 	"android/soong/cc"
 | |
| 	"android/soong/tradefed"
 | |
| )
 | |
| 
 | |
| type TestProperties struct {
 | |
| 	// Disables the creation of a test-specific directory when used with
 | |
| 	// relative_install_path. Useful if several tests need to be in the same
 | |
| 	// directory, but test_per_src doesn't work.
 | |
| 	No_named_install_directory *bool
 | |
| 
 | |
| 	// the name of the test configuration (for example "AndroidTest.xml") that should be
 | |
| 	// installed with the module.
 | |
| 	Test_config *string `android:"path,arch_variant"`
 | |
| 
 | |
| 	// the name of the test configuration template (for example "AndroidTestTemplate.xml") that
 | |
| 	// should be installed with the module.
 | |
| 	Test_config_template *string `android:"path,arch_variant"`
 | |
| 
 | |
| 	// list of compatibility suites (for example "cts", "vts") that the module should be
 | |
| 	// installed into.
 | |
| 	Test_suites []string `android:"arch_variant"`
 | |
| 
 | |
| 	// list of files or filegroup modules that provide data that should be installed alongside
 | |
| 	// the test
 | |
| 	Data []string `android:"path,arch_variant"`
 | |
| 
 | |
| 	// list of shared library modules that should be installed alongside the test
 | |
| 	Data_libs []string `android:"arch_variant"`
 | |
| 
 | |
| 	// list of binary modules that should be installed alongside the test
 | |
| 	Data_bins []string `android:"arch_variant"`
 | |
| 
 | |
| 	// Flag to indicate whether or not to create test config automatically. If AndroidTest.xml
 | |
| 	// doesn't exist next to the Android.bp, this attribute doesn't need to be set to true
 | |
| 	// explicitly.
 | |
| 	Auto_gen_config *bool
 | |
| 
 | |
| 	// if set, build with the standard Rust test harness. Defaults to true.
 | |
| 	Test_harness *bool
 | |
| 
 | |
| 	// Test options.
 | |
| 	Test_options android.CommonTestOptions
 | |
| 
 | |
| 	// Add RootTargetPreparer to auto generated test config. This guarantees the test to run
 | |
| 	// with root permission.
 | |
| 	Require_root *bool
 | |
| }
 | |
| 
 | |
| // A test module is a binary module with extra --test compiler flag
 | |
| // and different default installation directory.
 | |
| // In golang, inheriance is written as a component.
 | |
| type testDecorator struct {
 | |
| 	*binaryDecorator
 | |
| 	Properties TestProperties
 | |
| 	testConfig android.Path
 | |
| 
 | |
| 	data []android.DataPath
 | |
| }
 | |
| 
 | |
| func (test *testDecorator) dataPaths() []android.DataPath {
 | |
| 	return test.data
 | |
| }
 | |
| 
 | |
| func (test *testDecorator) nativeCoverage() bool {
 | |
| 	return true
 | |
| }
 | |
| 
 | |
| func (test *testDecorator) testHarness() bool {
 | |
| 	return BoolDefault(test.Properties.Test_harness, true)
 | |
| }
 | |
| 
 | |
| func NewRustTest(hod android.HostOrDeviceSupported) (*Module, *testDecorator) {
 | |
| 	// Build both 32 and 64 targets for device tests.
 | |
| 	// Cannot build both for host tests yet if the test depends on
 | |
| 	// something like proc-macro2 that cannot be built for both.
 | |
| 	multilib := android.MultilibBoth
 | |
| 	if hod != android.DeviceSupported && hod != android.HostAndDeviceSupported {
 | |
| 		multilib = android.MultilibFirst
 | |
| 	}
 | |
| 	module := newModule(hod, multilib)
 | |
| 
 | |
| 	test := &testDecorator{
 | |
| 		binaryDecorator: &binaryDecorator{
 | |
| 			baseCompiler: NewBaseCompiler("nativetest", "nativetest64", InstallInData),
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	module.compiler = test
 | |
| 	return module, test
 | |
| }
 | |
| 
 | |
| func (test *testDecorator) compilerProps() []interface{} {
 | |
| 	return append(test.binaryDecorator.compilerProps(), &test.Properties)
 | |
| }
 | |
| 
 | |
| func (test *testDecorator) install(ctx ModuleContext) {
 | |
| 	testInstallBase := "/data/local/tests/unrestricted"
 | |
| 	if ctx.RustModule().InVendor() || ctx.RustModule().UseVndk() {
 | |
| 		testInstallBase = "/data/local/tests/vendor"
 | |
| 	}
 | |
| 
 | |
| 	var configs []tradefed.Config
 | |
| 	if Bool(test.Properties.Require_root) {
 | |
| 		configs = append(configs, tradefed.Object{"target_preparer", "com.android.tradefed.targetprep.RootTargetPreparer", nil})
 | |
| 	} else {
 | |
| 		var options []tradefed.Option
 | |
| 		options = append(options, tradefed.Option{Name: "force-root", Value: "false"})
 | |
| 		configs = append(configs, tradefed.Object{"target_preparer", "com.android.tradefed.targetprep.RootTargetPreparer", options})
 | |
| 	}
 | |
| 
 | |
| 	test.testConfig = tradefed.AutoGenRustTestConfig(ctx,
 | |
| 		test.Properties.Test_config,
 | |
| 		test.Properties.Test_config_template,
 | |
| 		test.Properties.Test_suites,
 | |
| 		configs,
 | |
| 		test.Properties.Auto_gen_config,
 | |
| 		testInstallBase)
 | |
| 
 | |
| 	dataSrcPaths := android.PathsForModuleSrc(ctx, test.Properties.Data)
 | |
| 
 | |
| 	ctx.VisitDirectDepsWithTag(dataLibDepTag, func(dep android.Module) {
 | |
| 		depName := ctx.OtherModuleName(dep)
 | |
| 		linkableDep, ok := dep.(cc.LinkableInterface)
 | |
| 		if !ok {
 | |
| 			ctx.ModuleErrorf("data_lib %q is not a linkable module", depName)
 | |
| 		}
 | |
| 		if linkableDep.OutputFile().Valid() {
 | |
| 			// Copy the output in "lib[64]" so that it's compatible with
 | |
| 			// the default rpath values.
 | |
| 			libDir := "lib"
 | |
| 			if linkableDep.Target().Arch.ArchType.Multilib == "lib64" {
 | |
| 				libDir = "lib64"
 | |
| 			}
 | |
| 			test.data = append(test.data,
 | |
| 				android.DataPath{SrcPath: linkableDep.OutputFile().Path(),
 | |
| 					RelativeInstallPath: filepath.Join(libDir, linkableDep.RelativeInstallPath())})
 | |
| 		}
 | |
| 	})
 | |
| 
 | |
| 	ctx.VisitDirectDepsWithTag(dataBinDepTag, func(dep android.Module) {
 | |
| 		depName := ctx.OtherModuleName(dep)
 | |
| 		linkableDep, ok := dep.(cc.LinkableInterface)
 | |
| 		if !ok {
 | |
| 			ctx.ModuleErrorf("data_bin %q is not a linkable module", depName)
 | |
| 		}
 | |
| 		if linkableDep.OutputFile().Valid() {
 | |
| 			test.data = append(test.data,
 | |
| 				android.DataPath{SrcPath: linkableDep.OutputFile().Path(),
 | |
| 					RelativeInstallPath: linkableDep.RelativeInstallPath()})
 | |
| 		}
 | |
| 	})
 | |
| 
 | |
| 	for _, dataSrcPath := range dataSrcPaths {
 | |
| 		test.data = append(test.data, android.DataPath{SrcPath: dataSrcPath})
 | |
| 	}
 | |
| 
 | |
| 	// default relative install path is module name
 | |
| 	if !Bool(test.Properties.No_named_install_directory) {
 | |
| 		test.baseCompiler.relative = ctx.ModuleName()
 | |
| 	} else if String(test.baseCompiler.Properties.Relative_install_path) == "" {
 | |
| 		ctx.PropertyErrorf("no_named_install_directory", "Module install directory may only be disabled if relative_install_path is set")
 | |
| 	}
 | |
| 
 | |
| 	if ctx.Host() && test.Properties.Test_options.Unit_test == nil {
 | |
| 		test.Properties.Test_options.Unit_test = proptools.BoolPtr(true)
 | |
| 	}
 | |
| 	test.binaryDecorator.install(ctx)
 | |
| }
 | |
| 
 | |
| func (test *testDecorator) compilerFlags(ctx ModuleContext, flags Flags) Flags {
 | |
| 	flags = test.binaryDecorator.compilerFlags(ctx, flags)
 | |
| 	if test.testHarness() {
 | |
| 		flags.RustFlags = append(flags.RustFlags, "--test")
 | |
| 	}
 | |
| 	if ctx.Device() {
 | |
| 		flags.RustFlags = append(flags.RustFlags, "-Z panic_abort_tests")
 | |
| 	}
 | |
| 	return flags
 | |
| }
 | |
| 
 | |
| func (test *testDecorator) autoDep(ctx android.BottomUpMutatorContext) autoDep {
 | |
| 	return rlibAutoDep
 | |
| }
 | |
| 
 | |
| func init() {
 | |
| 	// Rust tests are binary files built with --test.
 | |
| 	android.RegisterModuleType("rust_test", RustTestFactory)
 | |
| 	android.RegisterModuleType("rust_test_host", RustTestHostFactory)
 | |
| }
 | |
| 
 | |
| func RustTestFactory() android.Module {
 | |
| 	module, _ := NewRustTest(android.HostAndDeviceSupported)
 | |
| 
 | |
| 	// NewRustTest will set MultilibBoth true, however the host variant
 | |
| 	// cannot produce the non-primary target. Therefore, add the
 | |
| 	// rustTestHostMultilib load hook to set MultilibFirst for the
 | |
| 	// host target.
 | |
| 	android.AddLoadHook(module, rustTestHostMultilib)
 | |
| 	return module.Init()
 | |
| }
 | |
| 
 | |
| func RustTestHostFactory() android.Module {
 | |
| 	module, _ := NewRustTest(android.HostSupported)
 | |
| 	return module.Init()
 | |
| }
 | |
| 
 | |
| func (test *testDecorator) stdLinkage(ctx *depsContext) RustLinkage {
 | |
| 	return RlibLinkage
 | |
| }
 | |
| 
 | |
| func (test *testDecorator) compilerDeps(ctx DepsContext, deps Deps) Deps {
 | |
| 	deps = test.binaryDecorator.compilerDeps(ctx, deps)
 | |
| 
 | |
| 	deps.Rustlibs = append(deps.Rustlibs, "libtest")
 | |
| 
 | |
| 	deps.DataLibs = append(deps.DataLibs, test.Properties.Data_libs...)
 | |
| 	deps.DataBins = append(deps.DataBins, test.Properties.Data_bins...)
 | |
| 
 | |
| 	return deps
 | |
| }
 | |
| 
 | |
| func (test *testDecorator) testBinary() bool {
 | |
| 	return true
 | |
| }
 | |
| 
 | |
| func rustTestHostMultilib(ctx android.LoadHookContext) {
 | |
| 	type props struct {
 | |
| 		Target struct {
 | |
| 			Host struct {
 | |
| 				Compile_multilib *string
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	p := &props{}
 | |
| 	p.Target.Host.Compile_multilib = proptools.StringPtr("first")
 | |
| 	ctx.AppendProperties(p)
 | |
| }
 |