| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 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> | 
| Yabin Cui | 9df7040 | 2014-11-05 18:01:01 -0800 | [diff] [blame] | 18 | #include "BionicDeathTest.h" | 
|  | 19 |  | 
| Elliott Hughes | e437519 | 2013-10-21 17:09:52 -0700 | [diff] [blame] | 20 | #include <errno.h> | 
| Yabin Cui | 9df7040 | 2014-11-05 18:01:01 -0800 | [diff] [blame] | 21 | #include <sys/wait.h> | 
| Greg Hackmann | cb215a7 | 2013-02-13 14:41:48 -0800 | [diff] [blame] | 22 | #include <unistd.h> | 
|  | 23 | #include <string> | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 24 |  | 
| Christopher Ferris | f04935c | 2013-12-20 18:43:21 -0800 | [diff] [blame] | 25 | #if defined(__BIONIC__) | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 26 |  | 
|  | 27 | #define _REALLY_INCLUDE_SYS__SYSTEM_PROPERTIES_H_ | 
|  | 28 | #include <sys/_system_properties.h> | 
|  | 29 |  | 
| Greg Hackmann | 1540f60 | 2013-06-19 13:31:21 -0700 | [diff] [blame] | 30 | extern void *__system_property_area__; | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 31 |  | 
|  | 32 | struct LocalPropertyTestState { | 
| Greg Hackmann | cb215a7 | 2013-02-13 14:41:48 -0800 | [diff] [blame] | 33 | LocalPropertyTestState() : valid(false) { | 
| Elliott Hughes | e437519 | 2013-10-21 17:09:52 -0700 | [diff] [blame] | 34 | const char* ANDROID_DATA = getenv("ANDROID_DATA"); | 
|  | 35 | char dir_template[PATH_MAX]; | 
|  | 36 | snprintf(dir_template, sizeof(dir_template), "%s/local/tmp/prop-XXXXXX", ANDROID_DATA); | 
|  | 37 | char* dirname = mkdtemp(dir_template); | 
| Greg Hackmann | cb215a7 | 2013-02-13 14:41:48 -0800 | [diff] [blame] | 38 | if (!dirname) { | 
| Elliott Hughes | e437519 | 2013-10-21 17:09:52 -0700 | [diff] [blame] | 39 | fprintf(stderr, "making temp file for test state failed (is %s writable?): %s", | 
|  | 40 | dir_template, strerror(errno)); | 
| Greg Hackmann | cb215a7 | 2013-02-13 14:41:48 -0800 | [diff] [blame] | 41 | return; | 
|  | 42 | } | 
|  | 43 |  | 
| Greg Hackmann | cb215a7 | 2013-02-13 14:41:48 -0800 | [diff] [blame] | 44 | pa_dirname = dirname; | 
|  | 45 | pa_filename = pa_dirname + "/__properties__"; | 
|  | 46 |  | 
|  | 47 | __system_property_set_filename(pa_filename.c_str()); | 
|  | 48 | __system_property_area_init(); | 
|  | 49 | valid = true; | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 50 | } | 
|  | 51 |  | 
|  | 52 | ~LocalPropertyTestState() { | 
| Elliott Hughes | e437519 | 2013-10-21 17:09:52 -0700 | [diff] [blame] | 53 | if (!valid) { | 
| Greg Hackmann | cb215a7 | 2013-02-13 14:41:48 -0800 | [diff] [blame] | 54 | return; | 
| Elliott Hughes | e437519 | 2013-10-21 17:09:52 -0700 | [diff] [blame] | 55 | } | 
| Greg Hackmann | cb215a7 | 2013-02-13 14:41:48 -0800 | [diff] [blame] | 56 |  | 
| Greg Hackmann | cb215a7 | 2013-02-13 14:41:48 -0800 | [diff] [blame] | 57 | __system_property_set_filename(PROP_FILENAME); | 
| Tom Cherry | 49a309f | 2015-09-23 16:09:47 -0700 | [diff] [blame] | 58 | __system_properties_init(); | 
| Greg Hackmann | cb215a7 | 2013-02-13 14:41:48 -0800 | [diff] [blame] | 59 | unlink(pa_filename.c_str()); | 
|  | 60 | rmdir(pa_dirname.c_str()); | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 61 | } | 
| Greg Hackmann | cb215a7 | 2013-02-13 14:41:48 -0800 | [diff] [blame] | 62 | public: | 
|  | 63 | bool valid; | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 64 | private: | 
| Greg Hackmann | cb215a7 | 2013-02-13 14:41:48 -0800 | [diff] [blame] | 65 | std::string pa_dirname; | 
|  | 66 | std::string pa_filename; | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 67 | }; | 
|  | 68 |  | 
| Christopher Ferris | f04935c | 2013-12-20 18:43:21 -0800 | [diff] [blame] | 69 | static void foreach_test_callback(const prop_info *pi, void* cookie) { | 
|  | 70 | size_t *count = static_cast<size_t *>(cookie); | 
|  | 71 |  | 
| Dimitry Ivanov | 16b2a4d | 2017-01-24 20:43:29 +0000 | [diff] [blame] | 72 | ASSERT_TRUE(pi != nullptr); | 
| Christopher Ferris | f04935c | 2013-12-20 18:43:21 -0800 | [diff] [blame] | 73 | (*count)++; | 
|  | 74 | } | 
|  | 75 |  | 
|  | 76 | static void hierarchical_test_callback(const prop_info *pi, void *cookie) { | 
|  | 77 | bool (*ok)[8][8] = static_cast<bool (*)[8][8]>(cookie); | 
|  | 78 |  | 
|  | 79 | char name[PROP_NAME_MAX]; | 
|  | 80 | char value[PROP_VALUE_MAX]; | 
|  | 81 |  | 
|  | 82 | __system_property_read(pi, name, value); | 
|  | 83 |  | 
|  | 84 | int name_i, name_j, name_k; | 
|  | 85 | int value_i, value_j, value_k; | 
|  | 86 | ASSERT_EQ(3, sscanf(name, "property_%d.%d.%d", &name_i, &name_j, &name_k)); | 
|  | 87 | ASSERT_EQ(3, sscanf(value, "value_%d.%d.%d", &value_i, &value_j, &value_k)); | 
|  | 88 | ASSERT_EQ(name_i, value_i); | 
|  | 89 | ASSERT_GE(name_i, 0); | 
|  | 90 | ASSERT_LT(name_i, 8); | 
|  | 91 | ASSERT_EQ(name_j, value_j); | 
|  | 92 | ASSERT_GE(name_j, 0); | 
|  | 93 | ASSERT_LT(name_j, 8); | 
|  | 94 | ASSERT_EQ(name_k, value_k); | 
|  | 95 | ASSERT_GE(name_k, 0); | 
|  | 96 | ASSERT_LT(name_k, 8); | 
|  | 97 |  | 
|  | 98 | ok[name_i][name_j][name_k] = true; | 
|  | 99 | } | 
|  | 100 |  | 
| Dimitry Ivanov | 16b2a4d | 2017-01-24 20:43:29 +0000 | [diff] [blame] | 101 | static void* PropertyWaitHelperFn(void* arg) { | 
|  | 102 | int* flag = static_cast<int*>(arg); | 
|  | 103 | prop_info* pi = const_cast<prop_info*>(__system_property_find("property")); | 
| Christopher Ferris | f04935c | 2013-12-20 18:43:21 -0800 | [diff] [blame] | 104 | usleep(100000); | 
|  | 105 |  | 
|  | 106 | *flag = 1; | 
|  | 107 | __system_property_update(pi, "value3", 6); | 
|  | 108 |  | 
| Dimitry Ivanov | 16b2a4d | 2017-01-24 20:43:29 +0000 | [diff] [blame] | 109 | return nullptr; | 
| Christopher Ferris | f04935c | 2013-12-20 18:43:21 -0800 | [diff] [blame] | 110 | } | 
|  | 111 |  | 
|  | 112 | #endif // __BIONIC__ | 
|  | 113 |  | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 114 | TEST(properties, add) { | 
| Christopher Ferris | f04935c | 2013-12-20 18:43:21 -0800 | [diff] [blame] | 115 | #if defined(__BIONIC__) | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 116 | LocalPropertyTestState pa; | 
| Greg Hackmann | cb215a7 | 2013-02-13 14:41:48 -0800 | [diff] [blame] | 117 | ASSERT_TRUE(pa.valid); | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 118 |  | 
|  | 119 | char propvalue[PROP_VALUE_MAX]; | 
|  | 120 |  | 
|  | 121 | ASSERT_EQ(0, __system_property_add("property", 8, "value1", 6)); | 
|  | 122 | ASSERT_EQ(0, __system_property_add("other_property", 14, "value2", 6)); | 
|  | 123 | ASSERT_EQ(0, __system_property_add("property_other", 14, "value3", 6)); | 
|  | 124 |  | 
| Dimitry Ivanov | 16b2a4d | 2017-01-24 20:43:29 +0000 | [diff] [blame] | 125 | // check that there is no limit on property name length | 
|  | 126 | char name[PROP_NAME_MAX + 11]; | 
|  | 127 | name[0] = 'p'; | 
|  | 128 | for (size_t i = 1; i < sizeof(name); i++) { | 
|  | 129 | name[i] = 'x'; | 
|  | 130 | } | 
|  | 131 |  | 
|  | 132 | name[sizeof(name)-1] = '\0'; | 
|  | 133 | ASSERT_EQ(0, __system_property_add(name, strlen(name), "value", 5)); | 
|  | 134 |  | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 135 | ASSERT_EQ(6, __system_property_get("property", propvalue)); | 
|  | 136 | ASSERT_STREQ(propvalue, "value1"); | 
|  | 137 |  | 
|  | 138 | ASSERT_EQ(6, __system_property_get("other_property", propvalue)); | 
|  | 139 | ASSERT_STREQ(propvalue, "value2"); | 
|  | 140 |  | 
|  | 141 | ASSERT_EQ(6, __system_property_get("property_other", propvalue)); | 
|  | 142 | ASSERT_STREQ(propvalue, "value3"); | 
| Dimitry Ivanov | 16b2a4d | 2017-01-24 20:43:29 +0000 | [diff] [blame] | 143 |  | 
|  | 144 | ASSERT_EQ(5, __system_property_get(name, propvalue)); | 
|  | 145 | ASSERT_STREQ(propvalue, "value"); | 
| Christopher Ferris | f04935c | 2013-12-20 18:43:21 -0800 | [diff] [blame] | 146 | #else // __BIONIC__ | 
|  | 147 | GTEST_LOG_(INFO) << "This test does nothing.\n"; | 
|  | 148 | #endif // __BIONIC__ | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 149 | } | 
|  | 150 |  | 
|  | 151 | TEST(properties, update) { | 
| Christopher Ferris | f04935c | 2013-12-20 18:43:21 -0800 | [diff] [blame] | 152 | #if defined(__BIONIC__) | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 153 | LocalPropertyTestState pa; | 
| Greg Hackmann | cb215a7 | 2013-02-13 14:41:48 -0800 | [diff] [blame] | 154 | ASSERT_TRUE(pa.valid); | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 155 |  | 
|  | 156 | char propvalue[PROP_VALUE_MAX]; | 
|  | 157 | prop_info *pi; | 
|  | 158 |  | 
|  | 159 | ASSERT_EQ(0, __system_property_add("property", 8, "oldvalue1", 9)); | 
|  | 160 | ASSERT_EQ(0, __system_property_add("other_property", 14, "value2", 6)); | 
|  | 161 | ASSERT_EQ(0, __system_property_add("property_other", 14, "value3", 6)); | 
|  | 162 |  | 
|  | 163 | pi = (prop_info *)__system_property_find("property"); | 
|  | 164 | ASSERT_NE((prop_info *)NULL, pi); | 
|  | 165 | __system_property_update(pi, "value4", 6); | 
|  | 166 |  | 
|  | 167 | pi = (prop_info *)__system_property_find("other_property"); | 
|  | 168 | ASSERT_NE((prop_info *)NULL, pi); | 
|  | 169 | __system_property_update(pi, "newvalue5", 9); | 
|  | 170 |  | 
|  | 171 | pi = (prop_info *)__system_property_find("property_other"); | 
|  | 172 | ASSERT_NE((prop_info *)NULL, pi); | 
|  | 173 | __system_property_update(pi, "value6", 6); | 
|  | 174 |  | 
|  | 175 | ASSERT_EQ(6, __system_property_get("property", propvalue)); | 
|  | 176 | ASSERT_STREQ(propvalue, "value4"); | 
|  | 177 |  | 
|  | 178 | ASSERT_EQ(9, __system_property_get("other_property", propvalue)); | 
|  | 179 | ASSERT_STREQ(propvalue, "newvalue5"); | 
|  | 180 |  | 
|  | 181 | ASSERT_EQ(6, __system_property_get("property_other", propvalue)); | 
|  | 182 | ASSERT_STREQ(propvalue, "value6"); | 
| Christopher Ferris | f04935c | 2013-12-20 18:43:21 -0800 | [diff] [blame] | 183 | #else // __BIONIC__ | 
|  | 184 | GTEST_LOG_(INFO) << "This test does nothing.\n"; | 
|  | 185 | #endif // __BIONIC__ | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 186 | } | 
|  | 187 |  | 
| Greg Hackmann | cb215a7 | 2013-02-13 14:41:48 -0800 | [diff] [blame] | 188 | TEST(properties, fill) { | 
| Christopher Ferris | f04935c | 2013-12-20 18:43:21 -0800 | [diff] [blame] | 189 | #if defined(__BIONIC__) | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 190 | LocalPropertyTestState pa; | 
| Greg Hackmann | cb215a7 | 2013-02-13 14:41:48 -0800 | [diff] [blame] | 191 | ASSERT_TRUE(pa.valid); | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 192 | char prop_name[PROP_NAME_MAX]; | 
|  | 193 | char prop_value[PROP_VALUE_MAX]; | 
|  | 194 | char prop_value_ret[PROP_VALUE_MAX]; | 
| Greg Hackmann | cb215a7 | 2013-02-13 14:41:48 -0800 | [diff] [blame] | 195 | int count = 0; | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 196 | int ret; | 
|  | 197 |  | 
| Greg Hackmann | cb215a7 | 2013-02-13 14:41:48 -0800 | [diff] [blame] | 198 | while (true) { | 
|  | 199 | ret = snprintf(prop_name, PROP_NAME_MAX - 1, "property_%d", count); | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 200 | memset(prop_name + ret, 'a', PROP_NAME_MAX - 1 - ret); | 
| Greg Hackmann | cb215a7 | 2013-02-13 14:41:48 -0800 | [diff] [blame] | 201 | ret = snprintf(prop_value, PROP_VALUE_MAX - 1, "value_%d", count); | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 202 | memset(prop_value + ret, 'b', PROP_VALUE_MAX - 1 - ret); | 
|  | 203 | prop_name[PROP_NAME_MAX - 1] = 0; | 
|  | 204 | prop_value[PROP_VALUE_MAX - 1] = 0; | 
|  | 205 |  | 
| Greg Hackmann | cb215a7 | 2013-02-13 14:41:48 -0800 | [diff] [blame] | 206 | ret = __system_property_add(prop_name, PROP_NAME_MAX - 1, prop_value, PROP_VALUE_MAX - 1); | 
|  | 207 | if (ret < 0) | 
|  | 208 | break; | 
|  | 209 |  | 
|  | 210 | count++; | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 211 | } | 
|  | 212 |  | 
| Greg Hackmann | cb215a7 | 2013-02-13 14:41:48 -0800 | [diff] [blame] | 213 | // For historical reasons at least 247 properties must be supported | 
|  | 214 | ASSERT_GE(count, 247); | 
|  | 215 |  | 
|  | 216 | for (int i = 0; i < count; i++) { | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 217 | ret = snprintf(prop_name, PROP_NAME_MAX - 1, "property_%d", i); | 
|  | 218 | memset(prop_name + ret, 'a', PROP_NAME_MAX - 1 - ret); | 
|  | 219 | ret = snprintf(prop_value, PROP_VALUE_MAX - 1, "value_%d", i); | 
|  | 220 | memset(prop_value + ret, 'b', PROP_VALUE_MAX - 1 - ret); | 
|  | 221 | prop_name[PROP_NAME_MAX - 1] = 0; | 
|  | 222 | prop_value[PROP_VALUE_MAX - 1] = 0; | 
|  | 223 | memset(prop_value_ret, '\0', PROP_VALUE_MAX); | 
|  | 224 |  | 
|  | 225 | ASSERT_EQ(PROP_VALUE_MAX - 1, __system_property_get(prop_name, prop_value_ret)); | 
|  | 226 | ASSERT_EQ(0, memcmp(prop_value, prop_value_ret, PROP_VALUE_MAX)); | 
|  | 227 | } | 
| Christopher Ferris | f04935c | 2013-12-20 18:43:21 -0800 | [diff] [blame] | 228 | #else // __BIONIC__ | 
|  | 229 | GTEST_LOG_(INFO) << "This test does nothing.\n"; | 
|  | 230 | #endif // __BIONIC__ | 
| Greg Hackmann | c6ff844 | 2013-02-12 16:39:31 -0800 | [diff] [blame] | 231 | } | 
|  | 232 |  | 
|  | 233 | TEST(properties, foreach) { | 
| Christopher Ferris | f04935c | 2013-12-20 18:43:21 -0800 | [diff] [blame] | 234 | #if defined(__BIONIC__) | 
| Greg Hackmann | c6ff844 | 2013-02-12 16:39:31 -0800 | [diff] [blame] | 235 | LocalPropertyTestState pa; | 
| Greg Hackmann | cb215a7 | 2013-02-13 14:41:48 -0800 | [diff] [blame] | 236 | ASSERT_TRUE(pa.valid); | 
| Greg Hackmann | c6ff844 | 2013-02-12 16:39:31 -0800 | [diff] [blame] | 237 | size_t count = 0; | 
|  | 238 |  | 
|  | 239 | ASSERT_EQ(0, __system_property_add("property", 8, "value1", 6)); | 
|  | 240 | ASSERT_EQ(0, __system_property_add("other_property", 14, "value2", 6)); | 
|  | 241 | ASSERT_EQ(0, __system_property_add("property_other", 14, "value3", 6)); | 
|  | 242 |  | 
|  | 243 | ASSERT_EQ(0, __system_property_foreach(foreach_test_callback, &count)); | 
|  | 244 | ASSERT_EQ(3U, count); | 
| Christopher Ferris | f04935c | 2013-12-20 18:43:21 -0800 | [diff] [blame] | 245 | #else // __BIONIC__ | 
|  | 246 | GTEST_LOG_(INFO) << "This test does nothing.\n"; | 
|  | 247 | #endif // __BIONIC__ | 
| Greg Hackmann | c6ff844 | 2013-02-12 16:39:31 -0800 | [diff] [blame] | 248 | } | 
|  | 249 |  | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 250 | TEST(properties, find_nth) { | 
| Christopher Ferris | f04935c | 2013-12-20 18:43:21 -0800 | [diff] [blame] | 251 | #if defined(__BIONIC__) | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 252 | LocalPropertyTestState pa; | 
| Greg Hackmann | cb215a7 | 2013-02-13 14:41:48 -0800 | [diff] [blame] | 253 | ASSERT_TRUE(pa.valid); | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 254 |  | 
|  | 255 | ASSERT_EQ(0, __system_property_add("property", 8, "value1", 6)); | 
|  | 256 | ASSERT_EQ(0, __system_property_add("other_property", 14, "value2", 6)); | 
|  | 257 | ASSERT_EQ(0, __system_property_add("property_other", 14, "value3", 6)); | 
|  | 258 |  | 
| Dimitry Ivanov | 581b9f6 | 2017-01-09 11:05:52 -0800 | [diff] [blame] | 259 | // This method is no longer supported and should result in abort | 
|  | 260 | ASSERT_EXIT(__system_property_find_nth(0), testing::KilledBySignal(SIGABRT), | 
|  | 261 | "__system_property_find_nth is not supported since Android O," | 
|  | 262 | " please use __system_property_foreach instead."); | 
| Christopher Ferris | f04935c | 2013-12-20 18:43:21 -0800 | [diff] [blame] | 263 | #else // __BIONIC__ | 
|  | 264 | GTEST_LOG_(INFO) << "This test does nothing.\n"; | 
|  | 265 | #endif // __BIONIC__ | 
| Greg Hackmann | d527642 | 2013-06-17 12:37:09 -0700 | [diff] [blame] | 266 | } | 
|  | 267 |  | 
|  | 268 | TEST(properties, fill_hierarchical) { | 
| Christopher Ferris | f04935c | 2013-12-20 18:43:21 -0800 | [diff] [blame] | 269 | #if defined(__BIONIC__) | 
| Greg Hackmann | d527642 | 2013-06-17 12:37:09 -0700 | [diff] [blame] | 270 | LocalPropertyTestState pa; | 
|  | 271 | ASSERT_TRUE(pa.valid); | 
|  | 272 | char prop_name[PROP_NAME_MAX]; | 
|  | 273 | char prop_value[PROP_VALUE_MAX]; | 
|  | 274 | char prop_value_ret[PROP_VALUE_MAX]; | 
|  | 275 | int ret; | 
|  | 276 |  | 
|  | 277 | for (int i = 0; i < 8; i++) { | 
|  | 278 | for (int j = 0; j < 8; j++) { | 
|  | 279 | for (int k = 0; k < 8; k++) { | 
|  | 280 | ret = snprintf(prop_name, PROP_NAME_MAX - 1, "property_%d.%d.%d", i, j, k); | 
|  | 281 | memset(prop_name + ret, 'a', PROP_NAME_MAX - 1 - ret); | 
|  | 282 | ret = snprintf(prop_value, PROP_VALUE_MAX - 1, "value_%d.%d.%d", i, j, k); | 
|  | 283 | memset(prop_value + ret, 'b', PROP_VALUE_MAX - 1 - ret); | 
|  | 284 | prop_name[PROP_NAME_MAX - 1] = 0; | 
|  | 285 | prop_value[PROP_VALUE_MAX - 1] = 0; | 
|  | 286 |  | 
|  | 287 | ASSERT_EQ(0, __system_property_add(prop_name, PROP_NAME_MAX - 1, prop_value, PROP_VALUE_MAX - 1)); | 
|  | 288 | } | 
|  | 289 | } | 
|  | 290 | } | 
|  | 291 |  | 
|  | 292 | for (int i = 0; i < 8; i++) { | 
|  | 293 | for (int j = 0; j < 8; j++) { | 
|  | 294 | for (int k = 0; k < 8; k++) { | 
|  | 295 | ret = snprintf(prop_name, PROP_NAME_MAX - 1, "property_%d.%d.%d", i, j, k); | 
|  | 296 | memset(prop_name + ret, 'a', PROP_NAME_MAX - 1 - ret); | 
|  | 297 | ret = snprintf(prop_value, PROP_VALUE_MAX - 1, "value_%d.%d.%d", i, j, k); | 
|  | 298 | memset(prop_value + ret, 'b', PROP_VALUE_MAX - 1 - ret); | 
|  | 299 | prop_name[PROP_NAME_MAX - 1] = 0; | 
|  | 300 | prop_value[PROP_VALUE_MAX - 1] = 0; | 
|  | 301 | memset(prop_value_ret, '\0', PROP_VALUE_MAX); | 
|  | 302 |  | 
|  | 303 | ASSERT_EQ(PROP_VALUE_MAX - 1, __system_property_get(prop_name, prop_value_ret)); | 
|  | 304 | ASSERT_EQ(0, memcmp(prop_value, prop_value_ret, PROP_VALUE_MAX)); | 
|  | 305 | } | 
|  | 306 | } | 
|  | 307 | } | 
|  | 308 |  | 
|  | 309 | bool ok[8][8][8]; | 
|  | 310 | memset(ok, 0, sizeof(ok)); | 
|  | 311 | __system_property_foreach(hierarchical_test_callback, ok); | 
|  | 312 |  | 
|  | 313 | for (int i = 0; i < 8; i++) { | 
|  | 314 | for (int j = 0; j < 8; j++) { | 
|  | 315 | for (int k = 0; k < 8; k++) { | 
|  | 316 | ASSERT_TRUE(ok[i][j][k]); | 
|  | 317 | } | 
|  | 318 | } | 
|  | 319 | } | 
| Christopher Ferris | f04935c | 2013-12-20 18:43:21 -0800 | [diff] [blame] | 320 | #else // __BIONIC__ | 
|  | 321 | GTEST_LOG_(INFO) << "This test does nothing.\n"; | 
|  | 322 | #endif // __BIONIC__ | 
| Greg Hackmann | d527642 | 2013-06-17 12:37:09 -0700 | [diff] [blame] | 323 | } | 
|  | 324 |  | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 325 | TEST(properties, errors) { | 
| Christopher Ferris | f04935c | 2013-12-20 18:43:21 -0800 | [diff] [blame] | 326 | #if defined(__BIONIC__) | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 327 | LocalPropertyTestState pa; | 
| Greg Hackmann | cb215a7 | 2013-02-13 14:41:48 -0800 | [diff] [blame] | 328 | ASSERT_TRUE(pa.valid); | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 329 | char prop_value[PROP_NAME_MAX]; | 
|  | 330 |  | 
|  | 331 | ASSERT_EQ(0, __system_property_add("property", 8, "value1", 6)); | 
|  | 332 | ASSERT_EQ(0, __system_property_add("other_property", 14, "value2", 6)); | 
|  | 333 | ASSERT_EQ(0, __system_property_add("property_other", 14, "value3", 6)); | 
|  | 334 |  | 
|  | 335 | ASSERT_EQ(0, __system_property_find("property1")); | 
|  | 336 | ASSERT_EQ(0, __system_property_get("property1", prop_value)); | 
|  | 337 |  | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 338 | ASSERT_EQ(-1, __system_property_add("name", 4, "value", PROP_VALUE_MAX)); | 
|  | 339 | ASSERT_EQ(-1, __system_property_update(NULL, "value", PROP_VALUE_MAX)); | 
| Christopher Ferris | f04935c | 2013-12-20 18:43:21 -0800 | [diff] [blame] | 340 | #else // __BIONIC__ | 
|  | 341 | GTEST_LOG_(INFO) << "This test does nothing.\n"; | 
|  | 342 | #endif // __BIONIC__ | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 343 | } | 
|  | 344 |  | 
|  | 345 | TEST(properties, serial) { | 
| Christopher Ferris | f04935c | 2013-12-20 18:43:21 -0800 | [diff] [blame] | 346 | #if defined(__BIONIC__) | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 347 | LocalPropertyTestState pa; | 
| Greg Hackmann | cb215a7 | 2013-02-13 14:41:48 -0800 | [diff] [blame] | 348 | ASSERT_TRUE(pa.valid); | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 349 | const prop_info *pi; | 
|  | 350 | unsigned int serial; | 
|  | 351 |  | 
|  | 352 | ASSERT_EQ(0, __system_property_add("property", 8, "value1", 6)); | 
|  | 353 | ASSERT_NE((const prop_info *)NULL, pi = __system_property_find("property")); | 
|  | 354 | serial = __system_property_serial(pi); | 
|  | 355 | ASSERT_EQ(0, __system_property_update((prop_info *)pi, "value2", 6)); | 
|  | 356 | ASSERT_NE(serial, __system_property_serial(pi)); | 
| Christopher Ferris | f04935c | 2013-12-20 18:43:21 -0800 | [diff] [blame] | 357 | #else // __BIONIC__ | 
|  | 358 | GTEST_LOG_(INFO) << "This test does nothing.\n"; | 
|  | 359 | #endif // __BIONIC__ | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 360 | } | 
|  | 361 |  | 
|  | 362 | TEST(properties, wait) { | 
| Christopher Ferris | f04935c | 2013-12-20 18:43:21 -0800 | [diff] [blame] | 363 | #if defined(__BIONIC__) | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 364 | LocalPropertyTestState pa; | 
| Greg Hackmann | cb215a7 | 2013-02-13 14:41:48 -0800 | [diff] [blame] | 365 | ASSERT_TRUE(pa.valid); | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 366 | unsigned int serial; | 
|  | 367 | prop_info *pi; | 
|  | 368 | pthread_t t; | 
|  | 369 | int flag = 0; | 
|  | 370 |  | 
|  | 371 | ASSERT_EQ(0, __system_property_add("property", 8, "value1", 6)); | 
|  | 372 | serial = __system_property_wait_any(0); | 
| Dimitry Ivanov | 16b2a4d | 2017-01-24 20:43:29 +0000 | [diff] [blame] | 373 |  | 
|  | 374 | pi = const_cast<prop_info*>(__system_property_find("property")); | 
|  | 375 | ASSERT_TRUE(pi != nullptr); | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 376 | __system_property_update(pi, "value2", 6); | 
|  | 377 | serial = __system_property_wait_any(serial); | 
|  | 378 |  | 
| Dimitry Ivanov | 16b2a4d | 2017-01-24 20:43:29 +0000 | [diff] [blame] | 379 | ASSERT_EQ(0, pthread_create(&t, nullptr, PropertyWaitHelperFn, &flag)); | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 380 | ASSERT_EQ(flag, 0); | 
|  | 381 | serial = __system_property_wait_any(serial); | 
|  | 382 | ASSERT_EQ(flag, 1); | 
|  | 383 |  | 
|  | 384 | void* result; | 
|  | 385 | ASSERT_EQ(0, pthread_join(t, &result)); | 
| Christopher Ferris | f04935c | 2013-12-20 18:43:21 -0800 | [diff] [blame] | 386 | #else // __BIONIC__ | 
|  | 387 | GTEST_LOG_(INFO) << "This test does nothing.\n"; | 
|  | 388 | #endif // __BIONIC__ | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 389 | } | 
|  | 390 |  | 
|  | 391 | class KilledByFault { | 
|  | 392 | public: | 
|  | 393 | explicit KilledByFault() {}; | 
|  | 394 | bool operator()(int exit_status) const; | 
|  | 395 | }; | 
|  | 396 |  | 
|  | 397 | bool KilledByFault::operator()(int exit_status) const { | 
|  | 398 | return WIFSIGNALED(exit_status) && | 
|  | 399 | (WTERMSIG(exit_status) == SIGSEGV || | 
|  | 400 | WTERMSIG(exit_status) == SIGBUS || | 
|  | 401 | WTERMSIG(exit_status) == SIGABRT); | 
|  | 402 | } | 
|  | 403 |  | 
| Yabin Cui | 9df7040 | 2014-11-05 18:01:01 -0800 | [diff] [blame] | 404 | class properties_DeathTest : public BionicDeathTest {}; | 
|  | 405 |  | 
|  | 406 | TEST_F(properties_DeathTest, read_only) { | 
| Christopher Ferris | f04935c | 2013-12-20 18:43:21 -0800 | [diff] [blame] | 407 | #if defined(__BIONIC__) | 
| Elliott Hughes | e437519 | 2013-10-21 17:09:52 -0700 | [diff] [blame] | 408 |  | 
|  | 409 | // This test only makes sense if we're talking to the real system property service. | 
|  | 410 | struct stat sb; | 
| Dimitry Ivanov | 16b2a4d | 2017-01-24 20:43:29 +0000 | [diff] [blame] | 411 | ASSERT_FALSE(stat(PROP_FILENAME, &sb) == -1 && errno == ENOENT); | 
| Elliott Hughes | e437519 | 2013-10-21 17:09:52 -0700 | [diff] [blame] | 412 |  | 
|  | 413 | ASSERT_EXIT(__system_property_add("property", 8, "value", 5), KilledByFault(), ""); | 
| Christopher Ferris | f04935c | 2013-12-20 18:43:21 -0800 | [diff] [blame] | 414 | #else // __BIONIC__ | 
|  | 415 | GTEST_LOG_(INFO) << "This test does nothing.\n"; | 
|  | 416 | #endif // __BIONIC__ | 
| Colin Cross | b27e200 | 2013-01-28 17:19:43 -0800 | [diff] [blame] | 417 | } |