blob: 6a55bb2b39893deb0916d014eb45d297eb48f587 [file] [log] [blame]
Dimitry Ivanov4cabfaa2017-03-07 11:19:05 -08001/*
2 * Copyright (C) 2017 The Android Open Source Project
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 * * Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * * Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in
12 * the documentation and/or other materials provided with the
13 * distribution.
14 *
15 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
16 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
17 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
18 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
19 * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
20 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
21 * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
22 * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
23 * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
24 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
25 * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26 * SUCH DAMAGE.
27 */
28
29#include <stdlib.h>
30#include <string.h>
31#include <sys/mman.h>
32
33#include <gtest/gtest.h>
34
Elliott Hughes15a2b7b2019-02-15 13:48:38 -080035#include "linker_config.h"
36#include "linker_utils.h"
Dimitry Ivanov4cabfaa2017-03-07 11:19:05 -080037
38#include <unistd.h>
39
Mark Salyzynba1a7232018-11-14 15:19:53 -080040#include <android-base/file.h>
Tom Cherry98f016f2017-04-05 16:20:29 -070041#include <android-base/scopeguard.h>
Dimitry Ivanov4cabfaa2017-03-07 11:19:05 -080042#include <android-base/stringprintf.h>
Dimitry Ivanov4cabfaa2017-03-07 11:19:05 -080043
Jaesung Chung9d970082017-06-16 14:43:16 +090044#if defined(__LP64__)
45#define ARCH_SUFFIX "64"
46#else
47#define ARCH_SUFFIX ""
48#endif
Dimitry Ivanov4cabfaa2017-03-07 11:19:05 -080049
50static const char* config_str =
51 "# comment \n"
52 "dir.test = /data/local/tmp\n"
53 "\n"
54 "[test]\n"
55 "\n"
56 "enable.target.sdk.version = true\n"
57 "additional.namespaces=system\n"
Jiyong Park8b029512017-11-29 18:30:53 +090058 "additional.namespaces+=vndk\n"
Dimitry Ivanov4cabfaa2017-03-07 11:19:05 -080059 "namespace.default.isolated = true\n"
60 "namespace.default.search.paths = /vendor/${LIB}\n"
61 "namespace.default.permitted.paths = /vendor/${LIB}\n"
Jiyong Park8b029512017-11-29 18:30:53 +090062 "namespace.default.asan.search.paths = /data\n"
63 "namespace.default.asan.search.paths += /vendor/${LIB}\n"
Dimitry Ivanov4cabfaa2017-03-07 11:19:05 -080064 "namespace.default.asan.permitted.paths = /data:/vendor\n"
65 "namespace.default.links = system\n"
Jiyong Park8b029512017-11-29 18:30:53 +090066 "namespace.default.links += vndk\n"
67 // irregular whitespaces are added intentionally for testing purpose
68 "namespace.default.link.system.shared_libs= libc.so\n"
69 "namespace.default.link.system.shared_libs += libm.so:libdl.so\n"
70 "namespace.default.link.system.shared_libs +=libstdc++.so\n"
71 "namespace.default.link.vndk.shared_libs = libcutils.so:libbase.so\n"
Dimitry Ivanov4cabfaa2017-03-07 11:19:05 -080072 "namespace.system.isolated = true\n"
Jiyong Parkd7c48322017-04-03 23:10:37 +090073 "namespace.system.visible = true\n"
Dimitry Ivanov4cabfaa2017-03-07 11:19:05 -080074 "namespace.system.search.paths = /system/${LIB}\n"
75 "namespace.system.permitted.paths = /system/${LIB}\n"
76 "namespace.system.asan.search.paths = /data:/system/${LIB}\n"
77 "namespace.system.asan.permitted.paths = /data:/system\n"
Jiyong Park8b029512017-11-29 18:30:53 +090078 "namespace.vndk.isolated = tr\n"
79 "namespace.vndk.isolated += ue\n" // should be ignored and return as 'false'.
80 "namespace.vndk.search.paths = /system/${LIB}/vndk\n"
81 "namespace.vndk.asan.search.paths = /data\n"
82 "namespace.vndk.asan.search.paths += /system/${LIB}/vndk\n"
Logan Chien9ee45912018-01-18 12:05:09 +080083 "namespace.vndk.links = default\n"
84 "namespace.vndk.link.default.allow_all_shared_libs = true\n"
Dimitry Ivanov4cabfaa2017-03-07 11:19:05 -080085 "\n";
86
87static bool write_version(const std::string& path, uint32_t version) {
88 std::string content = android::base::StringPrintf("%d", version);
89 return android::base::WriteStringToFile(content, path);
90}
91
Jaesung Chung9d970082017-06-16 14:43:16 +090092static std::vector<std::string> resolve_paths(std::vector<std::string> paths) {
93 std::vector<std::string> resolved_paths;
94 resolve_paths(paths, &resolved_paths);
95 return resolved_paths;
96}
97
Dimitry Ivanov4cabfaa2017-03-07 11:19:05 -080098static void run_linker_config_smoke_test(bool is_asan) {
Jaesung Chung9d970082017-06-16 14:43:16 +090099 const std::vector<std::string> kExpectedDefaultSearchPath =
100 resolve_paths(is_asan ? std::vector<std::string>({ "/data", "/vendor/lib" ARCH_SUFFIX }) :
101 std::vector<std::string>({ "/vendor/lib" ARCH_SUFFIX }));
Dimitry Ivanov4cabfaa2017-03-07 11:19:05 -0800102
Jaesung Chung9d970082017-06-16 14:43:16 +0900103 const std::vector<std::string> kExpectedDefaultPermittedPath =
104 resolve_paths(is_asan ? std::vector<std::string>({ "/data", "/vendor" }) :
105 std::vector<std::string>({ "/vendor/lib" ARCH_SUFFIX }));
Dimitry Ivanov4cabfaa2017-03-07 11:19:05 -0800106
Jaesung Chung9d970082017-06-16 14:43:16 +0900107 const std::vector<std::string> kExpectedSystemSearchPath =
108 resolve_paths(is_asan ? std::vector<std::string>({ "/data", "/system/lib" ARCH_SUFFIX }) :
109 std::vector<std::string>({ "/system/lib" ARCH_SUFFIX }));
Dimitry Ivanov4cabfaa2017-03-07 11:19:05 -0800110
Jaesung Chung9d970082017-06-16 14:43:16 +0900111 const std::vector<std::string> kExpectedSystemPermittedPath =
112 resolve_paths(is_asan ? std::vector<std::string>({ "/data", "/system" }) :
113 std::vector<std::string>({ "/system/lib" ARCH_SUFFIX }));
Dimitry Ivanov4cabfaa2017-03-07 11:19:05 -0800114
Jiyong Park8b029512017-11-29 18:30:53 +0900115 const std::vector<std::string> kExpectedVndkSearchPath =
116 resolve_paths(is_asan ? std::vector<std::string>({ "/data", "/system/lib" ARCH_SUFFIX "/vndk"}) :
117 std::vector<std::string>({ "/system/lib" ARCH_SUFFIX "/vndk"}));
118
Dimitry Ivanov4cabfaa2017-03-07 11:19:05 -0800119 TemporaryFile tmp_file;
120 close(tmp_file.fd);
121 tmp_file.fd = -1;
122
123 android::base::WriteStringToFile(config_str, tmp_file.path);
124
125 TemporaryDir tmp_dir;
126
127 std::string executable_path = std::string(tmp_dir.path) + "/some-binary";
128 std::string version_file = std::string(tmp_dir.path) + "/.version";
129
Tom Cherry98f016f2017-04-05 16:20:29 -0700130 auto file_guard =
131 android::base::make_scope_guard([&version_file] { unlink(version_file.c_str()); });
Dimitry Ivanov4cabfaa2017-03-07 11:19:05 -0800132
133 ASSERT_TRUE(write_version(version_file, 113U)) << strerror(errno);
134
135 // read config
136 const Config* config = nullptr;
137 std::string error_msg;
138 ASSERT_TRUE(Config::read_binary_config(tmp_file.path,
139 executable_path.c_str(),
140 is_asan,
141 &config,
142 &error_msg)) << error_msg;
143 ASSERT_TRUE(config != nullptr);
144 ASSERT_TRUE(error_msg.empty());
145
Elliott Hughesff1428a2018-11-12 16:01:37 -0800146 ASSERT_EQ(113, config->target_sdk_version());
Dimitry Ivanov4cabfaa2017-03-07 11:19:05 -0800147
148 const NamespaceConfig* default_ns_config = config->default_namespace_config();
149 ASSERT_TRUE(default_ns_config != nullptr);
150
151 ASSERT_TRUE(default_ns_config->isolated());
Jiyong Parkd7c48322017-04-03 23:10:37 +0900152 ASSERT_FALSE(default_ns_config->visible());
Dimitry Ivanov4cabfaa2017-03-07 11:19:05 -0800153 ASSERT_EQ(kExpectedDefaultSearchPath, default_ns_config->search_paths());
154 ASSERT_EQ(kExpectedDefaultPermittedPath, default_ns_config->permitted_paths());
155
156 const auto& default_ns_links = default_ns_config->links();
Jiyong Park8b029512017-11-29 18:30:53 +0900157 ASSERT_EQ(2U, default_ns_links.size());
Logan Chien9ee45912018-01-18 12:05:09 +0800158
Dimitry Ivanov4cabfaa2017-03-07 11:19:05 -0800159 ASSERT_EQ("system", default_ns_links[0].ns_name());
160 ASSERT_EQ("libc.so:libm.so:libdl.so:libstdc++.so", default_ns_links[0].shared_libs());
Logan Chien9ee45912018-01-18 12:05:09 +0800161 ASSERT_FALSE(default_ns_links[0].allow_all_shared_libs());
162
Jiyong Park8b029512017-11-29 18:30:53 +0900163 ASSERT_EQ("vndk", default_ns_links[1].ns_name());
164 ASSERT_EQ("libcutils.so:libbase.so", default_ns_links[1].shared_libs());
Logan Chien9ee45912018-01-18 12:05:09 +0800165 ASSERT_FALSE(default_ns_links[1].allow_all_shared_libs());
Dimitry Ivanov4cabfaa2017-03-07 11:19:05 -0800166
167 auto& ns_configs = config->namespace_configs();
Jiyong Park8b029512017-11-29 18:30:53 +0900168 ASSERT_EQ(3U, ns_configs.size());
Dimitry Ivanov4cabfaa2017-03-07 11:19:05 -0800169
170 // find second namespace
171 const NamespaceConfig* ns_system = nullptr;
Jiyong Park8b029512017-11-29 18:30:53 +0900172 const NamespaceConfig* ns_vndk = nullptr;
Dimitry Ivanov4cabfaa2017-03-07 11:19:05 -0800173 for (auto& ns : ns_configs) {
174 std::string ns_name = ns->name();
Jiyong Park8b029512017-11-29 18:30:53 +0900175 ASSERT_TRUE(ns_name == "system" || ns_name == "default" || ns_name == "vndk")
Dimitry Ivanov4cabfaa2017-03-07 11:19:05 -0800176 << "unexpected ns name: " << ns->name();
177
178 if (ns_name == "system") {
179 ns_system = ns.get();
Jiyong Park8b029512017-11-29 18:30:53 +0900180 } else if (ns_name == "vndk") {
181 ns_vndk = ns.get();
Dimitry Ivanov4cabfaa2017-03-07 11:19:05 -0800182 }
183 }
184
185 ASSERT_TRUE(ns_system != nullptr) << "system namespace was not found";
186
187 ASSERT_TRUE(ns_system->isolated());
Jiyong Parkd7c48322017-04-03 23:10:37 +0900188 ASSERT_TRUE(ns_system->visible());
Dimitry Ivanov4cabfaa2017-03-07 11:19:05 -0800189 ASSERT_EQ(kExpectedSystemSearchPath, ns_system->search_paths());
190 ASSERT_EQ(kExpectedSystemPermittedPath, ns_system->permitted_paths());
Jiyong Park8b029512017-11-29 18:30:53 +0900191
192 ASSERT_TRUE(ns_vndk != nullptr) << "vndk namespace was not found";
193
194 ASSERT_FALSE(ns_vndk->isolated()); // malformed bool property
Logan Chien9ee45912018-01-18 12:05:09 +0800195 ASSERT_FALSE(ns_vndk->visible()); // undefined bool property
Jiyong Park8b029512017-11-29 18:30:53 +0900196 ASSERT_EQ(kExpectedVndkSearchPath, ns_vndk->search_paths());
Logan Chien9ee45912018-01-18 12:05:09 +0800197
198 const auto& ns_vndk_links = ns_vndk->links();
199 ASSERT_EQ(1U, ns_vndk_links.size());
200 ASSERT_EQ("default", ns_vndk_links[0].ns_name());
201 ASSERT_TRUE(ns_vndk_links[0].allow_all_shared_libs());
Dimitry Ivanov4cabfaa2017-03-07 11:19:05 -0800202}
203
204TEST(linker_config, smoke) {
205 run_linker_config_smoke_test(false);
206}
207
Jiyong Park1a524d62017-08-04 10:25:46 +0900208TEST(linker_config, asan_smoke) {
209 run_linker_config_smoke_test(true);
210}
Logan Chien9ee45912018-01-18 12:05:09 +0800211
212TEST(linker_config, ns_link_shared_libs_invalid_settings) {
213 // This unit test ensures an error is emitted when a namespace link in ld.config.txt specifies
214 // both shared_libs and allow_all_shared_libs.
215
216 static const char config_str[] =
217 "dir.test = /data/local/tmp\n"
218 "\n"
219 "[test]\n"
220 "additional.namespaces = system\n"
221 "namespace.default.links = system\n"
222 "namespace.default.link.system.shared_libs = libc.so:libm.so\n"
223 "namespace.default.link.system.allow_all_shared_libs = true\n"
224 "\n";
225
226 TemporaryFile tmp_file;
227 close(tmp_file.fd);
228 tmp_file.fd = -1;
229
230 android::base::WriteStringToFile(config_str, tmp_file.path);
231
232 TemporaryDir tmp_dir;
233
234 std::string executable_path = std::string(tmp_dir.path) + "/some-binary";
235
236 const Config* config = nullptr;
237 std::string error_msg;
238 ASSERT_FALSE(Config::read_binary_config(tmp_file.path,
239 executable_path.c_str(),
240 false,
241 &config,
242 &error_msg));
243 ASSERT_TRUE(config == nullptr);
244 ASSERT_EQ(std::string(tmp_file.path) + ":6: "
245 "error: both shared_libs and allow_all_shared_libs are set for default->system link.",
246 error_msg);
247}
Inseob Kim216323b2018-06-18 15:30:18 +0900248
249TEST(linker_config, dir_path_resolve) {
250 // This unit test ensures the linker resolves paths of dir.${section}
251 // properties to real path.
252
253 TemporaryDir tmp_dir;
254
255 std::string sub_dir = std::string(tmp_dir.path) + "/subdir";
256 mkdir(sub_dir.c_str(), 0755);
257
258 auto subdir_guard =
259 android::base::make_scope_guard([&sub_dir] { rmdir(sub_dir.c_str()); });
260
261 std::string symlink_path = std::string(tmp_dir.path) + "/symlink";
262 symlink(sub_dir.c_str(), symlink_path.c_str());
263
264 auto symlink_guard =
265 android::base::make_scope_guard([&symlink_path] { unlink(symlink_path.c_str()); });
266
267 std::string config_str =
268 "dir.test = " + symlink_path + "\n"
269 "\n"
270 "[test]\n";
271
272 TemporaryFile tmp_file;
273 close(tmp_file.fd);
274 tmp_file.fd = -1;
275
276 android::base::WriteStringToFile(config_str, tmp_file.path);
277
278 std::string executable_path = sub_dir + "/some-binary";
279
280 const Config* config = nullptr;
281 std::string error_msg;
282
283 ASSERT_TRUE(Config::read_binary_config(tmp_file.path,
284 executable_path.c_str(),
285 false,
286 &config,
287 &error_msg)) << error_msg;
288
289 ASSERT_TRUE(config != nullptr) << error_msg;
290 ASSERT_TRUE(error_msg.empty()) << error_msg;
291}