blob: 045cc4f881b6ca3155900c2626a1d163ca97ea64 [file] [log] [blame]
Colin Cross4d9c2d12016-07-29 12:48:20 -07001// 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 "strings"
20
Colin Cross4d9c2d12016-07-29 12:48:20 -070021 "android/soong/android"
Colin Cross303e21f2018-08-07 16:49:25 -070022 "android/soong/tradefed"
Colin Cross4d9c2d12016-07-29 12:48:20 -070023)
24
Colin Crossb916a382016-07-29 17:28:03 -070025type TestProperties struct {
Colin Cross4d9c2d12016-07-29 12:48:20 -070026 // if set, build against the gtest library. Defaults to true.
Colin Cross600c9df2016-09-13 12:26:16 -070027 Gtest *bool
Christopher Ferris9df92d62018-08-21 12:40:08 -070028
29 // if set, use the isolated gtest runner. Defaults to false.
30 Isolated *bool
Colin Crossb916a382016-07-29 17:28:03 -070031}
Colin Cross4d9c2d12016-07-29 12:48:20 -070032
yelinhsieh9fc60402018-10-01 19:23:14 +080033// Test option struct.
34type TestOptions struct {
35 // The UID that you want to run the test as on a device.
36 Run_test_as *string
37}
38
Colin Crossb916a382016-07-29 17:28:03 -070039type TestBinaryProperties struct {
Colin Cross4d9c2d12016-07-29 12:48:20 -070040 // Create a separate binary for each source file. Useful when there is
41 // global state that can not be torn down and reset between each test suite.
42 Test_per_src *bool
Dan Willemsen3340d602016-12-27 14:40:40 -080043
44 // Disables the creation of a test-specific directory when used with
45 // relative_install_path. Useful if several tests need to be in the same
46 // directory, but test_per_src doesn't work.
47 No_named_install_directory *bool
Colin Crossfaeb7aa2017-02-01 14:12:44 -080048
49 // list of files or filegroup modules that provide data that should be installed alongside
50 // the test
Colin Cross27b922f2019-03-04 22:35:41 -080051 Data []string `android:"path"`
Colin Crossa929db02017-03-27 16:27:50 -070052
53 // list of compatibility suites (for example "cts", "vts") that the module should be
54 // installed into.
Dan Willemsen15d54d52017-09-18 16:49:28 -070055 Test_suites []string `android:"arch_variant"`
Julien Despreze146e392018-08-02 15:00:46 -070056
57 // the name of the test configuration (for example "AndroidTest.xml") that should be
58 // installed with the module.
Colin Cross27b922f2019-03-04 22:35:41 -080059 Test_config *string `android:"path,arch_variant"`
Jack He33338892018-09-19 02:21:28 -070060
61 // the name of the test configuration template (for example "AndroidTestTemplate.xml") that
62 // should be installed with the module.
Colin Cross27b922f2019-03-04 22:35:41 -080063 Test_config_template *string `android:"path,arch_variant"`
yelinhsieh9fc60402018-10-01 19:23:14 +080064
65 // Test options.
66 Test_options TestOptions
Colin Cross4d9c2d12016-07-29 12:48:20 -070067}
68
69func init() {
Steven Moreland87c9d7b2017-11-02 21:38:28 -070070 android.RegisterModuleType("cc_test", TestFactory)
71 android.RegisterModuleType("cc_test_library", TestLibraryFactory)
72 android.RegisterModuleType("cc_benchmark", BenchmarkFactory)
73 android.RegisterModuleType("cc_test_host", TestHostFactory)
74 android.RegisterModuleType("cc_benchmark_host", BenchmarkHostFactory)
Colin Cross4d9c2d12016-07-29 12:48:20 -070075}
76
77// Module factory for tests
Steven Moreland87c9d7b2017-11-02 21:38:28 -070078func TestFactory() android.Module {
Colin Cross4d9c2d12016-07-29 12:48:20 -070079 module := NewTest(android.HostAndDeviceSupported)
80 return module.Init()
81}
82
83// Module factory for test libraries
Steven Moreland87c9d7b2017-11-02 21:38:28 -070084func TestLibraryFactory() android.Module {
Colin Cross4d9c2d12016-07-29 12:48:20 -070085 module := NewTestLibrary(android.HostAndDeviceSupported)
86 return module.Init()
87}
88
89// Module factory for benchmarks
Steven Moreland87c9d7b2017-11-02 21:38:28 -070090func BenchmarkFactory() android.Module {
Colin Cross4d9c2d12016-07-29 12:48:20 -070091 module := NewBenchmark(android.HostAndDeviceSupported)
92 return module.Init()
93}
94
95// Module factory for host tests
Steven Moreland87c9d7b2017-11-02 21:38:28 -070096func TestHostFactory() android.Module {
Colin Cross4d9c2d12016-07-29 12:48:20 -070097 module := NewTest(android.HostSupported)
98 return module.Init()
99}
100
101// Module factory for host benchmarks
Steven Moreland87c9d7b2017-11-02 21:38:28 -0700102func BenchmarkHostFactory() android.Module {
Colin Cross4d9c2d12016-07-29 12:48:20 -0700103 module := NewBenchmark(android.HostSupported)
104 return module.Init()
105}
106
Colin Crossb916a382016-07-29 17:28:03 -0700107type testPerSrc interface {
108 testPerSrc() bool
109 srcs() []string
110 setSrc(string, string)
111}
112
113func (test *testBinary) testPerSrc() bool {
114 return Bool(test.Properties.Test_per_src)
115}
116
117func (test *testBinary) srcs() []string {
118 return test.baseCompiler.Properties.Srcs
119}
120
121func (test *testBinary) setSrc(name, src string) {
122 test.baseCompiler.Properties.Srcs = []string{src}
Nan Zhang0007d812017-11-07 10:57:05 -0800123 test.binaryDecorator.Properties.Stem = StringPtr(name)
Colin Crossb916a382016-07-29 17:28:03 -0700124}
125
126var _ testPerSrc = (*testBinary)(nil)
127
Colin Cross4d9c2d12016-07-29 12:48:20 -0700128func testPerSrcMutator(mctx android.BottomUpMutatorContext) {
129 if m, ok := mctx.Module().(*Module); ok {
Colin Crossb916a382016-07-29 17:28:03 -0700130 if test, ok := m.linker.(testPerSrc); ok {
131 if test.testPerSrc() && len(test.srcs()) > 0 {
Jooyung Hana61ff2c2019-02-28 18:06:34 +0900132 if duplicate, found := checkDuplicate(test.srcs()); found {
133 mctx.PropertyErrorf("srcs", "found a duplicate entry %q", duplicate)
134 return
135 }
Colin Crossb916a382016-07-29 17:28:03 -0700136 testNames := make([]string, len(test.srcs()))
137 for i, src := range test.srcs() {
Colin Cross4d9c2d12016-07-29 12:48:20 -0700138 testNames[i] = strings.TrimSuffix(filepath.Base(src), filepath.Ext(src))
139 }
140 tests := mctx.CreateLocalVariations(testNames...)
Colin Crossb916a382016-07-29 17:28:03 -0700141 for i, src := range test.srcs() {
142 tests[i].(*Module).linker.(testPerSrc).setSrc(testNames[i], src)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700143 }
144 }
145 }
146 }
147}
148
Jooyung Hana61ff2c2019-02-28 18:06:34 +0900149func checkDuplicate(values []string) (duplicate string, found bool) {
150 seen := make(map[string]string)
151 for _, v := range values {
152 if duplicate, found = seen[v]; found {
153 return
154 }
155 seen[v] = v
156 }
157 return
158}
159
Colin Crossb916a382016-07-29 17:28:03 -0700160type testDecorator struct {
161 Properties TestProperties
162 linker *baseLinker
Colin Cross4d9c2d12016-07-29 12:48:20 -0700163}
164
Colin Cross600c9df2016-09-13 12:26:16 -0700165func (test *testDecorator) gtest() bool {
Colin Cross38b40df2018-04-10 16:14:46 -0700166 return BoolDefault(test.Properties.Gtest, true)
Colin Cross600c9df2016-09-13 12:26:16 -0700167}
168
Colin Crossb916a382016-07-29 17:28:03 -0700169func (test *testDecorator) linkerFlags(ctx ModuleContext, flags Flags) Flags {
Colin Cross600c9df2016-09-13 12:26:16 -0700170 if !test.gtest() {
Colin Cross4d9c2d12016-07-29 12:48:20 -0700171 return flags
172 }
173
174 flags.CFlags = append(flags.CFlags, "-DGTEST_HAS_STD_STRING")
175 if ctx.Host() {
176 flags.CFlags = append(flags.CFlags, "-O0", "-g")
177
178 switch ctx.Os() {
179 case android.Windows:
180 flags.CFlags = append(flags.CFlags, "-DGTEST_OS_WINDOWS")
181 case android.Linux:
182 flags.CFlags = append(flags.CFlags, "-DGTEST_OS_LINUX")
Colin Cross4d9c2d12016-07-29 12:48:20 -0700183 case android.Darwin:
184 flags.CFlags = append(flags.CFlags, "-DGTEST_OS_MAC")
Colin Cross4d9c2d12016-07-29 12:48:20 -0700185 }
186 } else {
187 flags.CFlags = append(flags.CFlags, "-DGTEST_OS_LINUX_ANDROID")
188 }
189
190 return flags
191}
192
Colin Crossb916a382016-07-29 17:28:03 -0700193func (test *testDecorator) linkerDeps(ctx BaseModuleContext, deps Deps) Deps {
Colin Cross600c9df2016-09-13 12:26:16 -0700194 if test.gtest() {
Jeff Gastonaf3cc2d2017-09-27 17:01:44 -0700195 if ctx.useSdk() && ctx.Device() {
Dan Albert7dd58992018-02-09 15:22:59 -0800196 deps.StaticLibs = append(deps.StaticLibs, "libgtest_main_ndk_c++", "libgtest_ndk_c++")
Christopher Ferris9df92d62018-08-21 12:40:08 -0700197 } else if BoolDefault(test.Properties.Isolated, false) {
198 deps.StaticLibs = append(deps.StaticLibs, "libgtest_isolated_main")
Colin Cross4d9c2d12016-07-29 12:48:20 -0700199 } else {
200 deps.StaticLibs = append(deps.StaticLibs, "libgtest_main", "libgtest")
201 }
202 }
Colin Crossb916a382016-07-29 17:28:03 -0700203
Colin Cross4d9c2d12016-07-29 12:48:20 -0700204 return deps
205}
206
Colin Crossb916a382016-07-29 17:28:03 -0700207func (test *testDecorator) linkerInit(ctx BaseModuleContext, linker *baseLinker) {
yangbillb3174d12018-05-07 06:41:20 +0000208 // 1. Add ../../lib[64] to rpath so that out/host/linux-x86/nativetest/<test dir>/<test> can
Colin Crossbd75e1d2017-06-30 17:27:55 -0700209 // find out/host/linux-x86/lib[64]/library.so
yangbillb3174d12018-05-07 06:41:20 +0000210 // 2. Add ../../../lib[64] to rpath so that out/host/linux-x86/testcases/<test dir>/<CPU>/<test> can
211 // also find out/host/linux-x86/lib[64]/library.so
212 runpaths := []string{"../../lib", "../../../lib"}
213 for _, runpath := range runpaths {
214 if ctx.toolchain().Is64Bit() {
215 runpath += "64"
216 }
217 linker.dynamicProperties.RunPaths = append(linker.dynamicProperties.RunPaths, runpath)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700218 }
Colin Crossbd75e1d2017-06-30 17:27:55 -0700219
220 // add "" to rpath so that test binaries can find libraries in their own test directory
221 linker.dynamicProperties.RunPaths = append(linker.dynamicProperties.RunPaths, "")
Colin Cross4d9c2d12016-07-29 12:48:20 -0700222}
223
Colin Crossb916a382016-07-29 17:28:03 -0700224func (test *testDecorator) linkerProps() []interface{} {
225 return []interface{}{&test.Properties}
Colin Cross4d9c2d12016-07-29 12:48:20 -0700226}
227
Colin Crossb916a382016-07-29 17:28:03 -0700228func NewTestInstaller() *baseInstaller {
229 return NewBaseInstaller("nativetest", "nativetest64", InstallInData)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700230}
231
Colin Crossb916a382016-07-29 17:28:03 -0700232type testBinary struct {
233 testDecorator
234 *binaryDecorator
235 *baseCompiler
Colin Crossb916a382016-07-29 17:28:03 -0700236 Properties TestBinaryProperties
Colin Crossfaeb7aa2017-02-01 14:12:44 -0800237 data android.Paths
Colin Cross303e21f2018-08-07 16:49:25 -0700238 testConfig android.Path
Colin Crossb916a382016-07-29 17:28:03 -0700239}
240
241func (test *testBinary) linkerProps() []interface{} {
242 props := append(test.testDecorator.linkerProps(), test.binaryDecorator.linkerProps()...)
243 props = append(props, &test.Properties)
244 return props
245}
246
247func (test *testBinary) linkerInit(ctx BaseModuleContext) {
248 test.testDecorator.linkerInit(ctx, test.binaryDecorator.baseLinker)
249 test.binaryDecorator.linkerInit(ctx)
250}
251
Colin Cross37047f12016-12-13 17:06:13 -0800252func (test *testBinary) linkerDeps(ctx DepsContext, deps Deps) Deps {
Colin Crossb916a382016-07-29 17:28:03 -0700253 deps = test.testDecorator.linkerDeps(ctx, deps)
254 deps = test.binaryDecorator.linkerDeps(ctx, deps)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700255 return deps
256}
257
Colin Crossb916a382016-07-29 17:28:03 -0700258func (test *testBinary) linkerFlags(ctx ModuleContext, flags Flags) Flags {
259 flags = test.binaryDecorator.linkerFlags(ctx, flags)
260 flags = test.testDecorator.linkerFlags(ctx, flags)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700261 return flags
262}
263
Colin Crossb916a382016-07-29 17:28:03 -0700264func (test *testBinary) install(ctx ModuleContext, file android.Path) {
Colin Cross8a497952019-03-05 22:25:09 -0800265 test.data = android.PathsForModuleSrc(ctx, test.Properties.Data)
Julien Desprezeb7398e2019-02-28 08:45:28 -0800266 optionsMap := map[string]string{}
267 if Bool(test.testDecorator.Properties.Isolated) {
268 optionsMap["not-shardable"] = "true"
269 }
yelinhsieh9fc60402018-10-01 19:23:14 +0800270
271 if test.Properties.Test_options.Run_test_as != nil {
272 optionsMap["run-test-as"] = String(test.Properties.Test_options.Run_test_as)
273 }
274
Jack He33338892018-09-19 02:21:28 -0700275 test.testConfig = tradefed.AutoGenNativeTestConfig(ctx, test.Properties.Test_config,
Julien Desprezeb7398e2019-02-28 08:45:28 -0800276 test.Properties.Test_config_template,
277 test.Properties.Test_suites, optionsMap)
Colin Crossfaeb7aa2017-02-01 14:12:44 -0800278
Colin Cross600c9df2016-09-13 12:26:16 -0700279 test.binaryDecorator.baseInstaller.dir = "nativetest"
280 test.binaryDecorator.baseInstaller.dir64 = "nativetest64"
Dan Willemsen3340d602016-12-27 14:40:40 -0800281
282 if !Bool(test.Properties.No_named_install_directory) {
283 test.binaryDecorator.baseInstaller.relative = ctx.ModuleName()
Nan Zhang0007d812017-11-07 10:57:05 -0800284 } else if String(test.binaryDecorator.baseInstaller.Properties.Relative_install_path) == "" {
Dan Willemsen3340d602016-12-27 14:40:40 -0800285 ctx.PropertyErrorf("no_named_install_directory", "Module install directory may only be disabled if relative_install_path is set")
286 }
287
Dan Willemsen1d577e22016-08-29 15:53:15 -0700288 test.binaryDecorator.baseInstaller.install(ctx, file)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700289}
290
291func NewTest(hod android.HostOrDeviceSupported) *Module {
Colin Crossb916a382016-07-29 17:28:03 -0700292 module, binary := NewBinary(hod)
293 module.multilib = android.MultilibBoth
Dan Willemsen1d577e22016-08-29 15:53:15 -0700294 binary.baseInstaller = NewTestInstaller()
Colin Crossb916a382016-07-29 17:28:03 -0700295
296 test := &testBinary{
297 testDecorator: testDecorator{
298 linker: binary.baseLinker,
Colin Cross4d9c2d12016-07-29 12:48:20 -0700299 },
Colin Crossb916a382016-07-29 17:28:03 -0700300 binaryDecorator: binary,
301 baseCompiler: NewBaseCompiler(),
Colin Cross4d9c2d12016-07-29 12:48:20 -0700302 }
Colin Crossb916a382016-07-29 17:28:03 -0700303 module.compiler = test
304 module.linker = test
305 module.installer = test
Colin Cross4d9c2d12016-07-29 12:48:20 -0700306 return module
307}
308
Colin Crossb916a382016-07-29 17:28:03 -0700309type testLibrary struct {
310 testDecorator
311 *libraryDecorator
312}
313
314func (test *testLibrary) linkerProps() []interface{} {
315 return append(test.testDecorator.linkerProps(), test.libraryDecorator.linkerProps()...)
316}
317
318func (test *testLibrary) linkerInit(ctx BaseModuleContext) {
319 test.testDecorator.linkerInit(ctx, test.libraryDecorator.baseLinker)
320 test.libraryDecorator.linkerInit(ctx)
321}
322
Colin Cross37047f12016-12-13 17:06:13 -0800323func (test *testLibrary) linkerDeps(ctx DepsContext, deps Deps) Deps {
Colin Crossb916a382016-07-29 17:28:03 -0700324 deps = test.testDecorator.linkerDeps(ctx, deps)
325 deps = test.libraryDecorator.linkerDeps(ctx, deps)
326 return deps
327}
328
329func (test *testLibrary) linkerFlags(ctx ModuleContext, flags Flags) Flags {
330 flags = test.libraryDecorator.linkerFlags(ctx, flags)
331 flags = test.testDecorator.linkerFlags(ctx, flags)
332 return flags
333}
334
Colin Cross4d9c2d12016-07-29 12:48:20 -0700335func NewTestLibrary(hod android.HostOrDeviceSupported) *Module {
Colin Crossab3b7322016-12-09 14:46:15 -0800336 module, library := NewLibrary(android.HostAndDeviceSupported)
Dan Willemsen28bda512016-08-31 16:32:55 -0700337 library.baseInstaller = NewTestInstaller()
Colin Crossb916a382016-07-29 17:28:03 -0700338 test := &testLibrary{
339 testDecorator: testDecorator{
340 linker: library.baseLinker,
Colin Cross4d9c2d12016-07-29 12:48:20 -0700341 },
Colin Crossb916a382016-07-29 17:28:03 -0700342 libraryDecorator: library,
Colin Cross4d9c2d12016-07-29 12:48:20 -0700343 }
Colin Crossb916a382016-07-29 17:28:03 -0700344 module.linker = test
Colin Cross4d9c2d12016-07-29 12:48:20 -0700345 return module
346}
347
Colin Crosse28f4e22017-04-24 18:10:29 -0700348type BenchmarkProperties struct {
Anders Lewisb97e8182017-07-14 15:20:13 -0700349 // list of files or filegroup modules that provide data that should be installed alongside
350 // the test
Colin Cross27b922f2019-03-04 22:35:41 -0800351 Data []string `android:"path"`
Anders Lewisb97e8182017-07-14 15:20:13 -0700352
Colin Crosse28f4e22017-04-24 18:10:29 -0700353 // list of compatibility suites (for example "cts", "vts") that the module should be
354 // installed into.
Julien Despreze146e392018-08-02 15:00:46 -0700355 Test_suites []string `android:"arch_variant"`
356
357 // the name of the test configuration (for example "AndroidTest.xml") that should be
358 // installed with the module.
Colin Cross27b922f2019-03-04 22:35:41 -0800359 Test_config *string `android:"path,arch_variant"`
Jack He33338892018-09-19 02:21:28 -0700360
361 // the name of the test configuration template (for example "AndroidTestTemplate.xml") that
362 // should be installed with the module.
Colin Cross27b922f2019-03-04 22:35:41 -0800363 Test_config_template *string `android:"path,arch_variant"`
Colin Crosse28f4e22017-04-24 18:10:29 -0700364}
365
Colin Crossb916a382016-07-29 17:28:03 -0700366type benchmarkDecorator struct {
367 *binaryDecorator
Colin Crosse28f4e22017-04-24 18:10:29 -0700368 Properties BenchmarkProperties
Anders Lewisb97e8182017-07-14 15:20:13 -0700369 data android.Paths
Colin Cross303e21f2018-08-07 16:49:25 -0700370 testConfig android.Path
Colin Cross4d9c2d12016-07-29 12:48:20 -0700371}
372
Colin Crossb916a382016-07-29 17:28:03 -0700373func (benchmark *benchmarkDecorator) linkerInit(ctx BaseModuleContext) {
374 runpath := "../../lib"
375 if ctx.toolchain().Is64Bit() {
376 runpath += "64"
377 }
378 benchmark.baseLinker.dynamicProperties.RunPaths = append(benchmark.baseLinker.dynamicProperties.RunPaths, runpath)
379 benchmark.binaryDecorator.linkerInit(ctx)
380}
381
Colin Crosse28f4e22017-04-24 18:10:29 -0700382func (benchmark *benchmarkDecorator) linkerProps() []interface{} {
383 props := benchmark.binaryDecorator.linkerProps()
384 props = append(props, &benchmark.Properties)
385 return props
386}
387
Colin Cross37047f12016-12-13 17:06:13 -0800388func (benchmark *benchmarkDecorator) linkerDeps(ctx DepsContext, deps Deps) Deps {
Colin Crossb916a382016-07-29 17:28:03 -0700389 deps = benchmark.binaryDecorator.linkerDeps(ctx, deps)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700390 deps.StaticLibs = append(deps.StaticLibs, "libgoogle-benchmark")
391 return deps
392}
393
Colin Crossb916a382016-07-29 17:28:03 -0700394func (benchmark *benchmarkDecorator) install(ctx ModuleContext, file android.Path) {
Colin Cross8a497952019-03-05 22:25:09 -0800395 benchmark.data = android.PathsForModuleSrc(ctx, benchmark.Properties.Data)
Jack He33338892018-09-19 02:21:28 -0700396 benchmark.testConfig = tradefed.AutoGenNativeBenchmarkTestConfig(ctx, benchmark.Properties.Test_config,
yangbill4f41bc22019-02-13 21:45:47 +0800397 benchmark.Properties.Test_config_template, benchmark.Properties.Test_suites)
Colin Cross303e21f2018-08-07 16:49:25 -0700398
Colin Cross28690e92017-09-08 16:20:30 -0700399 benchmark.binaryDecorator.baseInstaller.dir = filepath.Join("benchmarktest", ctx.ModuleName())
400 benchmark.binaryDecorator.baseInstaller.dir64 = filepath.Join("benchmarktest64", ctx.ModuleName())
Dan Willemsen1d577e22016-08-29 15:53:15 -0700401 benchmark.binaryDecorator.baseInstaller.install(ctx, file)
Colin Crossb916a382016-07-29 17:28:03 -0700402}
403
Colin Cross4d9c2d12016-07-29 12:48:20 -0700404func NewBenchmark(hod android.HostOrDeviceSupported) *Module {
Colin Crossb916a382016-07-29 17:28:03 -0700405 module, binary := NewBinary(hod)
406 module.multilib = android.MultilibBoth
Colin Cross28690e92017-09-08 16:20:30 -0700407 binary.baseInstaller = NewBaseInstaller("benchmarktest", "benchmarktest64", InstallInData)
Colin Crossb916a382016-07-29 17:28:03 -0700408
409 benchmark := &benchmarkDecorator{
410 binaryDecorator: binary,
Colin Cross4d9c2d12016-07-29 12:48:20 -0700411 }
Colin Crossb916a382016-07-29 17:28:03 -0700412 module.linker = benchmark
413 module.installer = benchmark
Colin Cross4d9c2d12016-07-29 12:48:20 -0700414 return module
415}