#include #include #include #include #include #include "hip_test_filesystem.hh" void TestContext::detectOS() { #if (HT_WIN == 1) p_windows = true; #elif (HT_LINUX == 1) p_linux = true; #endif } void TestContext::detectPlatform() { #if (HT_AMD == 1) amd = true; #elif (HT_NVIDIA == 1) nvidia = true; #endif } std::string TestContext::substringFound(std::vector list, std::string filename) { std::string match = ""; for (unsigned int i = 0; i < list.size(); i++) { if (filename.find(list.at(i)) != std::string::npos) { match = list.at(i); break; } } return match; } std::string TestContext::getMatchingConfigFile(std::string config_dir) { std::string configFileToUse; for (auto& p : fs::recursive_directory_iterator(config_dir)) { fs::path filename = p.path(); std::string cur_arch = "TODO"; std::string arch = substringFound(amd_arch_list_, filename.filename().string()); std::string platform = substringFound(platform_list_, filename.filename().string()); std::string os = substringFound(os_list_, filename.filename().string()); // if arch found then use that exit from loop if (arch == cur_arch) { configFileToUse = filename.string(); break; // match the platform/os and continue to look } else if ((platform == config_.platform) && (os == config_.os || os == "all")) { configFileToUse = filename.string(); } } return configFileToUse; } std::string& TestContext::getJsonFile() { fs::path config_dir = exe_path; config_dir = config_dir.parent_path(); int levels = 0; bool configFolderFound = false; std::vector configList; std::string configFile; // check a max of 5 levels down the executable path while (levels < 5) { fs::path temp_path = config_dir; temp_path /= "hipTestMain"; temp_path /= "config"; if (fs::exists(temp_path)) { config_dir = fs::absolute(temp_path); configFolderFound = true; break; } else { config_dir = config_dir.parent_path(); levels++; } } // get config.json files if config folder. if (configFolderFound) { json_file_ = getMatchingConfigFile(config_dir.string()); } return json_file_; } void TestContext::fillConfig() { config_.platform = (amd ? "amd" : (nvidia ? "nvidia" : "unknown")); config_.os = (p_windows ? "windows" : (p_linux ? "linux" : "unknown")); if (config_.os == "unknown" || config_.platform == "unknown") { LogPrintf("%s", "Either Config or Os is unknown, this wont end well"); abort(); } const char* env_config = std::getenv("HT_CONFIG_FILE"); LogPrintf("Env Config file: %s", (env_config != nullptr) ? env_config : "Not found, using default config"); // Check if path has been provided std::string def_config_json = "config.json"; std::string config_str; if (env_config != nullptr) { config_str = env_config; } else { config_str = def_config_json; } fs::path config_path = config_str; if (config_path.has_parent_path() && config_path.has_filename()) { config_.json_file = config_str; } else if (config_path.has_parent_path()) { config_.json_file = config_path.string() + def_config_json; } else { config_.json_file = getJsonFile(); } LogPrintf("Config file path: %s", config_.json_file.c_str()); } TestContext::TestContext(int argc, char** argv) { detectOS(); detectPlatform(); setExePath(argc, argv); fillConfig(); parseJsonFile(); parseOptions(argc, argv); } void TestContext::setExePath(int argc, char** argv) { if (argc == 0) return; fs::path p = std::string(argv[0]); if (p.has_filename()) p.remove_filename(); exe_path = p.string(); } bool TestContext::isWindows() const { return p_windows; } bool TestContext::isLinux() const { return p_linux; } bool TestContext::isNvidia() const { return nvidia; } bool TestContext::isAmd() const { return amd; } void TestContext::parseOptions(int argc, char** argv) { // Test name is at [1] position if (argc != 2) return; current_test = std::string(argv[1]); } bool TestContext::skipTest() const { // Direct Match auto flags = std::regex::ECMAScript; for (const auto& i : skip_test) { auto regex = std::regex(i.c_str(), flags); if (std::regex_match(current_test, regex)) { return true; } } // TODO add test case skip as well return false; } std::string TestContext::currentPath() const { return fs::current_path().string(); } bool TestContext::parseJsonFile() { // Check if file exists if (!fs::exists(config_.json_file)) { LogPrintf("Unable to find the file: %s", config_.json_file.c_str()); return true; } // Open the file std::ifstream js_file(config_.json_file); std::string json_str((std::istreambuf_iterator(js_file)), std::istreambuf_iterator()); LogPrintf("Json contents:: %s", json_str.data()); picojson::value v; std::string err = picojson::parse(v, json_str); if (err.size() > 1) { LogPrintf("Error from PicoJson: %s", err.data()); return false; } if (!v.is()) { LogPrintf("%s", "Data in json is not in correct format, it should be an object"); return false; } const picojson::object& o = v.get(); for (picojson::object::const_iterator i = o.begin(); i != o.end(); ++i) { // Processing for DisabledTests if (i->first == "DisabledTests") { // Value should contain list of values if (!i->second.is()) return false; auto& val = i->second.get(); for (auto ai = val.begin(); ai != val.end(); ai++) { std::string tmp = ai->get(); std::string newRegexName; for (const auto& c : tmp) { if (c == '*') newRegexName += ".*"; else newRegexName += c; } skip_test.insert(newRegexName); } } } return true; } void TestContext::cleanContext() { for (auto& pair : compiledKernels) { REQUIRE(hipSuccess == hipModuleUnload(pair.second.module)); } } void TestContext::trackRtcState(std::string kernelNameExpression, hipModule_t loadedModule, hipFunction_t kernelFunction) { rtcState state{loadedModule, kernelFunction}; compiledKernels[kernelNameExpression] = state; } hipFunction_t TestContext::getFunction(const std::string kernelNameExpression) { auto it{compiledKernels.find(kernelNameExpression)}; if (it != compiledKernels.end()) { return it->second.kernelFunction; } else { return nullptr; } }