|  | // Copyright 2019 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 cc | 
|  |  | 
|  | import ( | 
|  | "fmt" | 
|  | "testing" | 
|  |  | 
|  | "android/soong/android" | 
|  | ) | 
|  |  | 
|  | func TestMinSdkVersionsOfCrtObjects(t *testing.T) { | 
|  | ctx := testCc(t, ` | 
|  | cc_object { | 
|  | name: "crt_foo", | 
|  | srcs: ["foo.c"], | 
|  | crt: true, | 
|  | stl: "none", | 
|  | min_sdk_version: "28", | 
|  | vendor_available: true, | 
|  | }`) | 
|  |  | 
|  | variants := []struct { | 
|  | variant string | 
|  | num     string | 
|  | }{ | 
|  | {"android_arm64_armv8-a", "10000"}, | 
|  | {"android_arm64_armv8-a_sdk_28", "28"}, | 
|  | {"android_arm64_armv8-a_sdk_29", "29"}, | 
|  | {"android_arm64_armv8-a_sdk_30", "30"}, | 
|  | {"android_arm64_armv8-a_sdk_current", "10000"}, | 
|  | {"android_vendor.29_arm64_armv8-a", "29"}, | 
|  | } | 
|  | for _, v := range variants { | 
|  | cflags := ctx.ModuleForTests("crt_foo", v.variant).Rule("cc").Args["cFlags"] | 
|  | expected := "-target aarch64-linux-android" + v.num + " " | 
|  | android.AssertStringDoesContain(t, "cflag", cflags, expected) | 
|  | } | 
|  | } | 
|  |  | 
|  | func TestUseCrtObjectOfCorrectVersion(t *testing.T) { | 
|  | ctx := testCc(t, ` | 
|  | cc_binary { | 
|  | name: "bin", | 
|  | srcs: ["foo.c"], | 
|  | stl: "none", | 
|  | min_sdk_version: "29", | 
|  | sdk_version: "current", | 
|  | } | 
|  | `) | 
|  |  | 
|  | // Sdk variant uses the crt object of the matching min_sdk_version | 
|  | variant := "android_arm64_armv8-a_sdk" | 
|  | crt := ctx.ModuleForTests("bin", variant).Rule("ld").Args["crtBegin"] | 
|  | android.AssertStringDoesContain(t, "crt dep of sdk variant", crt, | 
|  | "29/crtbegin_dynamic.o") | 
|  |  | 
|  | // platform variant uses the crt object built for platform | 
|  | variant = "android_arm64_armv8-a" | 
|  | crt = ctx.ModuleForTests("bin", variant).Rule("ld").Args["crtBegin"] | 
|  | android.AssertStringDoesContain(t, "crt dep of platform variant", crt, | 
|  | variant+"/crtbegin_dynamic.o") | 
|  | } | 
|  |  | 
|  | func TestLinkerScript(t *testing.T) { | 
|  | t.Run("script", func(t *testing.T) { | 
|  | testCc(t, ` | 
|  | cc_object { | 
|  | name: "foo", | 
|  | srcs: ["baz.o"], | 
|  | linker_script: "foo.lds", | 
|  | }`) | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestCcObjectWithBazel(t *testing.T) { | 
|  | bp := ` | 
|  | cc_object { | 
|  | name: "foo", | 
|  | srcs: ["baz.o"], | 
|  | bazel_module: { label: "//foo/bar:bar" }, | 
|  | }` | 
|  | config := TestConfig(t.TempDir(), android.Android, nil, bp, nil) | 
|  | config.BazelContext = android.MockBazelContext{ | 
|  | OutputBaseDir: "outputbase", | 
|  | LabelToOutputFiles: map[string][]string{ | 
|  | "//foo/bar:bar": []string{"bazel_out.o"}}} | 
|  | ctx := testCcWithConfig(t, config) | 
|  |  | 
|  | module := ctx.ModuleForTests("foo", "android_arm_armv7-a-neon").Module() | 
|  | outputFiles, err := module.(android.OutputFileProducer).OutputFiles("") | 
|  | if err != nil { | 
|  | t.Errorf("Unexpected error getting cc_object outputfiles %s", err) | 
|  | } | 
|  |  | 
|  | expectedOutputFiles := []string{"outputbase/execroot/__main__/bazel_out.o"} | 
|  | android.AssertDeepEquals(t, "output files", expectedOutputFiles, outputFiles.Strings()) | 
|  | } | 
|  |  | 
|  | func TestCcObjectOutputFile(t *testing.T) { | 
|  | testcases := []struct { | 
|  | name       string | 
|  | moduleName string | 
|  | bp         string | 
|  | }{ | 
|  | { | 
|  | name:       "normal", | 
|  | moduleName: "foo", | 
|  | bp: ` | 
|  | srcs: ["bar.c"], | 
|  | `, | 
|  | }, | 
|  | { | 
|  | name:       "suffix", | 
|  | moduleName: "foo.o", | 
|  | bp: ` | 
|  | srcs: ["bar.c"], | 
|  | `, | 
|  | }, | 
|  | { | 
|  | name:       "keep symbols", | 
|  | moduleName: "foo", | 
|  | bp: ` | 
|  | srcs: ["bar.c"], | 
|  | prefix_symbols: "foo_", | 
|  | `, | 
|  | }, | 
|  | { | 
|  | name:       "partial linking", | 
|  | moduleName: "foo", | 
|  | bp: ` | 
|  | srcs: ["bar.c", "baz.c"], | 
|  | `, | 
|  | }, | 
|  | { | 
|  | name:       "partial linking and prefix symbols", | 
|  | moduleName: "foo", | 
|  | bp: ` | 
|  | srcs: ["bar.c", "baz.c"], | 
|  | prefix_symbols: "foo_", | 
|  | `, | 
|  | }, | 
|  | } | 
|  |  | 
|  | for _, testcase := range testcases { | 
|  | bp := fmt.Sprintf(` | 
|  | cc_object { | 
|  | name: "%s", | 
|  | %s | 
|  | } | 
|  | `, testcase.moduleName, testcase.bp) | 
|  | t.Run(testcase.name, func(t *testing.T) { | 
|  | ctx := PrepareForIntegrationTestWithCc.RunTestWithBp(t, bp) | 
|  | android.AssertPathRelativeToTopEquals(t, "expected output file foo.o", | 
|  | fmt.Sprintf("out/soong/.intermediates/%s/android_arm64_armv8-a/foo.o", testcase.moduleName), | 
|  | ctx.ModuleForTests(testcase.moduleName, "android_arm64_armv8-a").Output("foo.o").Output) | 
|  | }) | 
|  | } | 
|  |  | 
|  | } |