1 /* 2 * Copyright (c) 2015, Intel Corporation 3 * All rights reserved. 4 * 5 * Redistribution and use in source and binary forms, with or without modification, 6 * are permitted provided that the following conditions are met: 7 * 8 * 1. Redistributions of source code must retain the above copyright notice, this 9 * list of conditions and the following disclaimer. 10 * 11 * 2. Redistributions in binary form must reproduce the above copyright notice, 12 * this list of conditions and the following disclaimer in the documentation and/or 13 * other materials provided with the distribution. 14 * 15 * 3. Neither the name of the copyright holder nor the names of its contributors 16 * may be used to endorse or promote products derived from this software without 17 * specific prior written permission. 18 * 19 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND 20 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED 21 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE 22 * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR 23 * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES 24 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; 25 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON 26 * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 27 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS 28 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 29 */ 30 31 #include "ParameterFramework.h" 32 33 #include "TmpFile.hpp" 34 35 #define CATCH_CONFIG_MAIN // This tells Catch to provide a main() 36 #include <catch.hpp> 37 38 #include <string> 39 #include <memory> 40 #include <vector> 41 #include <array> 42 43 #include <cstring> 44 #include <cerrno> 45 #include <climits> 46 47 struct Test 48 { 49 /** @return true if str is empty. */ emptyTest50 bool empty(const char *str) 51 { 52 REQUIRE(str != NULL); 53 return *str == '\0'; 54 } 55 REQUIRE_FAILURETest56 void REQUIRE_FAILURE(bool success) 57 { 58 THEN ("It should be an error") { 59 INFO("Previous pfw log: \n" + logLines); 60 CAPTURE(pfwGetLastError(pfw)); 61 CHECK(not success); 62 CHECK(not empty(pfwGetLastError(pfw))); 63 } 64 } 65 REQUIRE_SUCCESSTest66 void REQUIRE_SUCCESS(bool success) 67 { 68 THEN ("It should be a success") { 69 INFO("Previous pfw log: \n" + logLines); 70 CAPTURE(pfwGetLastError(pfw)); 71 CHECK(success); 72 CHECK(empty(pfwGetLastError(pfw))); 73 } 74 } 75 76 /** Wrap utility::TmpFile to add an implicit convertion to the temporary file. 77 * 78 * This avoids dozens of .getPath() in the following tests. */ 79 class TmpFile : private parameterFramework::utility::TmpFile 80 { 81 private: 82 using Base = parameterFramework::utility::TmpFile; 83 84 public: 85 using Base::TmpFile; 86 87 using Base::getPath; 88 /** Implicitly convert to the path of the temporary file. */ operator const char*() const89 operator const char *() const { return getPath().c_str(); } 90 }; 91 92 /** Log in logLines. */ logCbTest93 static void logCb(void *voidLogLines, PfwLogLevel level, const char *logLine) 94 { 95 std::string &logLines = *reinterpret_cast<std::string *>(voidLogLines); 96 switch (level) { 97 case pfwLogWarning: 98 logLines += "Warning: "; 99 break; 100 case pfwLogInfo: 101 logLines += "Info: "; 102 } 103 logLines += logLine; 104 logLines += '\n'; 105 } 106 107 /** Log buffer, will only be display in case of failure */ 108 std::string logLines; 109 110 /** Pfw handler used in the tests. */ 111 PfwHandler *pfw; 112 }; 113 114 TEST_CASE_METHOD(Test, "Parameter-framework c api use") 115 { 116 // Create criteria 117 const char *letterList[] = {"a", "b", "c", nullptr}; 118 const char *numberList[] = {"1", "2", "3", nullptr}; 119 const PfwCriterion criteria[] = { 120 {"inclusiveCrit", true, letterList}, {"exclusiveCrit", false, numberList}, 121 }; 122 size_t criterionNb = sizeof(criteria) / sizeof(criteria[0]); 123 PfwLogger logger = {&logLines, logCb}; 124 125 // Create valid pfw config file 126 const char *intParameterPath = "/test/system/integer"; 127 const char *stringParameterPath = "/test/system/string"; 128 TmpFile system("<?xml version='1.0' encoding='UTF-8'?>\ 129 <Subsystem Name='system' Type='Virtual'>\ 130 <ComponentLibrary/>\ 131 <InstanceDefinition>\ 132 <IntegerParameter Name='integer' Size='32' Signed='true' Max='100'/>\ 133 <StringParameter Name='string' MaxLength='9'/>\ 134 </InstanceDefinition>\ 135 </Subsystem>"); 136 TmpFile libraries("<?xml version='1.0' encoding='UTF-8'?>\ 137 <SystemClass Name='test'>\ 138 <SubsystemInclude Path='" + 139 system.getPath() + "'/>\ 140 </SystemClass>"); 141 TmpFile config("<?xml version='1.0' encoding='UTF-8'?>\ 142 <ParameterFrameworkConfiguration\ 143 SystemClassName='test' TuningAllowed='false'>\ 144 <SubsystemPlugins/>\ 145 <StructureDescriptionFileLocation Path='" + 146 libraries.getPath() + "'/>\ 147 </ParameterFrameworkConfiguration>"); 148 149 GIVEN ("A created parameter framework") { 150 pfw = pfwCreate(); 151 REQUIRE(pfw != NULL); 152 153 THEN ("Error message should be empty") { 154 CHECK(empty(pfwGetLastError(pfw))); 155 } 156 157 WHEN ("The pfw is started without an existent file") { 158 REQUIRE_FAILURE(pfwStart(pfw, "/doNotExist", criteria, criterionNb, &logger)); 159 } 160 161 WHEN ("The pfw is started with duplicated criterion value") { 162 const PfwCriterion duplicatedCriteria[] = { 163 {"duplicated name", true, letterList}, {"duplicated name", false, numberList}, 164 }; 165 REQUIRE_FAILURE(pfwStart(pfw, config, duplicatedCriteria, 2, &logger)); 166 } 167 WHEN ("The pfw is started with duplicated criterion value state") { 168 const char *values[] = {"a", "a", nullptr}; 169 const PfwCriterion duplicatedCriteria[] = {{"name", true, values}}; 170 171 WHEN ("Using test logger") { 172 REQUIRE_FAILURE(pfwStart(pfw, config, duplicatedCriteria, 1, &logger)); 173 } 174 WHEN ("Using default logger") { 175 // Test coverage of default logger warning 176 REQUIRE_FAILURE(pfwStart(pfw, config, duplicatedCriteria, 1, nullptr)); 177 } 178 } 179 WHEN ("The pfw is started with NULL name criterion") { 180 const PfwCriterion duplicatedCriteria[] = {{nullptr, true, letterList}}; 181 REQUIRE_FAILURE(pfwStart(pfw, config, duplicatedCriteria, 1, &logger)); 182 } 183 WHEN ("The pfw is started with NULL criterion state list") { 184 const PfwCriterion duplicatedCriteria[] = {{"name", true, nullptr}}; 185 REQUIRE_FAILURE(pfwStart(pfw, config, duplicatedCriteria, 1, &logger)); 186 } 187 GIVEN ("A criteria with lots of values") { 188 // Build a criterion with as many value as there is bits in int. 189 std::vector<char> names(sizeof(int) * CHAR_BIT + 1, 'a'); 190 names.back() = '\0'; 191 std::vector<const char *> values(names.size()); 192 for (size_t i = 0; i < values.size(); ++i) { 193 values[i] = &names[i]; 194 } 195 values.back() = nullptr; 196 /* The pfw c api requires criterion values to be a NULL terminated 197 * array of string. Each string is a pointer to a NULL terminated 198 * array of char. The pfw requires each string to be different 199 * from all others, ie strcmp(values[i], values[j]) != 0 for any 200 * i j. 201 * 202 * In order to generate easily an array of different strings, 203 * instantiate one string (names) big enough 204 * (@see PfwCriterion::values). 205 * Then instantiate an array of pointer (values), 206 * each pointing to a different position in the previously 207 * created string. 208 * 209 * Representation of the names and values vectors. 210 * 211 * n = number of bit in an int 212 * <--- n+1 elements ---> 213 * names = |a|a|a|a|...|a|a|a|\0| 214 * ^ ^ ^ 215 * values[0] = ´ | | 216 * values[1] = --´ | 217 * ... | 218 * values[n - 1] = -----------´ 219 * values[n] = NULL 220 * 221 */ 222 const PfwCriterion duplicatedCriteria[] = {{"name", true, &values[0]}}; 223 224 WHEN ("The pfw is started with a too long criterion state list") { 225 REQUIRE_FAILURE(pfwStart(pfw, config, duplicatedCriteria, 1, &logger)); 226 } 227 WHEN ("The pfw is started with max length criterion state list") { 228 values[values.size() - 2] = nullptr; // Hide last value 229 REQUIRE_SUCCESS(pfwStart(pfw, config, duplicatedCriteria, 1, &logger)); 230 } 231 } 232 233 WHEN ("The pfw is started with zero criteria") { 234 REQUIRE_SUCCESS(pfwStart(pfw, config, criteria, 0, &logger)); 235 } 236 237 WHEN ("The pfw is started twice a pfw") { 238 REQUIRE_SUCCESS(pfwStart(pfw, config, criteria, criterionNb, &logger)); 239 REQUIRE_FAILURE(pfwStart(pfw, config, criteria, criterionNb, &logger)); 240 } 241 242 WHEN ("The pfw is started without a logger callback") { 243 PfwLogger noLog = {nullptr, nullptr}; 244 REQUIRE_SUCCESS(pfwStart(pfw, config, criteria, criterionNb, &noLog)); 245 } 246 WHEN ("The pfw is started with default logger") { 247 REQUIRE_SUCCESS(pfwStart(pfw, config, criteria, criterionNb, nullptr)); 248 } 249 250 WHEN ("Get criterion of a stopped pfw") { 251 int value; 252 REQUIRE_FAILURE(pfwGetCriterion(pfw, criteria[0].name, &value)); 253 } 254 WHEN ("Set criterion of a stopped pfw") { 255 REQUIRE_FAILURE(pfwSetCriterion(pfw, criteria[0].name, 1)); 256 } 257 WHEN ("Commit criteria of a stopped pfw") { 258 REQUIRE_FAILURE(pfwApplyConfigurations(pfw)); 259 } 260 261 WHEN ("Bind parameter with a stopped pfw") { 262 REQUIRE(pfwBindParameter(pfw, intParameterPath) == NULL); 263 } 264 265 WHEN ("The pfw is started correctly") { 266 REQUIRE_SUCCESS(pfwStart(pfw, config, criteria, criterionNb, &logger)); 267 int value; 268 269 WHEN ("Get not existing criterion") { 270 REQUIRE_FAILURE(pfwGetCriterion(pfw, "Do not exist", &value)); 271 } 272 THEN ("All criterion should value 0") { 273 for (size_t i = 0; i < criterionNb; ++i) { 274 const char *criterionName = criteria[i].name; 275 CAPTURE(criterionName); 276 REQUIRE_SUCCESS(pfwGetCriterion(pfw, criterionName, &value)); 277 REQUIRE(value == 0); 278 } 279 } 280 281 WHEN ("Set not existing criterion") { 282 REQUIRE_FAILURE(pfwSetCriterion(pfw, "Do not exist", 3)); 283 } 284 WHEN ("Set criterion value") { 285 for (size_t i = 0; i < criterionNb; ++i) { 286 const char *criterionName = criteria[i].name; 287 CAPTURE(criterionName); 288 REQUIRE_SUCCESS(pfwSetCriterion(pfw, criterionName, 3)); 289 } 290 THEN ("Get criterion value should return what was set") { 291 for (size_t i = 0; i < criterionNb; ++i) { 292 const char *criterionName = criteria[i].name; 293 CAPTURE(criterionName); 294 REQUIRE_SUCCESS(pfwGetCriterion(pfw, criterionName, &value)); 295 REQUIRE(value == 3); 296 } 297 } 298 WHEN ("Set a new value to a criterion without committing first") { 299 const char *criterionName = criteria[0].name; 300 REQUIRE_SUCCESS(pfwSetCriterion(pfw, criterionName, 0)); 301 THEN ("A warning message should have been displayed") { 302 INFO("Previous pfw log: \n" + logLines); 303 size_t logPos = logLines.find("Warning: Selection criterion " 304 "'inclusiveCrit' has been modified 1 time(s)" 305 " without any configuration application"); 306 CHECK(logPos != std::string::npos); 307 } 308 } 309 } 310 WHEN ("Commit criteria of a started pfw") { 311 REQUIRE_SUCCESS(pfwApplyConfigurations(pfw)); 312 } 313 WHEN ("Bind a non existing parameter") { 314 REQUIRE_FAILURE(pfwBindParameter(pfw, "do/not/exist") != nullptr); 315 } 316 317 GIVEN ("An integer parameter handle") { 318 PfwParameterHandler *param = pfwBindParameter(pfw, intParameterPath); 319 REQUIRE_SUCCESS(param != nullptr); 320 321 WHEN ("Set parameter out of range") { 322 REQUIRE_FAILURE(pfwSetIntParameter(param, 101)); 323 } 324 325 WHEN ("Set parameter") { 326 REQUIRE_SUCCESS(pfwSetIntParameter(param, 11)); 327 THEN ("Get parameter should return what was set") { 328 REQUIRE_SUCCESS(pfwGetIntParameter(param, &value)); 329 REQUIRE(value == 11); 330 } 331 } 332 333 pfwUnbindParameter(param); 334 } 335 336 GIVEN ("An string parameter handle") { 337 PfwParameterHandler *param = pfwBindParameter(pfw, stringParameterPath); 338 REQUIRE_SUCCESS(param != nullptr); 339 340 WHEN ("Set parameter out of range") { 341 REQUIRE_FAILURE(pfwSetStringParameter(param, "ko_1234567")); 342 } 343 344 WHEN ("Set parameter") { 345 char *value; 346 REQUIRE_SUCCESS(pfwSetStringParameter(param, "ok")); 347 THEN ("Get parameter should return what was set") { 348 REQUIRE_SUCCESS(pfwGetStringParameter(param, &value)); 349 REQUIRE(value == std::string("ok")); 350 pfwFree(value); 351 } 352 } 353 354 pfwUnbindParameter(param); 355 } 356 } 357 358 pfwDestroy(pfw); 359 } 360 } 361