blob: 536797261f03e95a7425d5317a7176572c81e671 [file] [log] [blame]
Colin Crossb27e2002013-01-28 17:19:43 -08001/*
2 * Copyright (C) 2013 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#include <gtest/gtest.h>
18#include <sys/wait.h>
Elliott Hughese4375192013-10-21 17:09:52 -070019#include <errno.h>
Greg Hackmanncb215a72013-02-13 14:41:48 -080020#include <unistd.h>
21#include <string>
Colin Crossb27e2002013-01-28 17:19:43 -080022
23#if __BIONIC__
24
25#define _REALLY_INCLUDE_SYS__SYSTEM_PROPERTIES_H_
26#include <sys/_system_properties.h>
27
Greg Hackmann1540f602013-06-19 13:31:21 -070028extern void *__system_property_area__;
Colin Crossb27e2002013-01-28 17:19:43 -080029
30struct LocalPropertyTestState {
Greg Hackmanncb215a72013-02-13 14:41:48 -080031 LocalPropertyTestState() : valid(false) {
Elliott Hughese4375192013-10-21 17:09:52 -070032 const char* ANDROID_DATA = getenv("ANDROID_DATA");
33 char dir_template[PATH_MAX];
34 snprintf(dir_template, sizeof(dir_template), "%s/local/tmp/prop-XXXXXX", ANDROID_DATA);
35 char* dirname = mkdtemp(dir_template);
Greg Hackmanncb215a72013-02-13 14:41:48 -080036 if (!dirname) {
Elliott Hughese4375192013-10-21 17:09:52 -070037 fprintf(stderr, "making temp file for test state failed (is %s writable?): %s",
38 dir_template, strerror(errno));
Greg Hackmanncb215a72013-02-13 14:41:48 -080039 return;
40 }
41
Greg Hackmann1540f602013-06-19 13:31:21 -070042 old_pa = __system_property_area__;
43 __system_property_area__ = NULL;
Greg Hackmanncb215a72013-02-13 14:41:48 -080044
45 pa_dirname = dirname;
46 pa_filename = pa_dirname + "/__properties__";
47
48 __system_property_set_filename(pa_filename.c_str());
49 __system_property_area_init();
50 valid = true;
Colin Crossb27e2002013-01-28 17:19:43 -080051 }
52
53 ~LocalPropertyTestState() {
Elliott Hughese4375192013-10-21 17:09:52 -070054 if (!valid) {
Greg Hackmanncb215a72013-02-13 14:41:48 -080055 return;
Elliott Hughese4375192013-10-21 17:09:52 -070056 }
Greg Hackmanncb215a72013-02-13 14:41:48 -080057
Greg Hackmann1540f602013-06-19 13:31:21 -070058 __system_property_area__ = old_pa;
Greg Hackmanncb215a72013-02-13 14:41:48 -080059
60 __system_property_set_filename(PROP_FILENAME);
61 unlink(pa_filename.c_str());
62 rmdir(pa_dirname.c_str());
Colin Crossb27e2002013-01-28 17:19:43 -080063 }
Greg Hackmanncb215a72013-02-13 14:41:48 -080064public:
65 bool valid;
Colin Crossb27e2002013-01-28 17:19:43 -080066private:
Greg Hackmanncb215a72013-02-13 14:41:48 -080067 std::string pa_dirname;
68 std::string pa_filename;
Greg Hackmann1540f602013-06-19 13:31:21 -070069 void *old_pa;
Colin Crossb27e2002013-01-28 17:19:43 -080070};
71
72TEST(properties, add) {
73 LocalPropertyTestState pa;
Greg Hackmanncb215a72013-02-13 14:41:48 -080074 ASSERT_TRUE(pa.valid);
Colin Crossb27e2002013-01-28 17:19:43 -080075
76 char propvalue[PROP_VALUE_MAX];
77
78 ASSERT_EQ(0, __system_property_add("property", 8, "value1", 6));
79 ASSERT_EQ(0, __system_property_add("other_property", 14, "value2", 6));
80 ASSERT_EQ(0, __system_property_add("property_other", 14, "value3", 6));
81
82 ASSERT_EQ(6, __system_property_get("property", propvalue));
83 ASSERT_STREQ(propvalue, "value1");
84
85 ASSERT_EQ(6, __system_property_get("other_property", propvalue));
86 ASSERT_STREQ(propvalue, "value2");
87
88 ASSERT_EQ(6, __system_property_get("property_other", propvalue));
89 ASSERT_STREQ(propvalue, "value3");
90}
91
92TEST(properties, update) {
93 LocalPropertyTestState pa;
Greg Hackmanncb215a72013-02-13 14:41:48 -080094 ASSERT_TRUE(pa.valid);
Colin Crossb27e2002013-01-28 17:19:43 -080095
96 char propvalue[PROP_VALUE_MAX];
97 prop_info *pi;
98
99 ASSERT_EQ(0, __system_property_add("property", 8, "oldvalue1", 9));
100 ASSERT_EQ(0, __system_property_add("other_property", 14, "value2", 6));
101 ASSERT_EQ(0, __system_property_add("property_other", 14, "value3", 6));
102
103 pi = (prop_info *)__system_property_find("property");
104 ASSERT_NE((prop_info *)NULL, pi);
105 __system_property_update(pi, "value4", 6);
106
107 pi = (prop_info *)__system_property_find("other_property");
108 ASSERT_NE((prop_info *)NULL, pi);
109 __system_property_update(pi, "newvalue5", 9);
110
111 pi = (prop_info *)__system_property_find("property_other");
112 ASSERT_NE((prop_info *)NULL, pi);
113 __system_property_update(pi, "value6", 6);
114
115 ASSERT_EQ(6, __system_property_get("property", propvalue));
116 ASSERT_STREQ(propvalue, "value4");
117
118 ASSERT_EQ(9, __system_property_get("other_property", propvalue));
119 ASSERT_STREQ(propvalue, "newvalue5");
120
121 ASSERT_EQ(6, __system_property_get("property_other", propvalue));
122 ASSERT_STREQ(propvalue, "value6");
123}
124
Greg Hackmanncb215a72013-02-13 14:41:48 -0800125TEST(properties, fill) {
Colin Crossb27e2002013-01-28 17:19:43 -0800126 LocalPropertyTestState pa;
Greg Hackmanncb215a72013-02-13 14:41:48 -0800127 ASSERT_TRUE(pa.valid);
Colin Crossb27e2002013-01-28 17:19:43 -0800128 char prop_name[PROP_NAME_MAX];
129 char prop_value[PROP_VALUE_MAX];
130 char prop_value_ret[PROP_VALUE_MAX];
Greg Hackmanncb215a72013-02-13 14:41:48 -0800131 int count = 0;
Colin Crossb27e2002013-01-28 17:19:43 -0800132 int ret;
133
Greg Hackmanncb215a72013-02-13 14:41:48 -0800134 while (true) {
135 ret = snprintf(prop_name, PROP_NAME_MAX - 1, "property_%d", count);
Colin Crossb27e2002013-01-28 17:19:43 -0800136 memset(prop_name + ret, 'a', PROP_NAME_MAX - 1 - ret);
Greg Hackmanncb215a72013-02-13 14:41:48 -0800137 ret = snprintf(prop_value, PROP_VALUE_MAX - 1, "value_%d", count);
Colin Crossb27e2002013-01-28 17:19:43 -0800138 memset(prop_value + ret, 'b', PROP_VALUE_MAX - 1 - ret);
139 prop_name[PROP_NAME_MAX - 1] = 0;
140 prop_value[PROP_VALUE_MAX - 1] = 0;
141
Greg Hackmanncb215a72013-02-13 14:41:48 -0800142 ret = __system_property_add(prop_name, PROP_NAME_MAX - 1, prop_value, PROP_VALUE_MAX - 1);
143 if (ret < 0)
144 break;
145
146 count++;
Colin Crossb27e2002013-01-28 17:19:43 -0800147 }
148
Greg Hackmanncb215a72013-02-13 14:41:48 -0800149 // For historical reasons at least 247 properties must be supported
150 ASSERT_GE(count, 247);
151
152 for (int i = 0; i < count; i++) {
Colin Crossb27e2002013-01-28 17:19:43 -0800153 ret = snprintf(prop_name, PROP_NAME_MAX - 1, "property_%d", i);
154 memset(prop_name + ret, 'a', PROP_NAME_MAX - 1 - ret);
155 ret = snprintf(prop_value, PROP_VALUE_MAX - 1, "value_%d", i);
156 memset(prop_value + ret, 'b', PROP_VALUE_MAX - 1 - ret);
157 prop_name[PROP_NAME_MAX - 1] = 0;
158 prop_value[PROP_VALUE_MAX - 1] = 0;
159 memset(prop_value_ret, '\0', PROP_VALUE_MAX);
160
161 ASSERT_EQ(PROP_VALUE_MAX - 1, __system_property_get(prop_name, prop_value_ret));
162 ASSERT_EQ(0, memcmp(prop_value, prop_value_ret, PROP_VALUE_MAX));
163 }
164}
165
Greg Hackmannc6ff8442013-02-12 16:39:31 -0800166static void foreach_test_callback(const prop_info *pi, void* cookie) {
167 size_t *count = static_cast<size_t *>(cookie);
168
169 ASSERT_NE((prop_info *)NULL, pi);
170 (*count)++;
171}
172
173TEST(properties, foreach) {
174 LocalPropertyTestState pa;
Greg Hackmanncb215a72013-02-13 14:41:48 -0800175 ASSERT_TRUE(pa.valid);
Greg Hackmannc6ff8442013-02-12 16:39:31 -0800176 size_t count = 0;
177
178 ASSERT_EQ(0, __system_property_add("property", 8, "value1", 6));
179 ASSERT_EQ(0, __system_property_add("other_property", 14, "value2", 6));
180 ASSERT_EQ(0, __system_property_add("property_other", 14, "value3", 6));
181
182 ASSERT_EQ(0, __system_property_foreach(foreach_test_callback, &count));
183 ASSERT_EQ(3U, count);
184}
185
Colin Crossb27e2002013-01-28 17:19:43 -0800186TEST(properties, find_nth) {
187 LocalPropertyTestState pa;
Greg Hackmanncb215a72013-02-13 14:41:48 -0800188 ASSERT_TRUE(pa.valid);
Colin Crossb27e2002013-01-28 17:19:43 -0800189
190 ASSERT_EQ(0, __system_property_add("property", 8, "value1", 6));
191 ASSERT_EQ(0, __system_property_add("other_property", 14, "value2", 6));
192 ASSERT_EQ(0, __system_property_add("property_other", 14, "value3", 6));
193
194 ASSERT_NE((const prop_info *)NULL, __system_property_find_nth(0));
195 ASSERT_NE((const prop_info *)NULL, __system_property_find_nth(1));
196 ASSERT_NE((const prop_info *)NULL, __system_property_find_nth(2));
197
198 ASSERT_EQ((const prop_info *)NULL, __system_property_find_nth(3));
199 ASSERT_EQ((const prop_info *)NULL, __system_property_find_nth(4));
200 ASSERT_EQ((const prop_info *)NULL, __system_property_find_nth(5));
201 ASSERT_EQ((const prop_info *)NULL, __system_property_find_nth(100));
202 ASSERT_EQ((const prop_info *)NULL, __system_property_find_nth(200));
203 ASSERT_EQ((const prop_info *)NULL, __system_property_find_nth(247));
204}
205
Greg Hackmannd5276422013-06-17 12:37:09 -0700206static void hierarchical_test_callback(const prop_info *pi, void *cookie) {
207 bool (*ok)[8][8] = static_cast<bool (*)[8][8]>(cookie);
208
209 char name[PROP_NAME_MAX];
210 char value[PROP_VALUE_MAX];
211
212 __system_property_read(pi, name, value);
213
214 int name_i, name_j, name_k;
215 int value_i, value_j, value_k;
216 ASSERT_EQ(3, sscanf(name, "property_%d.%d.%d", &name_i, &name_j, &name_k));
217 ASSERT_EQ(3, sscanf(value, "value_%d.%d.%d", &value_i, &value_j, &value_k));
218 ASSERT_EQ(name_i, value_i);
219 ASSERT_GE(name_i, 0);
220 ASSERT_LT(name_i, 8);
221 ASSERT_EQ(name_j, value_j);
222 ASSERT_GE(name_j, 0);
223 ASSERT_LT(name_j, 8);
224 ASSERT_EQ(name_k, value_k);
225 ASSERT_GE(name_k, 0);
226 ASSERT_LT(name_k, 8);
227
228 ok[name_i][name_j][name_k] = true;
229}
230
231TEST(properties, fill_hierarchical) {
232 LocalPropertyTestState pa;
233 ASSERT_TRUE(pa.valid);
234 char prop_name[PROP_NAME_MAX];
235 char prop_value[PROP_VALUE_MAX];
236 char prop_value_ret[PROP_VALUE_MAX];
237 int ret;
238
239 for (int i = 0; i < 8; i++) {
240 for (int j = 0; j < 8; j++) {
241 for (int k = 0; k < 8; k++) {
242 ret = snprintf(prop_name, PROP_NAME_MAX - 1, "property_%d.%d.%d", i, j, k);
243 memset(prop_name + ret, 'a', PROP_NAME_MAX - 1 - ret);
244 ret = snprintf(prop_value, PROP_VALUE_MAX - 1, "value_%d.%d.%d", i, j, k);
245 memset(prop_value + ret, 'b', PROP_VALUE_MAX - 1 - ret);
246 prop_name[PROP_NAME_MAX - 1] = 0;
247 prop_value[PROP_VALUE_MAX - 1] = 0;
248
249 ASSERT_EQ(0, __system_property_add(prop_name, PROP_NAME_MAX - 1, prop_value, PROP_VALUE_MAX - 1));
250 }
251 }
252 }
253
254 for (int i = 0; i < 8; i++) {
255 for (int j = 0; j < 8; j++) {
256 for (int k = 0; k < 8; k++) {
257 ret = snprintf(prop_name, PROP_NAME_MAX - 1, "property_%d.%d.%d", i, j, k);
258 memset(prop_name + ret, 'a', PROP_NAME_MAX - 1 - ret);
259 ret = snprintf(prop_value, PROP_VALUE_MAX - 1, "value_%d.%d.%d", i, j, k);
260 memset(prop_value + ret, 'b', PROP_VALUE_MAX - 1 - ret);
261 prop_name[PROP_NAME_MAX - 1] = 0;
262 prop_value[PROP_VALUE_MAX - 1] = 0;
263 memset(prop_value_ret, '\0', PROP_VALUE_MAX);
264
265 ASSERT_EQ(PROP_VALUE_MAX - 1, __system_property_get(prop_name, prop_value_ret));
266 ASSERT_EQ(0, memcmp(prop_value, prop_value_ret, PROP_VALUE_MAX));
267 }
268 }
269 }
270
271 bool ok[8][8][8];
272 memset(ok, 0, sizeof(ok));
273 __system_property_foreach(hierarchical_test_callback, ok);
274
275 for (int i = 0; i < 8; i++) {
276 for (int j = 0; j < 8; j++) {
277 for (int k = 0; k < 8; k++) {
278 ASSERT_TRUE(ok[i][j][k]);
279 }
280 }
281 }
282}
283
Colin Crossb27e2002013-01-28 17:19:43 -0800284TEST(properties, errors) {
285 LocalPropertyTestState pa;
Greg Hackmanncb215a72013-02-13 14:41:48 -0800286 ASSERT_TRUE(pa.valid);
Colin Crossb27e2002013-01-28 17:19:43 -0800287 char prop_value[PROP_NAME_MAX];
288
289 ASSERT_EQ(0, __system_property_add("property", 8, "value1", 6));
290 ASSERT_EQ(0, __system_property_add("other_property", 14, "value2", 6));
291 ASSERT_EQ(0, __system_property_add("property_other", 14, "value3", 6));
292
293 ASSERT_EQ(0, __system_property_find("property1"));
294 ASSERT_EQ(0, __system_property_get("property1", prop_value));
295
296 ASSERT_EQ(-1, __system_property_add("name", PROP_NAME_MAX, "value", 5));
297 ASSERT_EQ(-1, __system_property_add("name", 4, "value", PROP_VALUE_MAX));
298 ASSERT_EQ(-1, __system_property_update(NULL, "value", PROP_VALUE_MAX));
299}
300
301TEST(properties, serial) {
302 LocalPropertyTestState pa;
Greg Hackmanncb215a72013-02-13 14:41:48 -0800303 ASSERT_TRUE(pa.valid);
Colin Crossb27e2002013-01-28 17:19:43 -0800304 const prop_info *pi;
305 unsigned int serial;
306
307 ASSERT_EQ(0, __system_property_add("property", 8, "value1", 6));
308 ASSERT_NE((const prop_info *)NULL, pi = __system_property_find("property"));
309 serial = __system_property_serial(pi);
310 ASSERT_EQ(0, __system_property_update((prop_info *)pi, "value2", 6));
311 ASSERT_NE(serial, __system_property_serial(pi));
312}
313
314static void *PropertyWaitHelperFn(void *arg)
315{
316 int *flag = (int *)arg;
317 prop_info *pi;
318 pi = (prop_info *)__system_property_find("property");
319 usleep(100000);
320
321 *flag = 1;
322 __system_property_update(pi, "value3", 6);
323
324 return NULL;
325}
326
327TEST(properties, wait) {
328 LocalPropertyTestState pa;
Greg Hackmanncb215a72013-02-13 14:41:48 -0800329 ASSERT_TRUE(pa.valid);
Colin Crossb27e2002013-01-28 17:19:43 -0800330 unsigned int serial;
331 prop_info *pi;
332 pthread_t t;
333 int flag = 0;
334
335 ASSERT_EQ(0, __system_property_add("property", 8, "value1", 6));
336 serial = __system_property_wait_any(0);
337 pi = (prop_info *)__system_property_find("property");
338 ASSERT_NE((prop_info *)NULL, pi);
339 __system_property_update(pi, "value2", 6);
340 serial = __system_property_wait_any(serial);
341
342 ASSERT_EQ(0, pthread_create(&t, NULL, PropertyWaitHelperFn, &flag));
343 ASSERT_EQ(flag, 0);
344 serial = __system_property_wait_any(serial);
345 ASSERT_EQ(flag, 1);
346
347 void* result;
348 ASSERT_EQ(0, pthread_join(t, &result));
349}
350
351class KilledByFault {
352 public:
353 explicit KilledByFault() {};
354 bool operator()(int exit_status) const;
355};
356
357bool KilledByFault::operator()(int exit_status) const {
358 return WIFSIGNALED(exit_status) &&
359 (WTERMSIG(exit_status) == SIGSEGV ||
360 WTERMSIG(exit_status) == SIGBUS ||
361 WTERMSIG(exit_status) == SIGABRT);
362}
363
364TEST(properties_DeathTest, read_only) {
Elliott Hughese4375192013-10-21 17:09:52 -0700365 ::testing::FLAGS_gtest_death_test_style = "threadsafe";
366
367 // This test only makes sense if we're talking to the real system property service.
368 struct stat sb;
369 if (stat(PROP_FILENAME, &sb) == -1 && errno == ENOENT) {
370 return;
371 }
372
373 ASSERT_EXIT(__system_property_add("property", 8, "value", 5), KilledByFault(), "");
Colin Crossb27e2002013-01-28 17:19:43 -0800374}
Elliott Hughese4375192013-10-21 17:09:52 -0700375
Colin Crossb27e2002013-01-28 17:19:43 -0800376#endif