blob: 5578299183d626174980d911b00857fc678e4d45 [file] [log] [blame]
Colin Crossca860ac2016-01-04 14:34:37 -08001// 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 (
Colin Cross635c3b02016-05-18 15:37:25 -070018 "android/soong/android"
Colin Crossca860ac2016-01-04 14:34:37 -080019 "fmt"
Dan Albert90b9bbc2018-11-15 11:29:28 -080020 "strconv"
Colin Crossca860ac2016-01-04 14:34:37 -080021)
22
Colin Crossb60190a2018-09-04 16:28:17 -070023func getNdkStlFamily(m *Module) string {
24 family, _ := getNdkStlFamilyAndLinkType(m)
25 return family
26}
27
28func getNdkStlFamilyAndLinkType(m *Module) (string, string) {
Dan Albert202fe492017-12-15 13:56:59 -080029 stl := m.stl.Properties.SelectedStl
30 switch stl {
Colin Crossb60190a2018-09-04 16:28:17 -070031 case "ndk_libc++_shared":
32 return "libc++", "shared"
33 case "ndk_libc++_static":
34 return "libc++", "static"
Dan Albert202fe492017-12-15 13:56:59 -080035 case "ndk_system":
Colin Crossb60190a2018-09-04 16:28:17 -070036 return "system", "shared"
Dan Albert202fe492017-12-15 13:56:59 -080037 case "":
Colin Crossb60190a2018-09-04 16:28:17 -070038 return "none", "none"
Dan Albert202fe492017-12-15 13:56:59 -080039 default:
Colin Crossb60190a2018-09-04 16:28:17 -070040 panic(fmt.Errorf("stl: %q is not a valid STL", stl))
Dan Albert202fe492017-12-15 13:56:59 -080041 }
42}
43
Colin Crossca860ac2016-01-04 14:34:37 -080044type StlProperties struct {
Dan Albert749fc782018-01-04 14:21:14 -080045 // Select the STL library to use. Possible values are "libc++",
46 // "libc++_static", "libstdc++", or "none". Leave blank to select the
47 // default.
Colin Cross245ced72017-07-20 16:57:46 -070048 Stl *string `android:"arch_variant"`
Colin Crossca860ac2016-01-04 14:34:37 -080049
50 SelectedStl string `blueprint:"mutated"`
51}
52
Colin Crossa8e07cc2016-04-04 15:07:06 -070053type stl struct {
Colin Crossca860ac2016-01-04 14:34:37 -080054 Properties StlProperties
55}
56
Colin Crossa8e07cc2016-04-04 15:07:06 -070057func (stl *stl) props() []interface{} {
Colin Crossca860ac2016-01-04 14:34:37 -080058 return []interface{}{&stl.Properties}
59}
60
Colin Crossa8e07cc2016-04-04 15:07:06 -070061func (stl *stl) begin(ctx BaseModuleContext) {
Colin Crossca860ac2016-01-04 14:34:37 -080062 stl.Properties.SelectedStl = func() string {
Colin Cross79248852016-07-12 13:12:33 -070063 s := ""
64 if stl.Properties.Stl != nil {
65 s = *stl.Properties.Stl
66 }
Jeff Gastonaf3cc2d2017-09-27 17:01:44 -070067 if ctx.useSdk() && ctx.Device() {
Colin Cross79248852016-07-12 13:12:33 -070068 switch s {
Sasha Smundakfc22e4e2019-03-24 14:17:56 -070069 case "", "system":
Colin Crossca860ac2016-01-04 14:34:37 -080070 return "ndk_system"
Dan Albert749fc782018-01-04 14:21:14 -080071 case "c++_shared", "c++_static":
Colin Cross79248852016-07-12 13:12:33 -070072 return "ndk_lib" + s
David Benjamin87f9f032017-01-25 14:10:04 -050073 case "libc++":
74 return "ndk_libc++_shared"
75 case "libc++_static":
76 return "ndk_libc++_static"
Colin Cross4a97cb42016-04-21 15:53:42 -070077 case "none":
78 return ""
Colin Crossca860ac2016-01-04 14:34:37 -080079 default:
Colin Cross79248852016-07-12 13:12:33 -070080 ctx.ModuleErrorf("stl: %q is not a supported STL with sdk_version set", s)
Colin Crossca860ac2016-01-04 14:34:37 -080081 return ""
82 }
Colin Cross3edeee12017-04-04 12:59:48 -070083 } else if ctx.Windows() {
Colin Cross79248852016-07-12 13:12:33 -070084 switch s {
Pirama Arumuga Nainara403cc72018-08-08 10:28:12 -070085 case "libc++", "libc++_static", "":
86 // Only use static libc++ for Windows.
87 return "libc++_static"
Colin Crossca860ac2016-01-04 14:34:37 -080088 case "none":
89 return ""
90 default:
Colin Cross79248852016-07-12 13:12:33 -070091 ctx.ModuleErrorf("stl: %q is not a supported STL for windows", s)
Colin Crossca860ac2016-01-04 14:34:37 -080092 return ""
93 }
Doug Hornc32c6b02019-01-17 14:44:05 -080094 } else if ctx.Fuchsia() {
95 switch s {
96 case "c++_static":
97 return "libc++_static"
98 case "c++_shared":
99 return "libc++"
100 case "libc++", "libc++_static":
101 return s
102 case "none":
103 return ""
104 case "":
105 if ctx.static() {
106 return "libc++_static"
107 } else {
108 return "libc++"
109 }
110 default:
111 ctx.ModuleErrorf("stl: %q is not a supported STL on Fuchsia", s)
112 return ""
113 }
Colin Crossca860ac2016-01-04 14:34:37 -0800114 } else {
Colin Cross79248852016-07-12 13:12:33 -0700115 switch s {
Dan Willemsen141d5662016-06-15 13:47:51 -0700116 case "libc++", "libc++_static":
Colin Cross79248852016-07-12 13:12:33 -0700117 return s
Colin Crossca860ac2016-01-04 14:34:37 -0800118 case "none":
119 return ""
120 case "":
121 if ctx.static() {
122 return "libc++_static"
123 } else {
124 return "libc++"
125 }
126 default:
Colin Cross79248852016-07-12 13:12:33 -0700127 ctx.ModuleErrorf("stl: %q is not a supported STL", s)
Colin Crossca860ac2016-01-04 14:34:37 -0800128 return ""
129 }
130 }
131 }()
132}
133
Dan Albert90b9bbc2018-11-15 11:29:28 -0800134func needsLibAndroidSupport(ctx BaseModuleContext) bool {
135 versionStr, err := normalizeNdkApiLevel(ctx, ctx.sdkVersion(), ctx.Arch())
136 if err != nil {
137 ctx.PropertyErrorf("sdk_version", err.Error())
138 }
139
140 if versionStr == "current" {
141 return false
142 }
143
144 version, err := strconv.Atoi(versionStr)
145 if err != nil {
146 panic(fmt.Sprintf(
147 "invalid API level returned from normalizeNdkApiLevel: %q",
148 versionStr))
149 }
150
151 return version < 21
152}
153
Colin Crossa8e07cc2016-04-04 15:07:06 -0700154func (stl *stl) deps(ctx BaseModuleContext, deps Deps) Deps {
Colin Crossca860ac2016-01-04 14:34:37 -0800155 switch stl.Properties.SelectedStl {
156 case "libstdc++":
Dan Willemsen141d5662016-06-15 13:47:51 -0700157 // Nothing
Colin Crossca860ac2016-01-04 14:34:37 -0800158 case "libc++", "libc++_static":
159 if stl.Properties.SelectedStl == "libc++" {
160 deps.SharedLibs = append(deps.SharedLibs, stl.Properties.SelectedStl)
161 } else {
162 deps.StaticLibs = append(deps.StaticLibs, stl.Properties.SelectedStl)
163 }
Dan Albert2da19cb2019-07-24 12:17:40 -0700164 if ctx.Device() && !ctx.useSdk() {
165 // __cxa_demangle is not a part of libc++.so on the device since
166 // it's large and most processes don't need it. Statically link
167 // libc++demangle into every process so that users still have it if
168 // needed, but the linker won't include this unless it is actually
169 // called.
170 // http://b/138245375
171 deps.StaticLibs = append(deps.StaticLibs, "libc++demangle")
172 }
Dan Willemsen2e47b342016-11-17 01:02:25 -0800173 if ctx.toolchain().Bionic() {
Colin Cross635c3b02016-05-18 15:37:25 -0700174 if ctx.Arch().ArchType == android.Arm {
Colin Crossca860ac2016-01-04 14:34:37 -0800175 deps.StaticLibs = append(deps.StaticLibs, "libunwind_llvm")
176 }
177 if ctx.staticBinary() {
Colin Cross3d92b272016-07-14 15:59:32 -0700178 deps.StaticLibs = append(deps.StaticLibs, "libm", "libc", "libdl")
Colin Crossca860ac2016-01-04 14:34:37 -0800179 }
180 }
181 case "":
182 // None or error.
183 case "ndk_system":
184 // TODO: Make a system STL prebuilt for the NDK.
185 // The system STL doesn't have a prebuilt (it uses the system's libstdc++), but it does have
186 // its own includes. The includes are handled in CCBase.Flags().
187 deps.SharedLibs = append([]string{"libstdc++"}, deps.SharedLibs...)
Ryan Prichardb1703652018-03-26 16:30:31 -0700188 case "ndk_libc++_shared", "ndk_libc++_static":
189 if stl.Properties.SelectedStl == "ndk_libc++_shared" {
190 deps.SharedLibs = append(deps.SharedLibs, stl.Properties.SelectedStl)
191 } else {
192 deps.StaticLibs = append(deps.StaticLibs, stl.Properties.SelectedStl, "ndk_libc++abi")
193 }
Dan Albert90b9bbc2018-11-15 11:29:28 -0800194 if needsLibAndroidSupport(ctx) {
195 deps.StaticLibs = append(deps.StaticLibs, "ndk_libandroid_support")
196 }
Ryan Prichardb1703652018-03-26 16:30:31 -0700197 if ctx.Arch().ArchType == android.Arm {
198 deps.StaticLibs = append(deps.StaticLibs, "ndk_libunwind")
199 }
Colin Crossca860ac2016-01-04 14:34:37 -0800200 default:
201 panic(fmt.Errorf("Unknown stl: %q", stl.Properties.SelectedStl))
202 }
203
204 return deps
205}
206
Colin Crossa8e07cc2016-04-04 15:07:06 -0700207func (stl *stl) flags(ctx ModuleContext, flags Flags) Flags {
Colin Crossca860ac2016-01-04 14:34:37 -0800208 switch stl.Properties.SelectedStl {
209 case "libc++", "libc++_static":
210 flags.CFlags = append(flags.CFlags, "-D_USING_LIBCXX")
Dan Alberta07b8452018-01-11 13:00:46 -0800211
212 if ctx.Darwin() {
213 // libc++'s headers are annotated with availability macros that
214 // indicate which version of Mac OS was the first to ship with a
215 // libc++ feature available in its *system's* libc++.dylib. We do
216 // not use the system's library, but rather ship our own. As such,
217 // these availability attributes are meaningless for us but cause
218 // build breaks when we try to use code that would not be available
219 // in the system's dylib.
220 flags.CppFlags = append(flags.CppFlags,
221 "-D_LIBCPP_DISABLE_AVAILABILITY")
222 }
223
Dan Willemsen2e47b342016-11-17 01:02:25 -0800224 if !ctx.toolchain().Bionic() {
Colin Crossca860ac2016-01-04 14:34:37 -0800225 flags.CppFlags = append(flags.CppFlags, "-nostdinc++")
Pete Bentley99f2fc22019-08-02 14:02:20 +0100226 flags.extraLibFlags = append(flags.extraLibFlags, "-nodefaultlibs")
Colin Crossca860ac2016-01-04 14:34:37 -0800227 if ctx.staticBinary() {
Pete Bentley99f2fc22019-08-02 14:02:20 +0100228 flags.extraLibFlags = append(flags.extraLibFlags, hostStaticGccLibs[ctx.Os()]...)
Colin Crossca860ac2016-01-04 14:34:37 -0800229 } else {
Pete Bentley99f2fc22019-08-02 14:02:20 +0100230 flags.extraLibFlags = append(flags.extraLibFlags, hostDynamicGccLibs[ctx.Os()]...)
Colin Crossca860ac2016-01-04 14:34:37 -0800231 }
Pirama Arumuga Nainara403cc72018-08-08 10:28:12 -0700232 if ctx.Windows() {
233 // Use SjLj exceptions for 32-bit. libgcc_eh implements SjLj
234 // exception model for 32-bit.
235 if ctx.Arch().ArchType == android.X86 {
236 flags.CppFlags = append(flags.CppFlags, "-fsjlj-exceptions")
237 }
238 flags.CppFlags = append(flags.CppFlags,
239 // Disable visiblity annotations since we're using static
240 // libc++.
241 "-D_LIBCPP_DISABLE_VISIBILITY_ANNOTATIONS",
242 "-D_LIBCXXABI_DISABLE_VISIBILITY_ANNOTATIONS",
243 // Use Win32 threads in libc++.
244 "-D_LIBCPP_HAS_THREAD_API_WIN32")
245 }
Colin Crossca860ac2016-01-04 14:34:37 -0800246 } else {
Colin Cross635c3b02016-05-18 15:37:25 -0700247 if ctx.Arch().ArchType == android.Arm {
Colin Crossca860ac2016-01-04 14:34:37 -0800248 flags.LdFlags = append(flags.LdFlags, "-Wl,--exclude-libs,libunwind_llvm.a")
249 }
250 }
251 case "libstdc++":
Dan Willemsen141d5662016-06-15 13:47:51 -0700252 // Nothing
Colin Crossca860ac2016-01-04 14:34:37 -0800253 case "ndk_system":
Colin Cross635c3b02016-05-18 15:37:25 -0700254 ndkSrcRoot := android.PathForSource(ctx, "prebuilts/ndk/current/sources/cxx-stl/system/include")
Colin Crossca860ac2016-01-04 14:34:37 -0800255 flags.CFlags = append(flags.CFlags, "-isystem "+ndkSrcRoot.String())
256 case "ndk_libc++_shared", "ndk_libc++_static":
Christopher Ferrisc3a1e222019-04-10 17:57:50 -0700257 if ctx.Arch().ArchType == android.Arm {
258 // Make sure the _Unwind_XXX symbols are not re-exported.
259 flags.LdFlags = append(flags.LdFlags, "-Wl,--exclude-libs,libunwind.a")
260 }
Colin Crossca860ac2016-01-04 14:34:37 -0800261 case "":
262 // None or error.
Dan Willemsen2e47b342016-11-17 01:02:25 -0800263 if !ctx.toolchain().Bionic() {
Colin Crossca860ac2016-01-04 14:34:37 -0800264 flags.CppFlags = append(flags.CppFlags, "-nostdinc++")
Pete Bentley99f2fc22019-08-02 14:02:20 +0100265 flags.extraLibFlags = append(flags.extraLibFlags, "-nodefaultlibs")
Colin Crossca860ac2016-01-04 14:34:37 -0800266 if ctx.staticBinary() {
Pete Bentley99f2fc22019-08-02 14:02:20 +0100267 flags.extraLibFlags = append(flags.extraLibFlags, hostStaticGccLibs[ctx.Os()]...)
Colin Crossca860ac2016-01-04 14:34:37 -0800268 } else {
Pete Bentley99f2fc22019-08-02 14:02:20 +0100269 flags.extraLibFlags = append(flags.extraLibFlags, hostDynamicGccLibs[ctx.Os()]...)
Colin Crossca860ac2016-01-04 14:34:37 -0800270 }
271 }
272 default:
273 panic(fmt.Errorf("Unknown stl: %q", stl.Properties.SelectedStl))
274 }
275
276 return flags
277}
278
Colin Crossa1ad8d12016-06-01 17:09:44 -0700279var hostDynamicGccLibs, hostStaticGccLibs map[android.OsType][]string
Colin Crossca860ac2016-01-04 14:34:37 -0800280
281func init() {
Colin Crossa1ad8d12016-06-01 17:09:44 -0700282 hostDynamicGccLibs = map[android.OsType][]string{
Doug Hornc32c6b02019-01-17 14:44:05 -0800283 android.Fuchsia: []string{"-lc", "-lunwind"},
284 android.Linux: []string{"-lgcc_s", "-lgcc", "-lc", "-lgcc_s", "-lgcc"},
285 android.Darwin: []string{"-lc", "-lSystem"},
Pirama Arumuga Nainara403cc72018-08-08 10:28:12 -0700286 android.Windows: []string{"-Wl,--start-group", "-lmingw32", "-lgcc", "-lgcc_eh",
Pirama Arumuga Nainar087bba72018-11-29 17:19:25 -0800287 "-lmoldname", "-lmingwex", "-lmsvcrt", "-lucrt", "-lpthread",
Pirama Arumuga Nainara403cc72018-08-08 10:28:12 -0700288 "-ladvapi32", "-lshell32", "-luser32", "-lkernel32", "-lpsapi",
289 "-Wl,--end-group"},
Colin Crossca860ac2016-01-04 14:34:37 -0800290 }
Colin Crossa1ad8d12016-06-01 17:09:44 -0700291 hostStaticGccLibs = map[android.OsType][]string{
Colin Cross635c3b02016-05-18 15:37:25 -0700292 android.Linux: []string{"-Wl,--start-group", "-lgcc", "-lgcc_eh", "-lc", "-Wl,--end-group"},
293 android.Darwin: []string{"NO_STATIC_HOST_BINARIES_ON_DARWIN"},
294 android.Windows: []string{"NO_STATIC_HOST_BINARIES_ON_WINDOWS"},
Colin Crossca860ac2016-01-04 14:34:37 -0800295 }
296}