1// Copyright 2017 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	"reflect"
19	"testing"
20
21	"android/soong/android"
22)
23
24func TestLibraryReuse(t *testing.T) {
25	t.Run("simple", func(t *testing.T) {
26		ctx := testCc(t, `
27		cc_library {
28			name: "libfoo",
29			srcs: ["foo.c", "baz.o"],
30		}`)
31
32		libfooShared := ctx.ModuleForTests("libfoo", "android_arm_armv7-a-neon_shared").Rule("ld")
33		libfooStatic := ctx.ModuleForTests("libfoo", "android_arm_armv7-a-neon_static").Output("libfoo.a")
34
35		if len(libfooShared.Inputs) != 2 {
36			t.Fatalf("unexpected inputs to libfoo shared: %#v", libfooShared.Inputs.Strings())
37		}
38
39		if len(libfooStatic.Inputs) != 2 {
40			t.Fatalf("unexpected inputs to libfoo static: %#v", libfooStatic.Inputs.Strings())
41		}
42
43		if libfooShared.Inputs[0] != libfooStatic.Inputs[0] {
44			t.Errorf("static object not reused for shared library")
45		}
46		if libfooShared.Inputs[1] != libfooStatic.Inputs[1] {
47			t.Errorf("static object not reused for shared library")
48		}
49	})
50
51	t.Run("extra static source", func(t *testing.T) {
52		ctx := testCc(t, `
53		cc_library {
54			name: "libfoo",
55			srcs: ["foo.c"],
56			static: {
57				srcs: ["bar.c"]
58			},
59		}`)
60
61		libfooShared := ctx.ModuleForTests("libfoo", "android_arm_armv7-a-neon_shared").Rule("ld")
62		libfooStatic := ctx.ModuleForTests("libfoo", "android_arm_armv7-a-neon_static").Output("libfoo.a")
63
64		if len(libfooShared.Inputs) != 1 {
65			t.Fatalf("unexpected inputs to libfoo shared: %#v", libfooShared.Inputs.Strings())
66		}
67
68		if len(libfooStatic.Inputs) != 2 {
69			t.Fatalf("unexpected inputs to libfoo static: %#v", libfooStatic.Inputs.Strings())
70		}
71
72		if libfooShared.Inputs[0] != libfooStatic.Inputs[0] {
73			t.Errorf("static object not reused for shared library")
74		}
75	})
76
77	t.Run("extra shared source", func(t *testing.T) {
78		ctx := testCc(t, `
79		cc_library {
80			name: "libfoo",
81			srcs: ["foo.c"],
82			shared: {
83				srcs: ["bar.c"]
84			},
85		}`)
86
87		libfooShared := ctx.ModuleForTests("libfoo", "android_arm_armv7-a-neon_shared").Rule("ld")
88		libfooStatic := ctx.ModuleForTests("libfoo", "android_arm_armv7-a-neon_static").Output("libfoo.a")
89
90		if len(libfooShared.Inputs) != 2 {
91			t.Fatalf("unexpected inputs to libfoo shared: %#v", libfooShared.Inputs.Strings())
92		}
93
94		if len(libfooStatic.Inputs) != 1 {
95			t.Fatalf("unexpected inputs to libfoo static: %#v", libfooStatic.Inputs.Strings())
96		}
97
98		if libfooShared.Inputs[0] != libfooStatic.Inputs[0] {
99			t.Errorf("static object not reused for shared library")
100		}
101	})
102
103	t.Run("extra static cflags", func(t *testing.T) {
104		ctx := testCc(t, `
105		cc_library {
106			name: "libfoo",
107			srcs: ["foo.c"],
108			static: {
109				cflags: ["-DFOO"],
110			},
111		}`)
112
113		libfooShared := ctx.ModuleForTests("libfoo", "android_arm_armv7-a-neon_shared").Rule("ld")
114		libfooStatic := ctx.ModuleForTests("libfoo", "android_arm_armv7-a-neon_static").Output("libfoo.a")
115
116		if len(libfooShared.Inputs) != 1 {
117			t.Fatalf("unexpected inputs to libfoo shared: %#v", libfooShared.Inputs.Strings())
118		}
119
120		if len(libfooStatic.Inputs) != 1 {
121			t.Fatalf("unexpected inputs to libfoo static: %#v", libfooStatic.Inputs.Strings())
122		}
123
124		if libfooShared.Inputs[0] == libfooStatic.Inputs[0] {
125			t.Errorf("static object reused for shared library when it shouldn't be")
126		}
127	})
128
129	t.Run("extra shared cflags", func(t *testing.T) {
130		ctx := testCc(t, `
131		cc_library {
132			name: "libfoo",
133			srcs: ["foo.c"],
134			shared: {
135				cflags: ["-DFOO"],
136			},
137		}`)
138
139		libfooShared := ctx.ModuleForTests("libfoo", "android_arm_armv7-a-neon_shared").Rule("ld")
140		libfooStatic := ctx.ModuleForTests("libfoo", "android_arm_armv7-a-neon_static").Output("libfoo.a")
141
142		if len(libfooShared.Inputs) != 1 {
143			t.Fatalf("unexpected inputs to libfoo shared: %#v", libfooShared.Inputs.Strings())
144		}
145
146		if len(libfooStatic.Inputs) != 1 {
147			t.Fatalf("unexpected inputs to libfoo static: %#v", libfooStatic.Inputs.Strings())
148		}
149
150		if libfooShared.Inputs[0] == libfooStatic.Inputs[0] {
151			t.Errorf("static object reused for shared library when it shouldn't be")
152		}
153	})
154
155	t.Run("global cflags for reused generated sources", func(t *testing.T) {
156		ctx := testCc(t, `
157		cc_library {
158			name: "libfoo",
159			srcs: [
160				"foo.c",
161				"a.proto",
162			],
163			shared: {
164				srcs: [
165					"bar.c",
166				],
167			},
168		}`)
169
170		libfooShared := ctx.ModuleForTests("libfoo", "android_arm_armv7-a-neon_shared").Rule("ld")
171		libfooStatic := ctx.ModuleForTests("libfoo", "android_arm_armv7-a-neon_static").Output("libfoo.a")
172
173		if len(libfooShared.Inputs) != 3 {
174			t.Fatalf("unexpected inputs to libfoo shared: %#v", libfooShared.Inputs.Strings())
175		}
176
177		if len(libfooStatic.Inputs) != 2 {
178			t.Fatalf("unexpected inputs to libfoo static: %#v", libfooStatic.Inputs.Strings())
179		}
180
181		if !reflect.DeepEqual(libfooShared.Inputs[0:2].Strings(), libfooStatic.Inputs.Strings()) {
182			t.Errorf("static objects not reused for shared library")
183		}
184
185		libfoo := ctx.ModuleForTests("libfoo", "android_arm_armv7-a-neon_shared").Module().(*Module)
186		if !inList("-DGOOGLE_PROTOBUF_NO_RTTI", libfoo.flags.Local.CFlags) {
187			t.Errorf("missing protobuf cflags")
188		}
189	})
190}
191
192func TestStubsVersions(t *testing.T) {
193	bp := `
194		cc_library {
195			name: "libfoo",
196			srcs: ["foo.c"],
197			stubs: {
198				versions: ["29", "R", "current"],
199			},
200		}
201	`
202	config := TestConfig(t.TempDir(), android.Android, nil, bp, nil)
203	config.TestProductVariables.Platform_version_active_codenames = []string{"R"}
204	ctx := testCcWithConfig(t, config)
205
206	variants := ctx.ModuleVariantsForTests("libfoo")
207	for _, expectedVer := range []string{"29", "R", "current"} {
208		expectedVariant := "android_arm_armv7-a-neon_shared_" + expectedVer
209		if !inList(expectedVariant, variants) {
210			t.Errorf("missing expected variant: %q", expectedVariant)
211		}
212	}
213}
214
215func TestStubsVersions_NotSorted(t *testing.T) {
216	bp := `
217		cc_library {
218			name: "libfoo",
219			srcs: ["foo.c"],
220			stubs: {
221				versions: ["29", "current", "R"],
222			},
223		}
224	`
225	config := TestConfig(t.TempDir(), android.Android, nil, bp, nil)
226	config.TestProductVariables.Platform_version_active_codenames = []string{"R"}
227	testCcErrorWithConfig(t, `"libfoo" .*: versions: not sorted`, config)
228}
229
230func TestStubsVersions_ParseError(t *testing.T) {
231	bp := `
232		cc_library {
233			name: "libfoo",
234			srcs: ["foo.c"],
235			stubs: {
236				versions: ["29", "current", "X"],
237			},
238		}
239	`
240
241	testCcError(t, `"libfoo" .*: versions: "X" could not be parsed as an integer and is not a recognized codename`, bp)
242}
243