1 /*
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 "BionicDeathTest.h"
19 
20 #include <errno.h>
21 #include <sys/wait.h>
22 #include <unistd.h>
23 
24 #include <string>
25 #include <thread>
26 
27 #include <android-base/file.h>
28 
29 using namespace std::literals;
30 
31 #if defined(__BIONIC__)
32 
33 #define _REALLY_INCLUDE_SYS__SYSTEM_PROPERTIES_H_
34 #include <sys/_system_properties.h>
35 
36 #include <system_properties/system_properties.h>
37 
38 class SystemPropertiesTest : public SystemProperties {
39  public:
SystemPropertiesTest()40   SystemPropertiesTest() : SystemProperties(false) {
41     valid_ = AreaInit(dir_.path, nullptr);
42   }
~SystemPropertiesTest()43   ~SystemPropertiesTest() {
44     if (valid_) {
45       contexts_->FreeAndUnmap();
46     }
47   }
48 
valid() const49   bool valid() const {
50     return valid_;
51   }
52 
53  private:
54   TemporaryDir dir_;
55   bool valid_;
56 };
57 
foreach_test_callback(const prop_info * pi,void * cookie)58 static void foreach_test_callback(const prop_info *pi, void* cookie) {
59     size_t *count = static_cast<size_t *>(cookie);
60 
61     ASSERT_TRUE(pi != nullptr);
62     (*count)++;
63 }
64 
hierarchical_test_callback(const prop_info * pi,void * cookie)65 static void hierarchical_test_callback(const prop_info *pi, void *cookie) {
66     bool (*ok)[8][8] = static_cast<bool (*)[8][8]>(cookie);
67 
68     char name[PROP_NAME_MAX];
69     char value[PROP_VALUE_MAX];
70 
71     __system_property_read(pi, name, value);
72 
73     int name_i, name_j, name_k;
74     int value_i, value_j, value_k;
75     ASSERT_EQ(3, sscanf(name, "property_%d.%d.%d", &name_i, &name_j, &name_k));
76     ASSERT_EQ(3, sscanf(value, "value_%d.%d.%d", &value_i, &value_j, &value_k));
77     ASSERT_EQ(name_i, value_i);
78     ASSERT_GE(name_i, 0);
79     ASSERT_LT(name_i, 8);
80     ASSERT_EQ(name_j, value_j);
81     ASSERT_GE(name_j, 0);
82     ASSERT_LT(name_j, 8);
83     ASSERT_EQ(name_k, value_k);
84     ASSERT_GE(name_k, 0);
85     ASSERT_LT(name_k, 8);
86 
87     ok[name_i][name_j][name_k] = true;
88 }
89 
90 #endif // __BIONIC__
91 
TEST(properties,__system_property_add)92 TEST(properties, __system_property_add) {
93 #if defined(__BIONIC__)
94     SystemPropertiesTest system_properties;
95     ASSERT_TRUE(system_properties.valid());
96 
97     ASSERT_EQ(0, system_properties.Add("property", 8, "value1", 6));
98     ASSERT_EQ(0, system_properties.Add("other_property", 14, "value2", 6));
99     ASSERT_EQ(0, system_properties.Add("property_other", 14, "value3", 6));
100 
101     // check that there is no limit on property name length
102     char name[PROP_NAME_MAX + 11];
103     name[0] = 'p';
104     for (size_t i = 1; i < sizeof(name); i++) {
105       name[i] = 'x';
106     }
107 
108     name[sizeof(name)-1] = '\0';
109     ASSERT_EQ(0, system_properties.Add(name, strlen(name), "value", 5));
110 
111     char propvalue[PROP_VALUE_MAX];
112     ASSERT_EQ(6, system_properties.Get("property", propvalue));
113     ASSERT_STREQ(propvalue, "value1");
114 
115     ASSERT_EQ(6, system_properties.Get("other_property", propvalue));
116     ASSERT_STREQ(propvalue, "value2");
117 
118     ASSERT_EQ(6, system_properties.Get("property_other", propvalue));
119     ASSERT_STREQ(propvalue, "value3");
120 
121     ASSERT_EQ(5, system_properties.Get(name, propvalue));
122     ASSERT_STREQ(propvalue, "value");
123 #else // __BIONIC__
124     GTEST_SKIP() << "bionic-only test";
125 #endif // __BIONIC__
126 }
127 
TEST(properties,__system_property_update)128 TEST(properties, __system_property_update) {
129 #if defined(__BIONIC__)
130     SystemPropertiesTest system_properties;
131     ASSERT_TRUE(system_properties.valid());
132 
133     ASSERT_EQ(0, system_properties.Add("property", 8, "oldvalue1", 9));
134     ASSERT_EQ(0, system_properties.Add("other_property", 14, "value2", 6));
135     ASSERT_EQ(0, system_properties.Add("property_other", 14, "value3", 6));
136 
137     const prop_info* pi = system_properties.Find("property");
138     ASSERT_TRUE(pi != nullptr);
139     system_properties.Update(const_cast<prop_info*>(pi), "value4", 6);
140 
141     pi = system_properties.Find("other_property");
142     ASSERT_TRUE(pi != nullptr);
143     system_properties.Update(const_cast<prop_info*>(pi), "newvalue5", 9);
144 
145     pi = system_properties.Find("property_other");
146     ASSERT_TRUE(pi != nullptr);
147     system_properties.Update(const_cast<prop_info*>(pi), "value6", 6);
148 
149     char propvalue[PROP_VALUE_MAX];
150     ASSERT_EQ(6, system_properties.Get("property", propvalue));
151     ASSERT_STREQ(propvalue, "value4");
152 
153     ASSERT_EQ(9, system_properties.Get("other_property", propvalue));
154     ASSERT_STREQ(propvalue, "newvalue5");
155 
156     ASSERT_EQ(6, system_properties.Get("property_other", propvalue));
157     ASSERT_STREQ(propvalue, "value6");
158 #else // __BIONIC__
159     GTEST_SKIP() << "bionic-only test";
160 #endif // __BIONIC__
161 }
162 
TEST(properties,fill)163 TEST(properties, fill) {
164 #if defined(__BIONIC__)
165     SystemPropertiesTest system_properties;
166     ASSERT_TRUE(system_properties.valid());
167 
168     char prop_name[PROP_NAME_MAX];
169     char prop_value[PROP_VALUE_MAX];
170     char prop_value_ret[PROP_VALUE_MAX];
171     int count = 0;
172     int ret;
173 
174     while (true) {
175         ret = snprintf(prop_name, PROP_NAME_MAX - 1, "property_%d", count);
176         memset(prop_name + ret, 'a', PROP_NAME_MAX - 1 - ret);
177         ret = snprintf(prop_value, PROP_VALUE_MAX - 1, "value_%d", count);
178         memset(prop_value + ret, 'b', PROP_VALUE_MAX - 1 - ret);
179         prop_name[PROP_NAME_MAX - 1] = 0;
180         prop_value[PROP_VALUE_MAX - 1] = 0;
181 
182         ret = system_properties.Add(prop_name, PROP_NAME_MAX - 1, prop_value, PROP_VALUE_MAX - 1);
183         if (ret < 0)
184             break;
185 
186         count++;
187     }
188 
189     // For historical reasons at least 247 properties must be supported
190     ASSERT_GE(count, 247);
191 
192     for (int i = 0; i < count; i++) {
193         ret = snprintf(prop_name, PROP_NAME_MAX - 1, "property_%d", i);
194         memset(prop_name + ret, 'a', PROP_NAME_MAX - 1 - ret);
195         ret = snprintf(prop_value, PROP_VALUE_MAX - 1, "value_%d", i);
196         memset(prop_value + ret, 'b', PROP_VALUE_MAX - 1 - ret);
197         prop_name[PROP_NAME_MAX - 1] = 0;
198         prop_value[PROP_VALUE_MAX - 1] = 0;
199         memset(prop_value_ret, '\0', PROP_VALUE_MAX);
200 
201         ASSERT_EQ(PROP_VALUE_MAX - 1, system_properties.Get(prop_name, prop_value_ret));
202         ASSERT_EQ(0, memcmp(prop_value, prop_value_ret, PROP_VALUE_MAX));
203     }
204 #else // __BIONIC__
205     GTEST_SKIP() << "bionic-only test";
206 #endif // __BIONIC__
207 }
208 
TEST(properties,__system_property_foreach)209 TEST(properties, __system_property_foreach) {
210 #if defined(__BIONIC__)
211     SystemPropertiesTest system_properties;
212     ASSERT_TRUE(system_properties.valid());
213 
214     ASSERT_EQ(0, system_properties.Add("property", 8, "value1", 6));
215     ASSERT_EQ(0, system_properties.Add("other_property", 14, "value2", 6));
216     ASSERT_EQ(0, system_properties.Add("property_other", 14, "value3", 6));
217 
218     size_t count = 0;
219     ASSERT_EQ(0, system_properties.Foreach(foreach_test_callback, &count));
220     ASSERT_EQ(3U, count);
221 #else // __BIONIC__
222     GTEST_SKIP() << "bionic-only test";
223 #endif // __BIONIC__
224 }
225 
TEST(properties,__system_property_find_nth)226 TEST(properties, __system_property_find_nth) {
227 #if defined(__BIONIC__)
228     SystemPropertiesTest system_properties;
229     ASSERT_TRUE(system_properties.valid());
230 
231     ASSERT_EQ(0, system_properties.Add("property", 8, "value1", 6));
232     ASSERT_EQ(0, system_properties.Add("other_property", 14, "value2", 6));
233     ASSERT_EQ(0, system_properties.Add("property_other", 14, "value3", 6));
234 
235     char name[PROP_NAME_MAX];
236     char value[PROP_VALUE_MAX];
237     EXPECT_EQ(6, system_properties.Read(system_properties.FindNth(0), name, value));
238     EXPECT_STREQ("property", name);
239     EXPECT_STREQ("value1", value);
240     EXPECT_EQ(6, system_properties.Read(system_properties.FindNth(1), name, value));
241     EXPECT_STREQ("other_property", name);
242     EXPECT_STREQ("value2", value);
243     EXPECT_EQ(6, system_properties.Read(system_properties.FindNth(2), name, value));
244     EXPECT_STREQ("property_other", name);
245     EXPECT_STREQ("value3", value);
246 
247     for (unsigned i = 3; i < 1024; ++i) {
248       ASSERT_TRUE(system_properties.FindNth(i) == nullptr);
249     }
250 #else // __BIONIC__
251     GTEST_SKIP() << "bionic-only test";
252 #endif // __BIONIC__
253 }
254 
TEST(properties,fill_hierarchical)255 TEST(properties, fill_hierarchical) {
256 #if defined(__BIONIC__)
257     SystemPropertiesTest system_properties;
258     ASSERT_TRUE(system_properties.valid());
259 
260     char prop_name[PROP_NAME_MAX];
261     char prop_value[PROP_VALUE_MAX];
262     char prop_value_ret[PROP_VALUE_MAX];
263     int ret;
264 
265     for (int i = 0; i < 8; i++) {
266         for (int j = 0; j < 8; j++) {
267             for (int k = 0; k < 8; k++) {
268                 ret = snprintf(prop_name, PROP_NAME_MAX - 1, "property_%d.%d.%d", i, j, k);
269                 memset(prop_name + ret, 'a', PROP_NAME_MAX - 1 - ret);
270                 ret = snprintf(prop_value, PROP_VALUE_MAX - 1, "value_%d.%d.%d", i, j, k);
271                 memset(prop_value + ret, 'b', PROP_VALUE_MAX - 1 - ret);
272                 prop_name[PROP_NAME_MAX - 1] = 0;
273                 prop_value[PROP_VALUE_MAX - 1] = 0;
274 
275                 ASSERT_EQ(0, system_properties.Add(
276                     prop_name, PROP_NAME_MAX - 1, prop_value, PROP_VALUE_MAX - 1));
277             }
278         }
279     }
280 
281     for (int i = 0; i < 8; i++) {
282         for (int j = 0; j < 8; j++) {
283             for (int k = 0; k < 8; k++) {
284                 ret = snprintf(prop_name, PROP_NAME_MAX - 1, "property_%d.%d.%d", i, j, k);
285                 memset(prop_name + ret, 'a', PROP_NAME_MAX - 1 - ret);
286                 ret = snprintf(prop_value, PROP_VALUE_MAX - 1, "value_%d.%d.%d", i, j, k);
287                 memset(prop_value + ret, 'b', PROP_VALUE_MAX - 1 - ret);
288                 prop_name[PROP_NAME_MAX - 1] = 0;
289                 prop_value[PROP_VALUE_MAX - 1] = 0;
290                 memset(prop_value_ret, '\0', PROP_VALUE_MAX);
291 
292                 ASSERT_EQ(PROP_VALUE_MAX - 1, system_properties.Get(prop_name, prop_value_ret));
293                 ASSERT_EQ(0, memcmp(prop_value, prop_value_ret, PROP_VALUE_MAX));
294             }
295         }
296     }
297 
298     bool ok[8][8][8];
299     memset(ok, 0, sizeof(ok));
300     system_properties.Foreach(hierarchical_test_callback, ok);
301 
302     for (int i = 0; i < 8; i++) {
303         for (int j = 0; j < 8; j++) {
304             for (int k = 0; k < 8; k++) {
305                 ASSERT_TRUE(ok[i][j][k]);
306             }
307         }
308     }
309 #else // __BIONIC__
310     GTEST_SKIP() << "bionic-only test";
311 #endif // __BIONIC__
312 }
313 
TEST(properties,errors)314 TEST(properties, errors) {
315 #if defined(__BIONIC__)
316     SystemPropertiesTest system_properties;
317     ASSERT_TRUE(system_properties.valid());
318 
319     char prop_value[PROP_NAME_MAX];
320 
321     ASSERT_EQ(0, system_properties.Add("property", 8, "value1", 6));
322     ASSERT_EQ(0, system_properties.Add("other_property", 14, "value2", 6));
323     ASSERT_EQ(0, system_properties.Add("property_other", 14, "value3", 6));
324 
325     ASSERT_EQ(0, system_properties.Find("property1"));
326     ASSERT_EQ(0, system_properties.Get("property1", prop_value));
327 
328     ASSERT_EQ(-1, system_properties.Add("name", 4, "value", PROP_VALUE_MAX));
329     ASSERT_EQ(-1, system_properties.Update(NULL, "value", PROP_VALUE_MAX));
330 #else // __BIONIC__
331     GTEST_SKIP() << "bionic-only test";
332 #endif // __BIONIC__
333 }
334 
TEST(properties,__system_property_serial)335 TEST(properties, __system_property_serial) {
336 #if defined(__BIONIC__)
337     SystemPropertiesTest system_properties;
338     ASSERT_TRUE(system_properties.valid());
339 
340     ASSERT_EQ(0, system_properties.Add("property", 8, "value1", 6));
341     const prop_info* pi = system_properties.Find("property");
342     ASSERT_TRUE(pi != nullptr);
343     unsigned serial = __system_property_serial(pi);
344     ASSERT_EQ(0, system_properties.Update(const_cast<prop_info*>(pi), "value2", 6));
345     ASSERT_NE(serial, __system_property_serial(pi));
346 #else // __BIONIC__
347     GTEST_SKIP() << "bionic-only test";
348 #endif // __BIONIC__
349 }
350 
TEST(properties,__system_property_wait_any)351 TEST(properties, __system_property_wait_any) {
352 #if defined(__BIONIC__)
353     SystemPropertiesTest system_properties;
354     ASSERT_TRUE(system_properties.valid());
355 
356     ASSERT_EQ(0, system_properties.Add("property", 8, "value1", 6));
357     unsigned serial = system_properties.WaitAny(0);
358 
359     prop_info* pi = const_cast<prop_info*>(system_properties.Find("property"));
360     ASSERT_TRUE(pi != nullptr);
361     system_properties.Update(pi, "value2", 6);
362     serial = system_properties.WaitAny(serial);
363 
364     int flag = 0;
365     std::thread thread([&system_properties, &flag]() {
366         prop_info* pi = const_cast<prop_info*>(system_properties.Find("property"));
367         usleep(100000);
368 
369         flag = 1;
370         system_properties.Update(pi, "value3", 6);
371     });
372     ASSERT_EQ(flag, 0);
373     serial = system_properties.WaitAny(serial);
374     ASSERT_EQ(flag, 1);
375 
376     thread.join();
377 #else // __BIONIC__
378     GTEST_SKIP() << "bionic-only test";
379 #endif // __BIONIC__
380 }
381 
TEST(properties,__system_property_wait)382 TEST(properties, __system_property_wait) {
383 #if defined(__BIONIC__)
384     SystemPropertiesTest system_properties;
385     ASSERT_TRUE(system_properties.valid());
386 
387     ASSERT_EQ(0, system_properties.Add("property", 8, "value1", 6));
388 
389     prop_info* pi = const_cast<prop_info*>(system_properties.Find("property"));
390     ASSERT_TRUE(pi != nullptr);
391 
392     unsigned serial = __system_property_serial(pi);
393 
394     std::thread thread([&system_properties]() {
395         prop_info* pi = const_cast<prop_info*>(system_properties.Find("property"));
396         ASSERT_TRUE(pi != nullptr);
397 
398         system_properties.Update(pi, "value2", 6);
399     });
400 
401     uint32_t new_serial;
402     system_properties.Wait(pi, serial, &new_serial, nullptr);
403     ASSERT_GT(new_serial, serial);
404 
405     char value[PROP_VALUE_MAX];
406     ASSERT_EQ(6, system_properties.Get("property", value));
407     ASSERT_STREQ("value2", value);
408 
409     thread.join();
410 #else // __BIONIC__
411     GTEST_SKIP() << "bionic-only test";
412 #endif // __BIONIC__
413 }
414 
415 class KilledByFault {
416     public:
KilledByFault()417         explicit KilledByFault() {};
418         bool operator()(int exit_status) const;
419 };
420 
operator ()(int exit_status) const421 bool KilledByFault::operator()(int exit_status) const {
422     return WIFSIGNALED(exit_status) &&
423         (WTERMSIG(exit_status) == SIGSEGV ||
424          WTERMSIG(exit_status) == SIGBUS ||
425          WTERMSIG(exit_status) == SIGABRT);
426 }
427 
428 class properties_DeathTest : public BionicDeathTest {};
429 
TEST_F(properties_DeathTest,read_only)430 TEST_F(properties_DeathTest, read_only) {
431 #if defined(__BIONIC__)
432 
433   // This test only makes sense if we're talking to the real system property service.
434   struct stat sb;
435   ASSERT_FALSE(stat(PROP_FILENAME, &sb) == -1 && errno == ENOENT);
436 
437   ASSERT_EXIT(__system_property_add("property", 8, "value", 5), KilledByFault(), "");
438 #else // __BIONIC__
439   GTEST_SKIP() << "bionic-only test";
440 #endif // __BIONIC__
441 }
442 
TEST(properties,__system_property_extra_long_read_only)443 TEST(properties, __system_property_extra_long_read_only) {
444 #if defined(__BIONIC__)
445   SystemPropertiesTest system_properties;
446   ASSERT_TRUE(system_properties.valid());
447 
448   std::vector<std::pair<std::string, std::string>> short_properties = {
449     { "ro.0char", std::string() },
450     { "ro.50char", std::string(50, 'x') },
451     { "ro.91char", std::string(91, 'x') },
452   };
453 
454   std::vector<std::pair<std::string, std::string>> long_properties = {
455     { "ro.92char", std::string(92, 'x') },
456     { "ro.93char", std::string(93, 'x') },
457     { "ro.1000char", std::string(1000, 'x') },
458   };
459 
460   for (const auto& property : short_properties) {
461     const std::string& name = property.first;
462     const std::string& value = property.second;
463     ASSERT_EQ(0, system_properties.Add(name.c_str(), name.size(), value.c_str(), value.size()));
464   }
465 
466   for (const auto& property : long_properties) {
467     const std::string& name = property.first;
468     const std::string& value = property.second;
469     ASSERT_EQ(0, system_properties.Add(name.c_str(), name.size(), value.c_str(), value.size()));
470   }
471 
472   auto check_with_legacy_read = [&system_properties](const std::string& name,
473                                                      const std::string& expected_value) {
474     char value[PROP_VALUE_MAX];
475     EXPECT_EQ(static_cast<int>(expected_value.size()), system_properties.Get(name.c_str(), value))
476         << name;
477     EXPECT_EQ(expected_value, value) << name;
478   };
479 
480   auto check_with_read_callback = [&system_properties](const std::string& name,
481                                                        const std::string& expected_value) {
482     const prop_info* pi = system_properties.Find(name.c_str());
483     ASSERT_NE(nullptr, pi);
484     std::string value;
485     system_properties.ReadCallback(pi,
486                                    [](void* cookie, const char*, const char* value, uint32_t) {
487                                      auto* out_value = reinterpret_cast<std::string*>(cookie);
488                                      *out_value = value;
489                                    },
490                                    &value);
491     EXPECT_EQ(expected_value, value) << name;
492   };
493 
494   for (const auto& property : short_properties) {
495     const std::string& name = property.first;
496     const std::string& value = property.second;
497     check_with_legacy_read(name, value);
498     check_with_read_callback(name, value);
499   }
500 
501   constexpr static const char* kExtraLongLegacyError =
502       "Must use __system_property_read_callback() to read";
503   for (const auto& property : long_properties) {
504     const std::string& name = property.first;
505     const std::string& value = property.second;
506     check_with_legacy_read(name, kExtraLongLegacyError);
507     check_with_read_callback(name, value);
508   }
509 
510 #else   // __BIONIC__
511   GTEST_SKIP() << "bionic-only test";
512 #endif  // __BIONIC__
513 }
514 
515 // pa_size is 128 * 1024 currently, if a property is longer then we expect it to fail gracefully.
TEST(properties,__system_property_extra_long_read_only_too_long)516 TEST(properties, __system_property_extra_long_read_only_too_long) {
517 #if defined(__BIONIC__)
518   SystemPropertiesTest system_properties;
519   ASSERT_TRUE(system_properties.valid());
520 
521   auto name = "ro.super_long_property"s;
522   auto value = std::string(128 * 1024 + 1, 'x');
523   ASSERT_NE(0, system_properties.Add(name.c_str(), name.size(), value.c_str(), value.size()));
524 
525 #else   // __BIONIC__
526   GTEST_SKIP() << "bionic-only test";
527 #endif  // __BIONIC__
528 }
529