|  | // Copyright 2020 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 android | 
|  |  | 
|  | import ( | 
|  | "testing" | 
|  |  | 
|  | "github.com/google/blueprint" | 
|  | "github.com/google/blueprint/proptools" | 
|  | ) | 
|  |  | 
|  | // Module to be packaged | 
|  | type componentTestModule struct { | 
|  | ModuleBase | 
|  | props struct { | 
|  | Deps         []string | 
|  | Skip_install *bool | 
|  | } | 
|  | } | 
|  |  | 
|  | // dep tag used in this test. All dependencies are considered as installable. | 
|  | type installDepTag struct { | 
|  | blueprint.BaseDependencyTag | 
|  | InstallAlwaysNeededDependencyTag | 
|  | } | 
|  |  | 
|  | func componentTestModuleFactory() Module { | 
|  | m := &componentTestModule{} | 
|  | m.AddProperties(&m.props) | 
|  | InitAndroidArchModule(m, HostAndDeviceSupported, MultilibBoth) | 
|  | return m | 
|  | } | 
|  |  | 
|  | func (m *componentTestModule) DepsMutator(ctx BottomUpMutatorContext) { | 
|  | ctx.AddDependency(ctx.Module(), installDepTag{}, m.props.Deps...) | 
|  | } | 
|  |  | 
|  | func (m *componentTestModule) GenerateAndroidBuildActions(ctx ModuleContext) { | 
|  | builtFile := PathForModuleOut(ctx, m.Name()) | 
|  | dir := ctx.Target().Arch.ArchType.Multilib | 
|  | installDir := PathForModuleInstall(ctx, dir) | 
|  | if proptools.Bool(m.props.Skip_install) { | 
|  | m.SkipInstall() | 
|  | } | 
|  | ctx.InstallFile(installDir, m.Name(), builtFile) | 
|  | } | 
|  |  | 
|  | // Module that itself is a package | 
|  | type packageTestModule struct { | 
|  | ModuleBase | 
|  | PackagingBase | 
|  | properties struct { | 
|  | Install_deps []string `android:` | 
|  | } | 
|  | entries []string | 
|  | } | 
|  |  | 
|  | func packageMultiTargetTestModuleFactory() Module { | 
|  | module := &packageTestModule{} | 
|  | InitPackageModule(module) | 
|  | InitAndroidMultiTargetsArchModule(module, DeviceSupported, MultilibCommon) | 
|  | module.AddProperties(&module.properties) | 
|  | return module | 
|  | } | 
|  |  | 
|  | func packageTestModuleFactory() Module { | 
|  | module := &packageTestModule{} | 
|  | InitPackageModule(module) | 
|  | InitAndroidArchModule(module, DeviceSupported, MultilibBoth) | 
|  | module.AddProperties(&module.properties) | 
|  | return module | 
|  | } | 
|  |  | 
|  | type packagingDepTag struct { | 
|  | blueprint.BaseDependencyTag | 
|  | PackagingItemAlwaysDepTag | 
|  | } | 
|  |  | 
|  | func (m *packageTestModule) DepsMutator(ctx BottomUpMutatorContext) { | 
|  | m.AddDeps(ctx, packagingDepTag{}) | 
|  | ctx.AddDependency(ctx.Module(), installDepTag{}, m.properties.Install_deps...) | 
|  | } | 
|  |  | 
|  | func (m *packageTestModule) GenerateAndroidBuildActions(ctx ModuleContext) { | 
|  | zipFile := PathForModuleOut(ctx, "myzip.zip") | 
|  | m.entries = m.CopyDepsToZip(ctx, m.GatherPackagingSpecs(ctx), zipFile) | 
|  | } | 
|  |  | 
|  | func runPackagingTest(t *testing.T, multitarget bool, bp string, expected []string) { | 
|  | t.Helper() | 
|  |  | 
|  | var archVariant string | 
|  | var moduleFactory ModuleFactory | 
|  | if multitarget { | 
|  | archVariant = "android_common" | 
|  | moduleFactory = packageMultiTargetTestModuleFactory | 
|  | } else { | 
|  | archVariant = "android_arm64_armv8-a" | 
|  | moduleFactory = packageTestModuleFactory | 
|  | } | 
|  |  | 
|  | result := GroupFixturePreparers( | 
|  | PrepareForTestWithArchMutator, | 
|  | FixtureRegisterWithContext(func(ctx RegistrationContext) { | 
|  | ctx.RegisterModuleType("component", componentTestModuleFactory) | 
|  | ctx.RegisterModuleType("package_module", moduleFactory) | 
|  | }), | 
|  | FixtureWithRootAndroidBp(bp), | 
|  | ).RunTest(t) | 
|  |  | 
|  | p := result.Module("package", archVariant).(*packageTestModule) | 
|  | actual := p.entries | 
|  | actual = SortedUniqueStrings(actual) | 
|  | expected = SortedUniqueStrings(expected) | 
|  | AssertDeepEquals(t, "package entries", expected, actual) | 
|  | } | 
|  |  | 
|  | func TestPackagingBaseMultiTarget(t *testing.T) { | 
|  | multiTarget := true | 
|  | runPackagingTest(t, multiTarget, | 
|  | ` | 
|  | component { | 
|  | name: "foo", | 
|  | } | 
|  |  | 
|  | package_module { | 
|  | name: "package", | 
|  | deps: ["foo"], | 
|  | } | 
|  | `, []string{"lib64/foo"}) | 
|  |  | 
|  | runPackagingTest(t, multiTarget, | 
|  | ` | 
|  | component { | 
|  | name: "foo", | 
|  | deps: ["bar"], | 
|  | } | 
|  |  | 
|  | component { | 
|  | name: "bar", | 
|  | } | 
|  |  | 
|  | package_module { | 
|  | name: "package", | 
|  | deps: ["foo"], | 
|  | } | 
|  | `, []string{"lib64/foo", "lib64/bar"}) | 
|  |  | 
|  | runPackagingTest(t, multiTarget, | 
|  | ` | 
|  | component { | 
|  | name: "foo", | 
|  | deps: ["bar"], | 
|  | } | 
|  |  | 
|  | component { | 
|  | name: "bar", | 
|  | } | 
|  |  | 
|  | package_module { | 
|  | name: "package", | 
|  | deps: ["foo"], | 
|  | compile_multilib: "both", | 
|  | } | 
|  | `, []string{"lib32/foo", "lib32/bar", "lib64/foo", "lib64/bar"}) | 
|  |  | 
|  | runPackagingTest(t, multiTarget, | 
|  | ` | 
|  | component { | 
|  | name: "foo", | 
|  | } | 
|  |  | 
|  | component { | 
|  | name: "bar", | 
|  | compile_multilib: "32", | 
|  | } | 
|  |  | 
|  | package_module { | 
|  | name: "package", | 
|  | deps: ["foo"], | 
|  | multilib: { | 
|  | lib32: { | 
|  | deps: ["bar"], | 
|  | }, | 
|  | }, | 
|  | compile_multilib: "both", | 
|  | } | 
|  | `, []string{"lib32/foo", "lib32/bar", "lib64/foo"}) | 
|  |  | 
|  | runPackagingTest(t, multiTarget, | 
|  | ` | 
|  | component { | 
|  | name: "foo", | 
|  | } | 
|  |  | 
|  | component { | 
|  | name: "bar", | 
|  | } | 
|  |  | 
|  | package_module { | 
|  | name: "package", | 
|  | deps: ["foo"], | 
|  | multilib: { | 
|  | first: { | 
|  | deps: ["bar"], | 
|  | }, | 
|  | }, | 
|  | compile_multilib: "both", | 
|  | } | 
|  | `, []string{"lib32/foo", "lib64/foo", "lib64/bar"}) | 
|  |  | 
|  | runPackagingTest(t, multiTarget, | 
|  | ` | 
|  | component { | 
|  | name: "foo", | 
|  | } | 
|  |  | 
|  | component { | 
|  | name: "bar", | 
|  | } | 
|  |  | 
|  | component { | 
|  | name: "baz", | 
|  | } | 
|  |  | 
|  | package_module { | 
|  | name: "package", | 
|  | deps: ["foo"], | 
|  | arch: { | 
|  | arm64: { | 
|  | deps: ["bar"], | 
|  | }, | 
|  | x86_64: { | 
|  | deps: ["baz"], | 
|  | }, | 
|  | }, | 
|  | compile_multilib: "both", | 
|  | } | 
|  | `, []string{"lib32/foo", "lib64/foo", "lib64/bar"}) | 
|  | } | 
|  |  | 
|  | func TestPackagingBaseSingleTarget(t *testing.T) { | 
|  | multiTarget := false | 
|  | runPackagingTest(t, multiTarget, | 
|  | ` | 
|  | component { | 
|  | name: "foo", | 
|  | } | 
|  |  | 
|  | package_module { | 
|  | name: "package", | 
|  | deps: ["foo"], | 
|  | } | 
|  | `, []string{"lib64/foo"}) | 
|  |  | 
|  | runPackagingTest(t, multiTarget, | 
|  | ` | 
|  | component { | 
|  | name: "foo", | 
|  | deps: ["bar"], | 
|  | } | 
|  |  | 
|  | component { | 
|  | name: "bar", | 
|  | } | 
|  |  | 
|  | package_module { | 
|  | name: "package", | 
|  | deps: ["foo"], | 
|  | } | 
|  | `, []string{"lib64/foo", "lib64/bar"}) | 
|  |  | 
|  | runPackagingTest(t, multiTarget, | 
|  | ` | 
|  | component { | 
|  | name: "foo", | 
|  | } | 
|  |  | 
|  | component { | 
|  | name: "bar", | 
|  | compile_multilib: "32", | 
|  | } | 
|  |  | 
|  | package_module { | 
|  | name: "package", | 
|  | deps: ["foo"], | 
|  | multilib: { | 
|  | lib32: { | 
|  | deps: ["bar"], | 
|  | }, | 
|  | }, | 
|  | } | 
|  | `, []string{"lib64/foo"}) | 
|  |  | 
|  | runPackagingTest(t, multiTarget, | 
|  | ` | 
|  | component { | 
|  | name: "foo", | 
|  | } | 
|  |  | 
|  | component { | 
|  | name: "bar", | 
|  | } | 
|  |  | 
|  | package_module { | 
|  | name: "package", | 
|  | deps: ["foo"], | 
|  | multilib: { | 
|  | lib64: { | 
|  | deps: ["bar"], | 
|  | }, | 
|  | }, | 
|  | } | 
|  | `, []string{"lib64/foo", "lib64/bar"}) | 
|  |  | 
|  | runPackagingTest(t, multiTarget, | 
|  | ` | 
|  | component { | 
|  | name: "foo", | 
|  | } | 
|  |  | 
|  | component { | 
|  | name: "bar", | 
|  | } | 
|  |  | 
|  | component { | 
|  | name: "baz", | 
|  | } | 
|  |  | 
|  | package_module { | 
|  | name: "package", | 
|  | deps: ["foo"], | 
|  | arch: { | 
|  | arm64: { | 
|  | deps: ["bar"], | 
|  | }, | 
|  | x86_64: { | 
|  | deps: ["baz"], | 
|  | }, | 
|  | }, | 
|  | } | 
|  | `, []string{"lib64/foo", "lib64/bar"}) | 
|  |  | 
|  | runPackagingTest(t, multiTarget, | 
|  | ` | 
|  | component { | 
|  | name: "foo", | 
|  | } | 
|  |  | 
|  | component { | 
|  | name: "bar", | 
|  | } | 
|  |  | 
|  | package_module { | 
|  | name: "package", | 
|  | deps: ["foo"], | 
|  | install_deps: ["bar"], | 
|  | } | 
|  | `, []string{"lib64/foo"}) | 
|  | } | 
|  |  | 
|  | func TestPackagingWithSkipInstallDeps(t *testing.T) { | 
|  | // package -[dep]-> foo -[dep]-> bar      -[dep]-> baz | 
|  | // Packaging should continue transitively through modules that are not installed. | 
|  | multiTarget := false | 
|  | runPackagingTest(t, multiTarget, | 
|  | ` | 
|  | component { | 
|  | name: "foo", | 
|  | deps: ["bar"], | 
|  | } | 
|  |  | 
|  | component { | 
|  | name: "bar", | 
|  | deps: ["baz"], | 
|  | skip_install: true, | 
|  | } | 
|  |  | 
|  | component { | 
|  | name: "baz", | 
|  | } | 
|  |  | 
|  | package_module { | 
|  | name: "package", | 
|  | deps: ["foo"], | 
|  | } | 
|  | `, []string{"lib64/foo", "lib64/bar", "lib64/baz"}) | 
|  | } |