renderdoc/os/os_specific.cpp (495 lines of code) (raw):

/****************************************************************************** * The MIT License (MIT) * * Copyright (c) 2019-2024 Baldur Karlsson * Copyright (c) 2014 Crytek * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. ******************************************************************************/ #include "os/os_specific.h" #include "api/replay/control_types.h" #include "common/formatting.h" #include "strings/string_utils.h" int utf8printv(char *buf, size_t bufsize, const char *fmt, va_list args); int utf8printf_custom(char *buf, size_t bufSize, const char *fmt, StringFormat::Args &args); bool Network::ParseIPRangeCIDR(const rdcstr &str, uint32_t &ip, uint32_t &mask) { uint32_t a = 0, b = 0, c = 0, d = 0, num = 0; #if ENABLED(RDOC_WIN32) int ret = sscanf_s(str.c_str(), "%u.%u.%u.%u/%u", &a, &b, &c, &d, &num); #else int ret = sscanf(str.c_str(), "%u.%u.%u.%u/%u", &a, &b, &c, &d, &num); #endif if(ret != 5 || a > 255 || b > 255 || c > 255 || d > 255 || num > 32) { ip = 0; mask = 0; return false; } ip = MakeIP(a, b, c, d); if(num == 0) { mask = 0; } else { num = 32 - num; mask = ((~0U) >> num) << num; } return true; } namespace StringFormat { rdcstr Fmt(const char *format, ...) { va_list args; va_start(args, format); va_list args2; va_copy(args2, args); int size = ::utf8printv(NULL, 0, format, args2); rdcstr ret; ret.resize(size); ::utf8printv(ret.data(), size + 1, format, args); va_end(args); va_end(args2); return ret; } rdcstr Fmt(rdcliteral format, ...) { // optimisation - if there are no format specifiers hence no arguments, preserve and return the // literal if(strchr(format.c_str(), '%') == NULL) return format; va_list args; va_start(args, format); va_list args2; va_copy(args2, args); int size = ::utf8printv(NULL, 0, format.c_str(), args2); rdcstr ret; ret.resize(size); ::utf8printv(ret.data(), size + 1, format.c_str(), args); va_end(args); va_end(args2); return ret; } rdcstr FmtArgs(const char *format, Args &args) { int size = ::utf8printf_custom(NULL, 0, format, args); args.reset(); rdcstr ret; ret.resize(size); ::utf8printf_custom(ret.data(), size + 1, format, args); return ret; } }; // namespace StringFormat rdcstr Callstack::AddressDetails::formattedString(const rdcstr &commonPath) { const char *f = filename.c_str(); if(!commonPath.empty()) { rdcstr common = strlower(commonPath); rdcstr fn = strlower(filename.substr(0, common.length())); if(common == fn) { f += common.length(); } } if(line > 0) return StringFormat::Fmt("%s line %d", function.c_str(), line); else return function; } rdcstr OSUtility::MakeMachineIdentString(uint64_t ident) { rdcstr ret = ""; if(ident & MachineIdent_Windows) ret += "Windows "; else if(ident & MachineIdent_Linux) ret += "Linux "; else if(ident & MachineIdent_macOS) ret += "macOS "; else if(ident & MachineIdent_Android) ret += "Android "; else if(ident & MachineIdent_iOS) ret += "iOS "; if(ident & MachineIdent_Arch_x86) ret += "x86 "; else if(ident & MachineIdent_Arch_ARM) ret += "ARM "; if(ident & MachineIdent_32bit) ret += "32-bit "; else if(ident & MachineIdent_64bit) ret += "64-bit "; switch(ident & MachineIdent_GPU_Mask) { case MachineIdent_GPU_ARM: ret += "ARM GPU "; break; case MachineIdent_GPU_AMD: ret += "AMD GPU "; break; case MachineIdent_GPU_IMG: ret += "Imagination GPU "; break; case MachineIdent_GPU_Intel: ret += "Intel GPU "; break; case MachineIdent_GPU_NV: ret += "nVidia GPU "; break; case MachineIdent_GPU_QUALCOMM: ret += "QUALCOMM GPU "; break; case MachineIdent_GPU_Samsung: ret += "Samsung GPU "; break; case MachineIdent_GPU_Verisilicon: ret += "Verisilicon GPU "; break; default: break; } return ret; } #if ENABLED(ENABLE_UNIT_TESTS) #include "catch/catch.hpp" TEST_CASE("Test OS-specific functions", "[osspecific]") { SECTION("GetLibraryFilename") { rdcstr libPath; FileIO::GetLibraryFilename(libPath); CHECK_FALSE(libPath.empty()); } SECTION("Environment Variables") { rdcstr var = Process::GetEnvVariable("TMP"); var = Process::GetEnvVariable("TEMP"); var = Process::GetEnvVariable("HOME"); CHECK(!var.empty()); CHECK(var.length() > 1); var = Process::GetEnvVariable("__renderdoc__unit_test_var"); CHECK(var.empty()); EnvironmentModification mod; mod.name = "__renderdoc__unit_test_var"; mod.value = "test_value"; mod.sep = EnvSep::SemiColon; mod.mod = EnvMod::Append; Process::RegisterEnvironmentModification(mod); Process::ApplyEnvironmentModification(); var = Process::GetEnvVariable("__renderdoc__unit_test_var"); CHECK(var == "test_value"); Process::RegisterEnvironmentModification(mod); Process::ApplyEnvironmentModification(); var = Process::GetEnvVariable("__renderdoc__unit_test_var"); CHECK(var == rdcstr("test_value;test_value")); mod.sep = EnvSep::Colon; Process::RegisterEnvironmentModification(mod); Process::ApplyEnvironmentModification(); var = Process::GetEnvVariable("__renderdoc__unit_test_var"); CHECK(var == rdcstr("test_value;test_value:test_value")); mod.value = "prepend1"; mod.sep = EnvSep::SemiColon; mod.mod = EnvMod::Prepend; Process::RegisterEnvironmentModification(mod); mod.value = "prepend2"; mod.sep = EnvSep::SemiColon; mod.mod = EnvMod::Append; Process::RegisterEnvironmentModification(mod); Process::ApplyEnvironmentModification(); var = Process::GetEnvVariable("__renderdoc__unit_test_var"); CHECK(var == rdcstr("prepend1;test_value;test_value:test_value;prepend2")); mod.value = "reset"; mod.sep = EnvSep::SemiColon; mod.mod = EnvMod::Set; Process::RegisterEnvironmentModification(mod); Process::ApplyEnvironmentModification(); var = Process::GetEnvVariable("__renderdoc__unit_test_var"); CHECK(var == rdcstr("reset")); }; SECTION("UTF-8 to wide conversion") { const rdcstr s = "ελληνικά"; rdcwstr ws = StringFormat::UTF82Wide(s); CHECK(ws.length() == 8); CHECK(ws[0] == L'\x3b5'); CHECK(ws[ws.length() - 1] == L'\x3ac'); CHECK(ws[ws.length()] == L'\0'); rdcstr s2 = StringFormat::Wide2UTF8(ws); CHECK(s == s2); CHECK(StringFormat::UTF82Wide(rdcstr()).length() == 0); CHECK(StringFormat::UTF82Wide(rdcstr()).c_str()[0] == 0); }; SECTION("Timing") { double freq = Timing::GetTickFrequency(); REQUIRE(freq > 0.0); { uint64_t startTick = Timing::GetTick(); CHECK(startTick > 0); uint64_t firstTick = Timing::GetTick(); Threading::Sleep(1500); uint64_t lastTick = Timing::GetTick(); double milliseconds1 = double(firstTick - startTick) / freq; double milliseconds2 = double(lastTick - firstTick) / freq; CHECK(milliseconds1 > 0.0); CHECK(milliseconds1 < 1.0); CHECK(milliseconds2 > 1480.0); CHECK(milliseconds2 < 1675.0); } // timestamp as of the creation of this test CHECK(Timing::GetUnixTimestamp() > 1504519614); }; SECTION("Bit counting") { SECTION("32-bits") { uint32_t val = 0; { INFO("val is " << val); CHECK(Bits::CountLeadingZeroes(val) == 32); CHECK(Bits::CountTrailingZeroes(val) == 32); CHECK(Bits::CountOnes(val) == 0); } val = 1; { INFO("val is " << val); CHECK(Bits::CountLeadingZeroes(val) == 31); CHECK(Bits::CountTrailingZeroes(val) == 0); CHECK(Bits::CountOnes(val) == 1); } val <<= 1; { INFO("val is " << val); CHECK(Bits::CountLeadingZeroes(val) == 30); CHECK(Bits::CountTrailingZeroes(val) == 1); CHECK(Bits::CountOnes(val) == 1); } val <<= 4; { INFO("val is " << val); CHECK(Bits::CountLeadingZeroes(val) == 26); CHECK(Bits::CountTrailingZeroes(val) == 5); CHECK(Bits::CountOnes(val) == 1); } val++; { INFO("val is " << val); CHECK(Bits::CountLeadingZeroes(val) == 26); CHECK(Bits::CountTrailingZeroes(val) == 0); CHECK(Bits::CountOnes(val) == 2); } val += 5; { INFO("val is " << val); CHECK(Bits::CountLeadingZeroes(val) == 26); CHECK(Bits::CountTrailingZeroes(val) == 1); CHECK(Bits::CountOnes(val) == 3); } val += 1000; { INFO("val is " << val); CHECK(Bits::CountLeadingZeroes(val) == 21); CHECK(Bits::CountTrailingZeroes(val) == 1); CHECK(Bits::CountOnes(val) == 4); } val *= 3; { INFO("val is " << val); CHECK(Bits::CountLeadingZeroes(val) == 20); CHECK(Bits::CountTrailingZeroes(val) == 1); CHECK(Bits::CountOnes(val) == 5); } val *= 200000; { INFO("val is " << val); CHECK(Bits::CountLeadingZeroes(val) == 2); CHECK(Bits::CountTrailingZeroes(val) == 7); CHECK(Bits::CountOnes(val) == 12); } val |= 0xFFFFFFFFu; { INFO("val is " << val); CHECK(Bits::CountLeadingZeroes(val) == 0); CHECK(Bits::CountTrailingZeroes(val) == 0); CHECK(Bits::CountOnes(val) == 32); } }; #if ENABLED(RDOC_X64) SECTION("64-bits") { uint64_t val = 0; { INFO("val is " << val); CHECK(Bits::CountLeadingZeroes(val) == 64); CHECK(Bits::CountTrailingZeroes(val) == 64); CHECK(Bits::CountOnes(val) == 0); } val = 1; { INFO("val is " << val); CHECK(Bits::CountLeadingZeroes(val) == 63); CHECK(Bits::CountTrailingZeroes(val) == 0); CHECK(Bits::CountOnes(val) == 1); } val <<= 1; { INFO("val is " << val); CHECK(Bits::CountLeadingZeroes(val) == 62); CHECK(Bits::CountTrailingZeroes(val) == 1); CHECK(Bits::CountOnes(val) == 1); } val <<= 4; { INFO("val is " << val); CHECK(Bits::CountLeadingZeroes(val) == 58); CHECK(Bits::CountTrailingZeroes(val) == 5); CHECK(Bits::CountOnes(val) == 1); } val++; { INFO("val is " << val); CHECK(Bits::CountLeadingZeroes(val) == 58); CHECK(Bits::CountTrailingZeroes(val) == 0); CHECK(Bits::CountOnes(val) == 2); } val += 5; { INFO("val is " << val); CHECK(Bits::CountLeadingZeroes(val) == 58); CHECK(Bits::CountTrailingZeroes(val) == 1); CHECK(Bits::CountOnes(val) == 3); } val += 1000; { INFO("val is " << val); CHECK(Bits::CountLeadingZeroes(val) == 53); CHECK(Bits::CountTrailingZeroes(val) == 1); CHECK(Bits::CountOnes(val) == 4); } val *= 3; { INFO("val is " << val); CHECK(Bits::CountLeadingZeroes(val) == 52); CHECK(Bits::CountTrailingZeroes(val) == 1); CHECK(Bits::CountOnes(val) == 5); } val *= 200000; { INFO("val is " << val); CHECK(Bits::CountLeadingZeroes(val) == 34); CHECK(Bits::CountTrailingZeroes(val) == 7); CHECK(Bits::CountOnes(val) == 12); } val *= 1000000; { INFO("val is " << val); CHECK(Bits::CountLeadingZeroes(val) == 14); CHECK(Bits::CountTrailingZeroes(val) == 13); CHECK(Bits::CountOnes(val) == 19); } val |= 0xFFFFFFFFFFFFFFFFull; { INFO("val is " << val); CHECK(Bits::CountLeadingZeroes(val) == 0); CHECK(Bits::CountTrailingZeroes(val) == 0); CHECK(Bits::CountOnes(val) == 64); } }; #endif }; const int numThreads = 8; const int numValues = 10; const int totalCount = numThreads * numValues; SECTION("Simple threads") { uint64_t value = Threading::GetCurrentID(); CHECK(value != 0); // check that a simple thread will run Threading::ThreadHandle th = Threading::CreateThread([&value]() { value = Threading::GetCurrentID(); }); Threading::JoinThread(th); Threading::CloseThread(th); CHECK(value != 0); CHECK(value != Threading::GetCurrentID()); int values[totalCount] = {0}; for(int i = 0; i < totalCount; i++) CHECK(values[i] == 0); // launch multiple threads, each setting a subset of the values. Ensure they don't trample or // write the wrong values Threading::ThreadHandle threads[numThreads]; for(int threadID = 0; threadID < numThreads; threadID++) { threads[threadID] = Threading::CreateThread([&values, numValues, threadID]() { for(int i = 0; i < numValues; i++) values[threadID * numValues + i] = threadID * 1000 + i; }); } for(int threadID = 0; threadID < numThreads; threadID++) { Threading::JoinThread(threads[threadID]); Threading::CloseThread(threads[threadID]); } for(int i = 0; i < totalCount; i++) { CHECK((values[i] / 1000) == (i / numValues)); CHECK((values[i] % 1000) == (i % numValues)); } }; SECTION("Atomics") { int32_t value = 0; // check that thread atomics work on multiple overlapping threads Threading::ThreadHandle threads[numThreads]; for(int threadID = 0; threadID < numThreads; threadID++) { threads[threadID] = Threading::CreateThread([&value, numValues]() { for(int i = 0; i < numValues; i++) Atomic::Inc32(&value); }); } for(int threadID = 0; threadID < numThreads; threadID++) { Threading::JoinThread(threads[threadID]); Threading::CloseThread(threads[threadID]); } // each thread incremented numValues times CHECK(value == numValues * numThreads); Atomic::Dec32(&value); CHECK(value == numValues * numThreads - 1); }; SECTION("Locks") { // check that holding the lock prevents a thread from modifying the value uint64_t value = 0; Threading::CriticalSection lock; lock.Lock(); Threading::ThreadHandle th = Threading::CreateThread([&value, &lock]() { lock.Lock(); value = Threading::GetCurrentID(); lock.Unlock(); }); CHECK(value == 0); Threading::Sleep(50); CHECK(value == 0); // allow the thread to run lock.Unlock(); Threading::JoinThread(th); Threading::CloseThread(th); CHECK(value != 0); // check that we can acquire the lock now bool locked = lock.Trylock(); CHECK(locked); if(locked) lock.Unlock(); }; SECTION("IP processing") { CHECK(Network::MakeIP(127, 0, 0, 1) == 0x7f000001); CHECK(Network::MakeIP(216, 58, 211, 174) == 0xD83AD3AE); CHECK(Network::GetIPOctet(Network::MakeIP(216, 58, 211, 174), 0) == 216); CHECK(Network::GetIPOctet(Network::MakeIP(216, 58, 211, 174), 1) == 58); CHECK(Network::GetIPOctet(Network::MakeIP(216, 58, 211, 174), 2) == 211); CHECK(Network::GetIPOctet(Network::MakeIP(216, 58, 211, 174), 3) == 174); CHECK(Network::MatchIPMask(Network::MakeIP(127, 0, 0, 1), 0x7f000001, 0xFFFFFFFF)); CHECK(Network::MatchIPMask(Network::MakeIP(127, 0, 0, 1), 0x7f000000, 0xFF000000)); CHECK(Network::MatchIPMask(Network::MakeIP(127, 8, 0, 1), 0x7f000000, 0xFF000000)); CHECK(Network::MatchIPMask(Network::MakeIP(127, 100, 22, 5), 0x7f000000, 0xFF000000)); CHECK(Network::MatchIPMask(Network::MakeIP(127, 66, 66, 66), 0x7f000000, 0xFF000000)); CHECK_FALSE(Network::MatchIPMask(Network::MakeIP(216, 58, 211, 174), 0x80000000, ~0U)); uint32_t ip = 0; uint32_t mask = 0; Network::ParseIPRangeCIDR("foobar", ip, mask); CHECK(ip == 0); CHECK(mask == 0); Network::ParseIPRangeCIDR("", ip, mask); CHECK(ip == 0); CHECK(mask == 0); Network::ParseIPRangeCIDR("1.23/4", ip, mask); CHECK(ip == 0); CHECK(mask == 0); Network::ParseIPRangeCIDR("1.23.4.5.6.7/8", ip, mask); CHECK(ip == 0); CHECK(mask == 0); Network::ParseIPRangeCIDR("999.888.777.666/555", ip, mask); CHECK(ip == 0); CHECK(mask == 0); Network::ParseIPRangeCIDR("216.58,211.174/16", ip, mask); CHECK(ip == 0); CHECK(mask == 0); Network::ParseIPRangeCIDR("216.58.211.174/16", ip, mask); CHECK(ip == Network::MakeIP(216, 58, 211, 174)); CHECK(mask == 0xFFFF0000); Network::ParseIPRangeCIDR("216.58.211.174/8", ip, mask); CHECK(ip == Network::MakeIP(216, 58, 211, 174)); CHECK(mask == 0xFF000000); Network::ParseIPRangeCIDR("216.58.211.174/31", ip, mask); CHECK(ip == Network::MakeIP(216, 58, 211, 174)); CHECK(mask == 0xFFFFFFFe); }; }; #endif // ENABLED(ENABLE_UNIT_TESTS)