1// Copyright 2016 Google Inc. All rights reserved.
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
15package cc
16
17import (
18	"path/filepath"
19	"strconv"
20	"strings"
21
22	"github.com/google/blueprint/proptools"
23
24	"android/soong/android"
25	"android/soong/tradefed"
26)
27
28// TestLinkerProperties properties to be registered via the linker
29type TestLinkerProperties struct {
30	// if set, build against the gtest library. Defaults to true.
31	Gtest *bool
32
33	// if set, use the isolated gtest runner. Defaults to true if gtest is also true and the arch is Windows, false
34	// otherwise.
35	Isolated *bool
36}
37
38// TestInstallerProperties properties to be registered via the installer
39type TestInstallerProperties struct {
40	// list of compatibility suites (for example "cts", "vts") that the module should be installed into.
41	Test_suites []string `android:"arch_variant"`
42}
43
44// Test option struct.
45type TestOptions struct {
46	android.CommonTestOptions
47
48	// The UID that you want to run the test as on a device.
49	Run_test_as *string
50
51	// A list of free-formed strings without spaces that categorize the test.
52	Test_suite_tag []string
53
54	// a list of extra test configuration files that should be installed with the module.
55	Extra_test_configs []string `android:"path,arch_variant"`
56
57	// Add ShippingApiLevelModuleController to auto generated test config. If the device properties
58	// for the shipping api level is less than the min_shipping_api_level, skip this module.
59	Min_shipping_api_level *int64
60
61	// Add ShippingApiLevelModuleController to auto generated test config. If any of the device
62	// shipping api level and vendor api level properties are less than the
63	// vsr_min_shipping_api_level, skip this module.
64	// As this includes the shipping api level check, it is not allowed to define
65	// min_shipping_api_level at the same time with this property.
66	Vsr_min_shipping_api_level *int64
67
68	// Add MinApiLevelModuleController with ro.vndk.version property. If ro.vndk.version has an
69	// integer value and the value is less than the min_vndk_version, skip this module.
70	Min_vndk_version *int64
71
72	// Extra <option> tags to add to the auto generated test xml file under the test runner, e.g., GTest.
73	// The "key" is optional in each of these.
74	Test_runner_options []tradefed.Option
75}
76
77type TestBinaryProperties struct {
78	// Create a separate binary for each source file.  Useful when there is
79	// global state that can not be torn down and reset between each test suite.
80	Test_per_src *bool
81
82	// Disables the creation of a test-specific directory when used with
83	// relative_install_path. Useful if several tests need to be in the same
84	// directory, but test_per_src doesn't work.
85	No_named_install_directory *bool
86
87	// list of files or filegroup modules that provide data that should be installed alongside
88	// the test
89	Data []string `android:"path,arch_variant"`
90
91	// list of shared library modules that should be installed alongside the test
92	Data_libs []string `android:"arch_variant"`
93
94	// list of binary modules that should be installed alongside the test
95	Data_bins []string `android:"arch_variant"`
96
97	// the name of the test configuration (for example "AndroidTest.xml") that should be
98	// installed with the module.
99	Test_config *string `android:"path,arch_variant"`
100
101	// the name of the test configuration template (for example "AndroidTestTemplate.xml") that
102	// should be installed with the module.
103	Test_config_template *string `android:"path,arch_variant"`
104
105	// Test options.
106	Test_options TestOptions
107
108	// Add RootTargetPreparer to auto generated test config. This guarantees the test to run
109	// with root permission.
110	Require_root *bool
111
112	// Add RunCommandTargetPreparer to stop framework before the test and start it after the test.
113	Disable_framework *bool
114
115	// Flag to indicate whether or not to create test config automatically. If AndroidTest.xml
116	// doesn't exist next to the Android.bp, this attribute doesn't need to be set to true
117	// explicitly.
118	Auto_gen_config *bool
119
120	// Add parameterized mainline modules to auto generated test config. The options will be
121	// handled by TradeFed to download and install the specified modules on the device.
122	Test_mainline_modules []string
123
124	// Install the test into a folder named for the module in all test suites.
125	Per_testcase_directory *bool
126}
127
128func init() {
129	android.RegisterModuleType("cc_test", TestFactory)
130	android.RegisterModuleType("cc_test_library", TestLibraryFactory)
131	android.RegisterModuleType("cc_benchmark", BenchmarkFactory)
132	android.RegisterModuleType("cc_test_host", TestHostFactory)
133	android.RegisterModuleType("cc_benchmark_host", BenchmarkHostFactory)
134}
135
136// cc_test generates a test config file and an executable binary file to test
137// specific functionality on a device. The executable binary gets an implicit
138// static_libs dependency on libgtests unless the gtest flag is set to false.
139func TestFactory() android.Module {
140	module := NewTest(android.HostAndDeviceSupported)
141	return module.Init()
142}
143
144// cc_test_library creates an archive of files (i.e. .o files) which is later
145// referenced by another module (such as cc_test, cc_defaults or cc_test_library)
146// for archiving or linking.
147func TestLibraryFactory() android.Module {
148	module := NewTestLibrary(android.HostAndDeviceSupported)
149	return module.Init()
150}
151
152// cc_benchmark compiles an executable binary that performs benchmark testing
153// of a specific component in a device. Additional files such as test suites
154// and test configuration are installed on the side of the compiled executed
155// binary.
156func BenchmarkFactory() android.Module {
157	module := NewBenchmark(android.HostAndDeviceSupported)
158	module.testModule = true
159	return module.Init()
160}
161
162// cc_test_host compiles a test host binary.
163func TestHostFactory() android.Module {
164	module := NewTest(android.HostSupported)
165	return module.Init()
166}
167
168// cc_benchmark_host compiles an executable binary that performs benchmark
169// testing of a specific component in the host. Additional files such as
170// test suites and test configuration are installed on the side of the
171// compiled executed binary.
172func BenchmarkHostFactory() android.Module {
173	module := NewBenchmark(android.HostSupported)
174	return module.Init()
175}
176
177type testPerSrc interface {
178	testPerSrc() bool
179	srcs() []string
180	isAllTestsVariation() bool
181	setSrc(string, string)
182	unsetSrc()
183}
184
185func (test *testBinary) testPerSrc() bool {
186	return Bool(test.Properties.Test_per_src)
187}
188
189func (test *testBinary) srcs() []string {
190	return test.baseCompiler.Properties.Srcs
191}
192
193func (test *testBinary) dataPaths() []android.DataPath {
194	return test.data
195}
196
197func (test *testBinary) isAllTestsVariation() bool {
198	stem := test.binaryDecorator.Properties.Stem
199	return stem != nil && *stem == ""
200}
201
202func (test *testBinary) setSrc(name, src string) {
203	test.baseCompiler.Properties.Srcs = []string{src}
204	test.binaryDecorator.Properties.Stem = StringPtr(name)
205}
206
207func (test *testBinary) unsetSrc() {
208	test.baseCompiler.Properties.Srcs = nil
209	test.binaryDecorator.Properties.Stem = StringPtr("")
210}
211
212func (test *testBinary) testBinary() bool {
213	return true
214}
215
216var _ testPerSrc = (*testBinary)(nil)
217
218func TestPerSrcMutator(mctx android.BottomUpMutatorContext) {
219	if m, ok := mctx.Module().(*Module); ok {
220		if test, ok := m.linker.(testPerSrc); ok {
221			numTests := len(test.srcs())
222			if test.testPerSrc() && numTests > 0 {
223				if duplicate, found := android.CheckDuplicate(test.srcs()); found {
224					mctx.PropertyErrorf("srcs", "found a duplicate entry %q", duplicate)
225					return
226				}
227				testNames := make([]string, numTests)
228				for i, src := range test.srcs() {
229					testNames[i] = strings.TrimSuffix(filepath.Base(src), filepath.Ext(src))
230				}
231				// In addition to creating one variation per test source file,
232				// create an additional "all tests" variation named "", and have it
233				// depends on all other test_per_src variations. This is useful to
234				// create subsequent dependencies of a given module on all
235				// test_per_src variations created above: by depending on
236				// variation "", that module will transitively depend on all the
237				// other test_per_src variations without the need to know their
238				// name or even their number.
239				testNames = append(testNames, "")
240				tests := mctx.CreateLocalVariations(testNames...)
241				allTests := tests[numTests]
242				allTests.(*Module).linker.(testPerSrc).unsetSrc()
243				// Prevent the "all tests" variation from being installable nor
244				// exporting to Make, as it won't create any output file.
245				allTests.(*Module).Properties.PreventInstall = true
246				allTests.(*Module).Properties.HideFromMake = true
247				for i, src := range test.srcs() {
248					tests[i].(*Module).linker.(testPerSrc).setSrc(testNames[i], src)
249					mctx.AddInterVariantDependency(testPerSrcDepTag, allTests, tests[i])
250				}
251				mctx.AliasVariation("")
252			}
253		}
254	}
255}
256
257type testDecorator struct {
258	LinkerProperties    TestLinkerProperties
259	InstallerProperties TestInstallerProperties
260	installer           *baseInstaller
261	linker              *baseLinker
262}
263
264func (test *testDecorator) gtest() bool {
265	return BoolDefault(test.LinkerProperties.Gtest, true)
266}
267
268func (test *testDecorator) isolated(ctx android.EarlyModuleContext) bool {
269	return BoolDefault(test.LinkerProperties.Isolated, false)
270}
271
272// NOTE: Keep this in sync with cc/cc_test.bzl#gtest_copts
273func (test *testDecorator) linkerFlags(ctx ModuleContext, flags Flags) Flags {
274	if !test.gtest() {
275		return flags
276	}
277
278	flags.Local.CFlags = append(flags.Local.CFlags, "-DGTEST_HAS_STD_STRING")
279	if ctx.Host() {
280		flags.Local.CFlags = append(flags.Local.CFlags, "-O0", "-g")
281
282		switch ctx.Os() {
283		case android.Windows:
284			flags.Local.CFlags = append(flags.Local.CFlags, "-DGTEST_OS_WINDOWS")
285		case android.Linux:
286			flags.Local.CFlags = append(flags.Local.CFlags, "-DGTEST_OS_LINUX")
287		case android.Darwin:
288			flags.Local.CFlags = append(flags.Local.CFlags, "-DGTEST_OS_MAC")
289		}
290	} else {
291		flags.Local.CFlags = append(flags.Local.CFlags, "-DGTEST_OS_LINUX_ANDROID")
292	}
293
294	return flags
295}
296
297func (test *testDecorator) linkerDeps(ctx BaseModuleContext, deps Deps) Deps {
298	if test.gtest() {
299		if ctx.useSdk() && ctx.Device() {
300			deps.StaticLibs = append(deps.StaticLibs, "libgtest_main_ndk_c++", "libgtest_ndk_c++")
301		} else if test.isolated(ctx) {
302			deps.StaticLibs = append(deps.StaticLibs, "libgtest_isolated_main")
303			// The isolated library requires liblog, but adding it
304			// as a static library means unit tests cannot override
305			// liblog functions. Instead make it a shared library
306			// dependency.
307			deps.SharedLibs = append(deps.SharedLibs, "liblog")
308		} else {
309			deps.StaticLibs = append(deps.StaticLibs, "libgtest_main", "libgtest")
310		}
311	}
312
313	return deps
314}
315
316func (test *testDecorator) linkerProps() []interface{} {
317	return []interface{}{&test.LinkerProperties}
318}
319
320func (test *testDecorator) installerProps() []interface{} {
321	return []interface{}{&test.InstallerProperties}
322}
323
324func (test *testDecorator) moduleInfoJSON(ctx android.ModuleContext, moduleInfoJSON *android.ModuleInfoJSON) {
325	if android.PrefixInList(moduleInfoJSON.CompatibilitySuites, "mts-") &&
326		!android.InList("mts", moduleInfoJSON.CompatibilitySuites) {
327		moduleInfoJSON.CompatibilitySuites = append(moduleInfoJSON.CompatibilitySuites, "mts")
328	}
329}
330
331func NewTestInstaller() *baseInstaller {
332	return NewBaseInstaller("nativetest", "nativetest64", InstallInData)
333}
334
335type testBinary struct {
336	*testDecorator
337	*binaryDecorator
338	*baseCompiler
339	Properties       TestBinaryProperties
340	data             []android.DataPath
341	testConfig       android.Path
342	extraTestConfigs android.Paths
343}
344
345func (test *testBinary) linkerProps() []interface{} {
346	props := append(test.testDecorator.linkerProps(), test.binaryDecorator.linkerProps()...)
347	props = append(props, &test.Properties)
348	return props
349}
350
351func (test *testBinary) linkerDeps(ctx DepsContext, deps Deps) Deps {
352	deps = test.testDecorator.linkerDeps(ctx, deps)
353	deps = test.binaryDecorator.linkerDeps(ctx, deps)
354	deps.DataLibs = append(deps.DataLibs, test.Properties.Data_libs...)
355	deps.DataBins = append(deps.DataBins, test.Properties.Data_bins...)
356	return deps
357}
358
359func (test *testBinary) linkerFlags(ctx ModuleContext, flags Flags) Flags {
360	flags = test.binaryDecorator.linkerFlags(ctx, flags)
361	flags = test.testDecorator.linkerFlags(ctx, flags)
362
363	// Add a default rpath to allow tests to dlopen libraries specified in data_libs.
364	// Host modules already get an rpath specified in linker.go.
365	if !ctx.Host() {
366		flags.Global.LdFlags = append(flags.Global.LdFlags, `-Wl,-rpath,\$$ORIGIN`)
367	}
368	return flags
369}
370
371func (test *testBinary) moduleInfoJSON(ctx ModuleContext, moduleInfoJSON *android.ModuleInfoJSON) {
372	if ctx.Host() && Bool(test.Properties.Test_options.Unit_test) {
373		moduleInfoJSON.CompatibilitySuites = append(moduleInfoJSON.CompatibilitySuites, "host-unit-tests")
374	}
375	moduleInfoJSON.TestOptionsTags = append(moduleInfoJSON.TestOptionsTags, test.Properties.Test_options.Tags...)
376	moduleInfoJSON.TestMainlineModules = append(moduleInfoJSON.TestMainlineModules, test.Properties.Test_mainline_modules...)
377	if test.testConfig != nil {
378		if _, ok := test.testConfig.(android.WritablePath); ok {
379			moduleInfoJSON.AutoTestConfig = []string{"true"}
380		}
381		moduleInfoJSON.TestConfig = append(moduleInfoJSON.TestConfig, test.testConfig.String())
382	}
383	moduleInfoJSON.TestConfig = append(moduleInfoJSON.TestConfig, test.extraTestConfigs.Strings()...)
384
385	if Bool(test.Properties.Test_per_src) {
386		moduleInfoJSON.SubName = "_" + String(test.binaryDecorator.Properties.Stem)
387	}
388
389	moduleInfoJSON.DataDependencies = append(moduleInfoJSON.DataDependencies, test.Properties.Data_bins...)
390
391	if len(test.InstallerProperties.Test_suites) > 0 {
392		moduleInfoJSON.CompatibilitySuites = append(moduleInfoJSON.CompatibilitySuites, test.InstallerProperties.Test_suites...)
393	} else {
394		moduleInfoJSON.CompatibilitySuites = append(moduleInfoJSON.CompatibilitySuites, "null-suite")
395	}
396
397	test.binaryDecorator.moduleInfoJSON(ctx, moduleInfoJSON)
398	test.testDecorator.moduleInfoJSON(ctx, moduleInfoJSON)
399	moduleInfoJSON.Class = []string{"NATIVE_TESTS"}
400
401}
402
403func (test *testBinary) installerProps() []interface{} {
404	return append(test.baseInstaller.installerProps(), test.testDecorator.installerProps()...)
405}
406
407func (test *testBinary) install(ctx ModuleContext, file android.Path) {
408	dataSrcPaths := android.PathsForModuleSrc(ctx, test.Properties.Data)
409
410	for _, dataSrcPath := range dataSrcPaths {
411		test.data = append(test.data, android.DataPath{SrcPath: dataSrcPath})
412	}
413
414	ctx.VisitDirectDepsWithTag(dataLibDepTag, func(dep android.Module) {
415		depName := ctx.OtherModuleName(dep)
416		linkableDep, ok := dep.(LinkableInterface)
417		if !ok {
418			ctx.ModuleErrorf("data_lib %q is not a LinkableInterface module", depName)
419		}
420		if linkableDep.OutputFile().Valid() {
421			test.data = append(test.data,
422				android.DataPath{SrcPath: linkableDep.OutputFile().Path(),
423					RelativeInstallPath: linkableDep.RelativeInstallPath()})
424		}
425	})
426	ctx.VisitDirectDepsWithTag(dataBinDepTag, func(dep android.Module) {
427		depName := ctx.OtherModuleName(dep)
428		linkableDep, ok := dep.(LinkableInterface)
429		if !ok {
430			ctx.ModuleErrorf("data_bin %q is not a LinkableInterface module", depName)
431		}
432		if linkableDep.OutputFile().Valid() {
433			test.data = append(test.data,
434				android.DataPath{SrcPath: linkableDep.OutputFile().Path(),
435					RelativeInstallPath: linkableDep.RelativeInstallPath()})
436		}
437	})
438
439	testInstallBase := getTestInstallBase(ctx.InVendorOrProduct())
440	configs := getTradefedConfigOptions(ctx, &test.Properties, test.isolated(ctx), ctx.Device())
441
442	test.testConfig = tradefed.AutoGenTestConfig(ctx, tradefed.AutoGenTestConfigOptions{
443		TestConfigProp:         test.Properties.Test_config,
444		TestConfigTemplateProp: test.Properties.Test_config_template,
445		TestSuites:             test.testDecorator.InstallerProperties.Test_suites,
446		Config:                 configs,
447		TestRunnerOptions:      test.Properties.Test_options.Test_runner_options,
448		AutoGenConfig:          test.Properties.Auto_gen_config,
449		TestInstallBase:        testInstallBase,
450		DeviceTemplate:         "${NativeTestConfigTemplate}",
451		HostTemplate:           "${NativeHostTestConfigTemplate}",
452	})
453
454	test.extraTestConfigs = android.PathsForModuleSrc(ctx, test.Properties.Test_options.Extra_test_configs)
455
456	test.binaryDecorator.baseInstaller.dir = "nativetest"
457	test.binaryDecorator.baseInstaller.dir64 = "nativetest64"
458
459	if !Bool(test.Properties.No_named_install_directory) {
460		test.binaryDecorator.baseInstaller.relative = ctx.ModuleName()
461	} else if String(test.binaryDecorator.baseInstaller.Properties.Relative_install_path) == "" {
462		ctx.PropertyErrorf("no_named_install_directory", "Module install directory may only be disabled if relative_install_path is set")
463	}
464
465	if ctx.Host() && test.gtest() && test.Properties.Test_options.Unit_test == nil {
466		test.Properties.Test_options.Unit_test = proptools.BoolPtr(true)
467	}
468
469	test.binaryDecorator.baseInstaller.installTestData(ctx, test.data)
470	test.binaryDecorator.baseInstaller.install(ctx, file)
471}
472
473func getTestInstallBase(useVendor bool) string {
474	// TODO: (b/167308193) Switch to /data/local/tests/unrestricted as the default install base.
475	testInstallBase := "/data/local/tmp"
476	if useVendor {
477		testInstallBase = "/data/local/tests/vendor"
478	}
479	return testInstallBase
480}
481
482func getTradefedConfigOptions(ctx android.EarlyModuleContext, properties *TestBinaryProperties, isolated bool, device bool) []tradefed.Config {
483	var configs []tradefed.Config
484
485	for _, module := range properties.Test_mainline_modules {
486		configs = append(configs, tradefed.Option{Name: "config-descriptor:metadata", Key: "mainline-param", Value: module})
487	}
488	if device {
489		if Bool(properties.Require_root) {
490			configs = append(configs, tradefed.Object{"target_preparer", "com.android.tradefed.targetprep.RootTargetPreparer", nil})
491		} else {
492			var options []tradefed.Option
493			options = append(options, tradefed.Option{Name: "force-root", Value: "false"})
494			configs = append(configs, tradefed.Object{"target_preparer", "com.android.tradefed.targetprep.RootTargetPreparer", options})
495		}
496		if Bool(properties.Disable_framework) {
497			var options []tradefed.Option
498			configs = append(configs, tradefed.Object{"target_preparer", "com.android.tradefed.targetprep.StopServicesSetup", options})
499		}
500	}
501	if isolated {
502		configs = append(configs, tradefed.Option{Name: "not-shardable", Value: "true"})
503	}
504	if properties.Test_options.Run_test_as != nil {
505		configs = append(configs, tradefed.Option{Name: "run-test-as", Value: String(properties.Test_options.Run_test_as)})
506	}
507	for _, tag := range properties.Test_options.Test_suite_tag {
508		configs = append(configs, tradefed.Option{Name: "test-suite-tag", Value: tag})
509	}
510	if properties.Test_options.Min_shipping_api_level != nil {
511		if properties.Test_options.Vsr_min_shipping_api_level != nil {
512			ctx.PropertyErrorf("test_options.min_shipping_api_level", "must not be set at the same time as 'vsr_min_shipping_api_level'.")
513		}
514		var options []tradefed.Option
515		options = append(options, tradefed.Option{Name: "min-api-level", Value: strconv.FormatInt(int64(*properties.Test_options.Min_shipping_api_level), 10)})
516		configs = append(configs, tradefed.Object{"module_controller", "com.android.tradefed.testtype.suite.module.ShippingApiLevelModuleController", options})
517	}
518	if properties.Test_options.Vsr_min_shipping_api_level != nil {
519		var options []tradefed.Option
520		options = append(options, tradefed.Option{Name: "vsr-min-api-level", Value: strconv.FormatInt(int64(*properties.Test_options.Vsr_min_shipping_api_level), 10)})
521		configs = append(configs, tradefed.Object{"module_controller", "com.android.tradefed.testtype.suite.module.ShippingApiLevelModuleController", options})
522	}
523	if properties.Test_options.Min_vndk_version != nil {
524		var options []tradefed.Option
525		options = append(options, tradefed.Option{Name: "min-api-level", Value: strconv.FormatInt(int64(*properties.Test_options.Min_vndk_version), 10)})
526		options = append(options, tradefed.Option{Name: "api-level-prop", Value: "ro.vndk.version"})
527		configs = append(configs, tradefed.Object{"module_controller", "com.android.tradefed.testtype.suite.module.MinApiLevelModuleController", options})
528	}
529	return configs
530}
531
532func NewTest(hod android.HostOrDeviceSupported) *Module {
533	module, binary := newBinary(hod)
534	module.multilib = android.MultilibBoth
535	module.testModule = true
536	binary.baseInstaller = NewTestInstaller()
537
538	test := &testBinary{
539		testDecorator: &testDecorator{
540			linker:    binary.baseLinker,
541			installer: binary.baseInstaller,
542		},
543		binaryDecorator: binary,
544		baseCompiler:    NewBaseCompiler(),
545	}
546	module.compiler = test
547	module.linker = test
548	module.installer = test
549	return module
550}
551
552type testLibrary struct {
553	*testDecorator
554	*libraryDecorator
555}
556
557func (test *testLibrary) testLibrary() bool {
558	return true
559}
560
561func (test *testLibrary) linkerProps() []interface{} {
562	var props []interface{}
563	props = append(props, test.testDecorator.linkerProps()...)
564	return append(props, test.libraryDecorator.linkerProps()...)
565}
566
567func (test *testLibrary) linkerDeps(ctx DepsContext, deps Deps) Deps {
568	deps = test.testDecorator.linkerDeps(ctx, deps)
569	deps = test.libraryDecorator.linkerDeps(ctx, deps)
570	return deps
571}
572
573func (test *testLibrary) linkerFlags(ctx ModuleContext, flags Flags) Flags {
574	flags = test.libraryDecorator.linkerFlags(ctx, flags)
575	flags = test.testDecorator.linkerFlags(ctx, flags)
576	return flags
577}
578
579func (test *testLibrary) moduleInfoJSON(ctx ModuleContext, moduleInfoJSON *android.ModuleInfoJSON) {
580	if len(test.InstallerProperties.Test_suites) > 0 {
581		moduleInfoJSON.CompatibilitySuites = append(moduleInfoJSON.CompatibilitySuites, test.InstallerProperties.Test_suites...)
582	}
583
584	test.libraryDecorator.moduleInfoJSON(ctx, moduleInfoJSON)
585	test.testDecorator.moduleInfoJSON(ctx, moduleInfoJSON)
586}
587
588func (test *testLibrary) installerProps() []interface{} {
589	return append(test.baseInstaller.installerProps(), test.testDecorator.installerProps()...)
590}
591
592func NewTestLibrary(hod android.HostOrDeviceSupported) *Module {
593	module, library := NewLibrary(android.HostAndDeviceSupported)
594	library.baseInstaller = NewTestInstaller()
595	test := &testLibrary{
596		testDecorator: &testDecorator{
597			linker:    library.baseLinker,
598			installer: library.baseInstaller,
599		},
600		libraryDecorator: library,
601	}
602	module.linker = test
603	module.installer = test
604	return module
605}
606
607type BenchmarkProperties struct {
608	// list of files or filegroup modules that provide data that should be installed alongside
609	// the test
610	Data []string `android:"path"`
611
612	// list of compatibility suites (for example "cts", "vts") that the module should be
613	// installed into.
614	Test_suites []string `android:"arch_variant"`
615
616	// the name of the test configuration (for example "AndroidTest.xml") that should be
617	// installed with the module.
618	Test_config *string `android:"path,arch_variant"`
619
620	// the name of the test configuration template (for example "AndroidTestTemplate.xml") that
621	// should be installed with the module.
622	Test_config_template *string `android:"path,arch_variant"`
623
624	// Add RootTargetPreparer to auto generated test config. This guarantees the test to run
625	// with root permission.
626	Require_root *bool
627
628	// Flag to indicate whether or not to create test config automatically. If AndroidTest.xml
629	// doesn't exist next to the Android.bp, this attribute doesn't need to be set to true
630	// explicitly.
631	Auto_gen_config *bool
632}
633
634type benchmarkDecorator struct {
635	*binaryDecorator
636	Properties BenchmarkProperties
637	data       []android.DataPath
638	testConfig android.Path
639}
640
641func (benchmark *benchmarkDecorator) benchmarkBinary() bool {
642	return true
643}
644
645func (benchmark *benchmarkDecorator) linkerProps() []interface{} {
646	props := benchmark.binaryDecorator.linkerProps()
647	props = append(props, &benchmark.Properties)
648	return props
649}
650
651func (benchmark *benchmarkDecorator) linkerDeps(ctx DepsContext, deps Deps) Deps {
652	deps = benchmark.binaryDecorator.linkerDeps(ctx, deps)
653	deps.StaticLibs = append(deps.StaticLibs, "libgoogle-benchmark")
654	return deps
655}
656
657func (benchmark *benchmarkDecorator) install(ctx ModuleContext, file android.Path) {
658	for _, d := range android.PathsForModuleSrc(ctx, benchmark.Properties.Data) {
659		benchmark.data = append(benchmark.data, android.DataPath{SrcPath: d})
660	}
661
662	var configs []tradefed.Config
663	if Bool(benchmark.Properties.Require_root) {
664		configs = append(configs, tradefed.Object{"target_preparer", "com.android.tradefed.targetprep.RootTargetPreparer", nil})
665	}
666	benchmark.testConfig = tradefed.AutoGenTestConfig(ctx, tradefed.AutoGenTestConfigOptions{
667		TestConfigProp:         benchmark.Properties.Test_config,
668		TestConfigTemplateProp: benchmark.Properties.Test_config_template,
669		TestSuites:             benchmark.Properties.Test_suites,
670		Config:                 configs,
671		AutoGenConfig:          benchmark.Properties.Auto_gen_config,
672		DeviceTemplate:         "${NativeBenchmarkTestConfigTemplate}",
673		HostTemplate:           "${NativeBenchmarkTestConfigTemplate}",
674	})
675
676	benchmark.binaryDecorator.baseInstaller.dir = filepath.Join("benchmarktest", ctx.ModuleName())
677	benchmark.binaryDecorator.baseInstaller.dir64 = filepath.Join("benchmarktest64", ctx.ModuleName())
678	benchmark.binaryDecorator.baseInstaller.installTestData(ctx, benchmark.data)
679	benchmark.binaryDecorator.baseInstaller.install(ctx, file)
680}
681
682func (benchmark *benchmarkDecorator) moduleInfoJSON(ctx ModuleContext, moduleInfoJSON *android.ModuleInfoJSON) {
683	benchmark.binaryDecorator.moduleInfoJSON(ctx, moduleInfoJSON)
684
685	moduleInfoJSON.Class = []string{"NATIVE_TESTS"}
686	if len(benchmark.Properties.Test_suites) > 0 {
687		moduleInfoJSON.CompatibilitySuites = append(moduleInfoJSON.CompatibilitySuites, benchmark.Properties.Test_suites...)
688	} else {
689		moduleInfoJSON.CompatibilitySuites = append(moduleInfoJSON.CompatibilitySuites, "null-suite")
690	}
691
692	if android.PrefixInList(moduleInfoJSON.CompatibilitySuites, "mts-") &&
693		!android.InList("mts", moduleInfoJSON.CompatibilitySuites) {
694		moduleInfoJSON.CompatibilitySuites = append(moduleInfoJSON.CompatibilitySuites, "mts")
695	}
696
697	if benchmark.testConfig != nil {
698		if _, ok := benchmark.testConfig.(android.WritablePath); ok {
699			moduleInfoJSON.AutoTestConfig = []string{"true"}
700		}
701		moduleInfoJSON.TestConfig = []string{benchmark.testConfig.String()}
702	}
703}
704
705func NewBenchmark(hod android.HostOrDeviceSupported) *Module {
706	module, binary := newBinary(hod)
707	module.multilib = android.MultilibBoth
708	binary.baseInstaller = NewBaseInstaller("benchmarktest", "benchmarktest64", InstallInData)
709
710	benchmark := &benchmarkDecorator{
711		binaryDecorator: binary,
712	}
713	module.linker = benchmark
714	module.installer = benchmark
715	return module
716}
717