| // Copyright 2018 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 etc |
| |
| import ( |
| "io/ioutil" |
| "os" |
| "path/filepath" |
| "reflect" |
| "testing" |
| |
| "android/soong/android" |
| ) |
| |
| var buildDir string |
| |
| func setUp() { |
| var err error |
| buildDir, err = ioutil.TempDir("", "soong_etc_test") |
| if err != nil { |
| panic(err) |
| } |
| } |
| |
| func tearDown() { |
| os.RemoveAll(buildDir) |
| } |
| |
| func TestMain(m *testing.M) { |
| run := func() int { |
| setUp() |
| defer tearDown() |
| |
| return m.Run() |
| } |
| |
| os.Exit(run()) |
| } |
| |
| func testPrebuiltEtcContext(t *testing.T, bp string) (*android.TestContext, android.Config) { |
| fs := map[string][]byte{ |
| "foo.conf": nil, |
| "bar.conf": nil, |
| "baz.conf": nil, |
| } |
| |
| config := android.TestArchConfig(buildDir, nil, bp, fs) |
| |
| ctx := android.NewTestArchContext() |
| ctx.RegisterModuleType("prebuilt_etc", PrebuiltEtcFactory) |
| ctx.RegisterModuleType("prebuilt_etc_host", PrebuiltEtcHostFactory) |
| ctx.RegisterModuleType("prebuilt_usr_share", PrebuiltUserShareFactory) |
| ctx.RegisterModuleType("prebuilt_usr_share_host", PrebuiltUserShareHostFactory) |
| ctx.RegisterModuleType("prebuilt_font", PrebuiltFontFactory) |
| ctx.RegisterModuleType("prebuilt_firmware", PrebuiltFirmwareFactory) |
| ctx.RegisterModuleType("prebuilt_dsp", PrebuiltDSPFactory) |
| ctx.Register(config) |
| |
| return ctx, config |
| } |
| |
| func testPrebuiltEtc(t *testing.T, bp string) (*android.TestContext, android.Config) { |
| t.Helper() |
| |
| ctx, config := testPrebuiltEtcContext(t, bp) |
| _, errs := ctx.ParseFileList(".", []string{"Android.bp"}) |
| android.FailIfErrored(t, errs) |
| _, errs = ctx.PrepareBuildActions(config) |
| android.FailIfErrored(t, errs) |
| |
| return ctx, config |
| } |
| |
| func testPrebuiltEtcError(t *testing.T, pattern, bp string) { |
| t.Helper() |
| |
| ctx, config := testPrebuiltEtcContext(t, bp) |
| _, errs := ctx.ParseFileList(".", []string{"Android.bp"}) |
| if len(errs) > 0 { |
| android.FailIfNoMatchingErrors(t, pattern, errs) |
| return |
| } |
| |
| _, errs = ctx.PrepareBuildActions(config) |
| if len(errs) > 0 { |
| android.FailIfNoMatchingErrors(t, pattern, errs) |
| return |
| } |
| |
| t.Fatalf("missing expected error %q (0 errors are returned)", pattern) |
| } |
| func TestPrebuiltEtcVariants(t *testing.T) { |
| ctx, _ := testPrebuiltEtc(t, ` |
| prebuilt_etc { |
| name: "foo.conf", |
| src: "foo.conf", |
| } |
| prebuilt_etc { |
| name: "bar.conf", |
| src: "bar.conf", |
| recovery_available: true, |
| } |
| prebuilt_etc { |
| name: "baz.conf", |
| src: "baz.conf", |
| recovery: true, |
| } |
| `) |
| |
| foo_variants := ctx.ModuleVariantsForTests("foo.conf") |
| if len(foo_variants) != 1 { |
| t.Errorf("expected 1, got %#v", foo_variants) |
| } |
| |
| bar_variants := ctx.ModuleVariantsForTests("bar.conf") |
| if len(bar_variants) != 2 { |
| t.Errorf("expected 2, got %#v", bar_variants) |
| } |
| |
| baz_variants := ctx.ModuleVariantsForTests("baz.conf") |
| if len(baz_variants) != 1 { |
| t.Errorf("expected 1, got %#v", bar_variants) |
| } |
| } |
| |
| func TestPrebuiltEtcOutputPath(t *testing.T) { |
| ctx, _ := testPrebuiltEtc(t, ` |
| prebuilt_etc { |
| name: "foo.conf", |
| src: "foo.conf", |
| filename: "foo.installed.conf", |
| } |
| `) |
| |
| p := ctx.ModuleForTests("foo.conf", "android_arm64_armv8-a").Module().(*PrebuiltEtc) |
| if p.outputFilePath.Base() != "foo.installed.conf" { |
| t.Errorf("expected foo.installed.conf, got %q", p.outputFilePath.Base()) |
| } |
| } |
| |
| func TestPrebuiltEtcGlob(t *testing.T) { |
| ctx, _ := testPrebuiltEtc(t, ` |
| prebuilt_etc { |
| name: "my_foo", |
| src: "foo.*", |
| } |
| prebuilt_etc { |
| name: "my_bar", |
| src: "bar.*", |
| filename_from_src: true, |
| } |
| `) |
| |
| p := ctx.ModuleForTests("my_foo", "android_arm64_armv8-a").Module().(*PrebuiltEtc) |
| if p.outputFilePath.Base() != "my_foo" { |
| t.Errorf("expected my_foo, got %q", p.outputFilePath.Base()) |
| } |
| |
| p = ctx.ModuleForTests("my_bar", "android_arm64_armv8-a").Module().(*PrebuiltEtc) |
| if p.outputFilePath.Base() != "bar.conf" { |
| t.Errorf("expected bar.conf, got %q", p.outputFilePath.Base()) |
| } |
| } |
| |
| func TestPrebuiltEtcAndroidMk(t *testing.T) { |
| ctx, config := testPrebuiltEtc(t, ` |
| prebuilt_etc { |
| name: "foo", |
| src: "foo.conf", |
| owner: "abc", |
| filename_from_src: true, |
| required: ["modA", "moduleB"], |
| host_required: ["hostModA", "hostModB"], |
| target_required: ["targetModA"], |
| } |
| `) |
| |
| expected := map[string][]string{ |
| "LOCAL_MODULE": {"foo"}, |
| "LOCAL_MODULE_CLASS": {"ETC"}, |
| "LOCAL_MODULE_OWNER": {"abc"}, |
| "LOCAL_INSTALLED_MODULE_STEM": {"foo.conf"}, |
| "LOCAL_REQUIRED_MODULES": {"modA", "moduleB"}, |
| "LOCAL_HOST_REQUIRED_MODULES": {"hostModA", "hostModB"}, |
| "LOCAL_TARGET_REQUIRED_MODULES": {"targetModA"}, |
| } |
| |
| mod := ctx.ModuleForTests("foo", "android_arm64_armv8-a").Module().(*PrebuiltEtc) |
| entries := android.AndroidMkEntriesForTest(t, config, "", mod)[0] |
| for k, expectedValue := range expected { |
| if value, ok := entries.EntryMap[k]; ok { |
| if !reflect.DeepEqual(value, expectedValue) { |
| t.Errorf("Incorrect %s '%s', expected '%s'", k, value, expectedValue) |
| } |
| } else { |
| t.Errorf("No %s defined, saw %q", k, entries.EntryMap) |
| } |
| } |
| } |
| |
| func TestPrebuiltEtcRelativeInstallPathInstallDirPath(t *testing.T) { |
| ctx, _ := testPrebuiltEtc(t, ` |
| prebuilt_etc { |
| name: "foo.conf", |
| src: "foo.conf", |
| relative_install_path: "bar", |
| } |
| `) |
| |
| p := ctx.ModuleForTests("foo.conf", "android_arm64_armv8-a").Module().(*PrebuiltEtc) |
| expected := buildDir + "/target/product/test_device/system/etc/bar" |
| if p.installDirPath.String() != expected { |
| t.Errorf("expected %q, got %q", expected, p.installDirPath.String()) |
| } |
| } |
| |
| func TestPrebuiltEtcCannotSetRelativeInstallPathAndSubDir(t *testing.T) { |
| testPrebuiltEtcError(t, "relative_install_path is set. Cannot set sub_dir", ` |
| prebuilt_etc { |
| name: "foo.conf", |
| src: "foo.conf", |
| sub_dir: "bar", |
| relative_install_path: "bar", |
| } |
| `) |
| } |
| |
| func TestPrebuiltEtcHost(t *testing.T) { |
| ctx, _ := testPrebuiltEtc(t, ` |
| prebuilt_etc_host { |
| name: "foo.conf", |
| src: "foo.conf", |
| } |
| `) |
| |
| buildOS := android.BuildOs.String() |
| p := ctx.ModuleForTests("foo.conf", buildOS+"_common").Module().(*PrebuiltEtc) |
| if !p.Host() { |
| t.Errorf("host bit is not set for a prebuilt_etc_host module.") |
| } |
| } |
| |
| func TestPrebuiltUserShareInstallDirPath(t *testing.T) { |
| ctx, _ := testPrebuiltEtc(t, ` |
| prebuilt_usr_share { |
| name: "foo.conf", |
| src: "foo.conf", |
| sub_dir: "bar", |
| } |
| `) |
| |
| p := ctx.ModuleForTests("foo.conf", "android_arm64_armv8-a").Module().(*PrebuiltEtc) |
| expected := buildDir + "/target/product/test_device/system/usr/share/bar" |
| if p.installDirPath.String() != expected { |
| t.Errorf("expected %q, got %q", expected, p.installDirPath.String()) |
| } |
| } |
| |
| func TestPrebuiltUserShareHostInstallDirPath(t *testing.T) { |
| ctx, config := testPrebuiltEtc(t, ` |
| prebuilt_usr_share_host { |
| name: "foo.conf", |
| src: "foo.conf", |
| sub_dir: "bar", |
| } |
| `) |
| |
| buildOS := android.BuildOs.String() |
| p := ctx.ModuleForTests("foo.conf", buildOS+"_common").Module().(*PrebuiltEtc) |
| expected := filepath.Join(buildDir, "host", config.PrebuiltOS(), "usr", "share", "bar") |
| if p.installDirPath.String() != expected { |
| t.Errorf("expected %q, got %q", expected, p.installDirPath.String()) |
| } |
| } |
| |
| func TestPrebuiltFontInstallDirPath(t *testing.T) { |
| ctx, _ := testPrebuiltEtc(t, ` |
| prebuilt_font { |
| name: "foo.conf", |
| src: "foo.conf", |
| } |
| `) |
| |
| p := ctx.ModuleForTests("foo.conf", "android_arm64_armv8-a").Module().(*PrebuiltEtc) |
| expected := buildDir + "/target/product/test_device/system/fonts" |
| if p.installDirPath.String() != expected { |
| t.Errorf("expected %q, got %q", expected, p.installDirPath.String()) |
| } |
| } |
| |
| func TestPrebuiltFirmwareDirPath(t *testing.T) { |
| targetPath := buildDir + "/target/product/test_device" |
| tests := []struct { |
| description string |
| config string |
| expectedPath string |
| }{{ |
| description: "prebuilt: system firmware", |
| config: ` |
| prebuilt_firmware { |
| name: "foo.conf", |
| src: "foo.conf", |
| }`, |
| expectedPath: filepath.Join(targetPath, "system/etc/firmware"), |
| }, { |
| description: "prebuilt: vendor firmware", |
| config: ` |
| prebuilt_firmware { |
| name: "foo.conf", |
| src: "foo.conf", |
| soc_specific: true, |
| sub_dir: "sub_dir", |
| }`, |
| expectedPath: filepath.Join(targetPath, "vendor/firmware/sub_dir"), |
| }} |
| for _, tt := range tests { |
| t.Run(tt.description, func(t *testing.T) { |
| ctx, _ := testPrebuiltEtc(t, tt.config) |
| p := ctx.ModuleForTests("foo.conf", "android_arm64_armv8-a").Module().(*PrebuiltEtc) |
| if p.installDirPath.String() != tt.expectedPath { |
| t.Errorf("expected %q, got %q", tt.expectedPath, p.installDirPath) |
| } |
| }) |
| } |
| } |
| |
| func TestPrebuiltDSPDirPath(t *testing.T) { |
| targetPath := filepath.Join(buildDir, "/target/product/test_device") |
| tests := []struct { |
| description string |
| config string |
| expectedPath string |
| }{{ |
| description: "prebuilt: system dsp", |
| config: ` |
| prebuilt_dsp { |
| name: "foo.conf", |
| src: "foo.conf", |
| }`, |
| expectedPath: filepath.Join(targetPath, "system/etc/dsp"), |
| }, { |
| description: "prebuilt: vendor dsp", |
| config: ` |
| prebuilt_dsp { |
| name: "foo.conf", |
| src: "foo.conf", |
| soc_specific: true, |
| sub_dir: "sub_dir", |
| }`, |
| expectedPath: filepath.Join(targetPath, "vendor/dsp/sub_dir"), |
| }} |
| for _, tt := range tests { |
| t.Run(tt.description, func(t *testing.T) { |
| ctx, _ := testPrebuiltEtc(t, tt.config) |
| p := ctx.ModuleForTests("foo.conf", "android_arm64_armv8-a").Module().(*PrebuiltEtc) |
| if p.installDirPath.String() != tt.expectedPath { |
| t.Errorf("expected %q, got %q", tt.expectedPath, p.installDirPath) |
| } |
| }) |
| } |
| } |