mirror of
https://github.com/baldurk/renderdoc.git
synced 2026-05-14 14:01:06 +00:00
685 lines
17 KiB
C++
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)
|