Files
renderdoc/renderdoc/os/os_specific.cpp
T
2023-02-01 12:23:32 +00:00

685 lines
17 KiB
C++

/******************************************************************************
* The MIT License (MIT)
*
* Copyright (c) 2019-2023 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 = "prepend";
mod.sep = EnvSep::SemiColon;
mod.mod = EnvMod::Prepend;
Process::RegisterEnvironmentModification(mod);
Process::ApplyEnvironmentModification();
var = Process::GetEnvVariable("__renderdoc__unit_test_var");
CHECK(var == rdcstr("prepend;test_value;test_value:test_value"));
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)