/* dt_check.cpp -- doctest check This file is part of the UPX executable compressor. Copyright (C) 1996-2023 Markus Franz Xaver Johannes Oberhumer All Rights Reserved. UPX and the UCL library are free software; you can redistribute them and/or modify them under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; see the file COPYING. If not, write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. Markus F.X.J. Oberhumer */ #include "../conf.h" /************************************************************************* // upx_doctest_check() // // honors environment variables: // UPX_DEBUG_DOCTEST_DISABLE // UPX_DEBUG_DOCTEST_VERBOSE // // HINT: set "UPX_DEBUG_DOCTEST_DISABLE=1" for improved debugging experience **************************************************************************/ int upx_doctest_check(int argc, char **argv) { #if defined(DOCTEST_CONFIG_DISABLE) UNUSED(argc); UNUSED(argv); return 0; #else const char *e = getenv("UPX_DEBUG_DOCTEST_DISABLE"); if (!e) e = getenv("UPX_DEBUG_DISABLE_DOCTEST"); // allow alternate spelling if (e && e[0] && strcmp(e, "0") != 0) return 0; bool minimal = true; // don't show summary bool duration = false; // don't show timings bool success = false; // don't show all succeeding tests #if DEBUG // default for debug builds: do show the [doctest] summary minimal = false; #endif e = getenv("UPX_DEBUG_DOCTEST_VERBOSE"); if (e && e[0]) { if (strcmp(e, "0") == 0) { minimal = true; } else if (strcmp(e, "1") == 0) { minimal = false; } else if (strcmp(e, "2") == 0) { minimal = false; duration = true; } else if (strcmp(e, "3") == 0) { minimal = false; duration = true; success = true; } } doctest::Context context; if (minimal) context.setOption("dt-minimal", true); if (duration) context.setOption("dt-duration", true); if (success) context.setOption("dt-success", true); // this requires that main_get_options() understands "--dt-XXX" options if (argc > 0 && argv != nullptr) context.applyCommandLine(argc, argv); int r = context.run(); if (r != 0) return 1; if (context.shouldExit()) return 2; return 0; #endif // DOCTEST_CONFIG_DISABLE } int upx_doctest_check() { return upx_doctest_check(0, nullptr); } /************************************************************************* // compile-time checks **************************************************************************/ // need extra parenthesis because the C preprocessor does not understand C++ templates ACC_COMPILE_TIME_ASSERT_HEADER((std::is_same::value)) ACC_COMPILE_TIME_ASSERT_HEADER((std::is_same::value)) ACC_COMPILE_TIME_ASSERT_HEADER((std::is_same::value)) ACC_COMPILE_TIME_ASSERT_HEADER((std::is_same::value)) ACC_COMPILE_TIME_ASSERT_HEADER((std::is_same::value)) ACC_COMPILE_TIME_ASSERT_HEADER((std::is_same::value)) ACC_COMPILE_TIME_ASSERT_HEADER(no_bswap16(0x04030201) == 0x0201) ACC_COMPILE_TIME_ASSERT_HEADER(no_bswap32(0x04030201) == 0x04030201) ACC_COMPILE_TIME_ASSERT_HEADER(no_bswap64(0x0807060504030201ull) == 0x0807060504030201ull) #if !(ACC_CC_MSC) // unfortunately *not* constexpr with current MSVC ACC_COMPILE_TIME_ASSERT_HEADER(bswap16(0x04030201) == 0x0102) ACC_COMPILE_TIME_ASSERT_HEADER(bswap32(0x04030201) == 0x01020304) ACC_COMPILE_TIME_ASSERT_HEADER(bswap64(0x0807060504030201ull) == 0x0102030405060708ull) ACC_COMPILE_TIME_ASSERT_HEADER(bswap16(bswap16(0xf4f3f2f1)) == no_bswap16(0xf4f3f2f1)) ACC_COMPILE_TIME_ASSERT_HEADER(bswap32(bswap32(0xf4f3f2f1)) == no_bswap32(0xf4f3f2f1)) ACC_COMPILE_TIME_ASSERT_HEADER(bswap64(bswap64(0xf8f7f6f5f4f3f2f1ull)) == no_bswap64(0xf8f7f6f5f4f3f2f1ull)) #endif ACC_COMPILE_TIME_ASSERT_HEADER(sign_extend(0u + 0, 8) == 0) ACC_COMPILE_TIME_ASSERT_HEADER(sign_extend(0u + 1, 8) == 1) ACC_COMPILE_TIME_ASSERT_HEADER(sign_extend(0u + 127, 8) == 127) ACC_COMPILE_TIME_ASSERT_HEADER(sign_extend(0u + 128, 8) == -128) ACC_COMPILE_TIME_ASSERT_HEADER(sign_extend(0u - 1, 8) == -1) ACC_COMPILE_TIME_ASSERT_HEADER(sign_extend(0u + 256, 8) == 0) ACC_COMPILE_TIME_ASSERT_HEADER(sign_extend(0u + 257, 8) == 1) ACC_COMPILE_TIME_ASSERT_HEADER(sign_extend(0u + 383, 8) == 127) ACC_COMPILE_TIME_ASSERT_HEADER(sign_extend(0u + 384, 8) == -128) ACC_COMPILE_TIME_ASSERT_HEADER(sign_extend(0u + 511, 8) == -1) ACC_COMPILE_TIME_ASSERT_HEADER(CHAR_BIT == 8) #if 0 // does not work with MSVC #if '\0' - 1 < 0 ACC_COMPILE_TIME_ASSERT_HEADER(CHAR_MAX == 127) #else ACC_COMPILE_TIME_ASSERT_HEADER(CHAR_MAX == 255) #endif #if L'\0' - 1 < 0 ACC_COMPILE_TIME_ASSERT_HEADER((wchar_t) -1 < 0) #else ACC_COMPILE_TIME_ASSERT_HEADER((wchar_t) -1 > 0) #endif #endif /************************************************************************* // upx_compiler_sanity_check() // assert a sane architecture and compiler // (modern compilers will optimize away most of this code) **************************************************************************/ namespace { template struct CheckIntegral { struct TestT { T a; T x[2]; }; template struct TestU { U a = {}; const U b = {}; static constexpr U c = {}; U x[2] = {}; const U y[2] = {}; static constexpr U z[2] = {}; }; template static void checkU(void) noexcept { { U a = {}; const U b = {}; constexpr U c = {}; U x[2] = {}; const U y[2] = {}; constexpr U z[2] = {}; assert_noexcept(a == 0); assert_noexcept(b == 0); assert_noexcept(c == 0); assert_noexcept(x[0] == 0 && x[1] == 0); assert_noexcept(y[0] == 0 && y[1] == 0); assert_noexcept(z[0] == 0 && z[1] == 0); } { TestU t; assert_noexcept(t.a == 0); assert_noexcept(t.b == 0); assert_noexcept(t.c == 0); assert_noexcept(t.x[0] == 0 && t.x[1] == 0); assert_noexcept(t.y[0] == 0 && t.y[1] == 0); assert_noexcept(t.z[0] == 0 && t.z[1] == 0); } #if __cplusplus < 202002L COMPILE_TIME_ASSERT(std::is_pod::value) // std::is_pod is deprecated in C++20 #endif COMPILE_TIME_ASSERT(std::is_standard_layout::value) COMPILE_TIME_ASSERT(std::is_trivial::value) // more checks, these are probably implied by std::is_trivial COMPILE_TIME_ASSERT(std::is_nothrow_default_constructible::value) COMPILE_TIME_ASSERT(std::is_nothrow_destructible::value) COMPILE_TIME_ASSERT(std::is_trivially_copyable::value) COMPILE_TIME_ASSERT(std::is_trivially_default_constructible::value) // UPX extras COMPILE_TIME_ASSERT(upx_is_integral::value) COMPILE_TIME_ASSERT(upx_is_integral_v) } static void check(void) noexcept { { TestT t = {}; assert_noexcept(t.a == 0); assert_noexcept(t.x[0] == 0 && t.x[1] == 0); } { const TestT t = {}; assert_noexcept(t.a == 0); assert_noexcept(t.x[0] == 0 && t.x[1] == 0); } { constexpr TestT t = {}; assert_noexcept(t.a == 0); assert_noexcept(t.x[0] == 0 && t.x[1] == 0); } { TestT t; mem_clear(&t); assert_noexcept(t.a == 0); assert_noexcept(t.x[0] == 0 && t.x[1] == 0); } checkU(); checkU::type>(); } }; template struct CheckAlignment { static void check(void) noexcept { COMPILE_TIME_ASSERT_ALIGNED1(T) struct alignas(1) Test1 { char a; T b; }; struct alignas(1) Test2 { char a; T b[3]; }; COMPILE_TIME_ASSERT_ALIGNED1(Test1) COMPILE_TIME_ASSERT_ALIGNED1(Test2) Test1 t1[7]; Test2 t2[7]; COMPILE_TIME_ASSERT(sizeof(Test1) == 1 + sizeof(T)) COMPILE_TIME_ASSERT(sizeof(t1) == 7 + 7 * sizeof(T)) COMPILE_TIME_ASSERT(sizeof(Test2) == 1 + 3 * sizeof(T)) COMPILE_TIME_ASSERT(sizeof(t2) == 7 + 21 * sizeof(T)) UNUSED(t1); UNUSED(t2); } }; template struct TestBELE { static noinline bool test(void) noexcept { CheckIntegral::check(); CheckAlignment::check(); // arithmetic checks T allbits = {}; assert_noexcept(allbits == 0); allbits += 1; allbits -= 2; T v1; v1 = 1; v1 *= 4; v1 /= 2; v1 -= 1; T v2; v2 = 1; assert_noexcept((v1 == v2)); assert_noexcept(!(v1 != v2)); assert_noexcept((v1 <= v2)); assert_noexcept((v1 >= v2)); assert_noexcept(!(v1 < v2)); assert_noexcept(!(v1 > v2)); v2 ^= allbits; assert_noexcept(!(v1 == v2)); assert_noexcept((v1 != v2)); assert_noexcept((v1 <= v2)); assert_noexcept(!(v1 >= v2)); assert_noexcept((v1 < v2)); assert_noexcept(!(v1 > v2)); v2 += 2; assert_noexcept(v1 == 1); assert_noexcept(v2 == 0); v1 <<= 1; v1 |= v2; v1 >>= 1; v2 &= v1; v2 /= v1; v2 *= v1; v1 += v2; v1 -= v2; assert_noexcept(v1 == 1); assert_noexcept(v2 == 0); if ((v1 ^ v2) != 1) return false; return true; } }; template struct TestNoAliasingStruct { static noinline bool test(A *a, B *b) noexcept { *a = 0; *b = 0; *b -= 3; return *a != 0; } }; template static forceinline bool testNoAliasing(A *a, B *b) noexcept { return TestNoAliasingStruct::test(a, b); } template struct TestIntegerWrap { static inline bool inc_gt(const T x) noexcept { return x + 1 > x; } static inline bool dec_lt(const T x) noexcept { return x - 1 < x; } static inline bool neg_eq(const T x) noexcept { return T(0) - x == x; } }; static noinline void throwSomeValue(int x) may_throw { if (x < 0) throw int(x); else throw size_t(x); } static noinline void check_basic_cxx_exception_handling(void (*func)(int)) noexcept { bool cxx_exception_handling_works = false; try { func(42); } catch (const size_t &e) { if (e == 42) cxx_exception_handling_works = true; } catch (...) { } assert_noexcept(cxx_exception_handling_works); } } // namespace #define ACC_WANT_ACC_CHK_CH 1 #undef ACCCHK_ASSERT #include "../miniacc.h" void upx_compiler_sanity_check(void) noexcept { const char *e = getenv("UPX_DEBUG_DOCTEST_DISABLE"); if (!e) e = getenv("UPX_DEBUG_DISABLE_DOCTEST"); // allow alternate spelling if (e && e[0] && strcmp(e, "0") != 0) { // If UPX_DEBUG_DOCTEST_DISABLE is set then we don't want to throw any // exceptions in order to improve debugging experience. } else { // check working C++ exception handling to catch toolchain/qemu/wine/etc problems check_basic_cxx_exception_handling(throwSomeValue); } #define ACC_WANT_ACC_CHK_CH 1 #undef ACCCHK_ASSERT #define ACCCHK_ASSERT(expr) ACC_COMPILE_TIME_ASSERT(expr) #include "../miniacc.h" #undef ACCCHK_ASSERT COMPILE_TIME_ASSERT(sizeof(char) == 1) COMPILE_TIME_ASSERT(sizeof(short) == 2) COMPILE_TIME_ASSERT(sizeof(int) == 4) COMPILE_TIME_ASSERT(sizeof(long) >= 4) COMPILE_TIME_ASSERT(sizeof(void *) >= 4) COMPILE_TIME_ASSERT(sizeof(BE16) == 2) COMPILE_TIME_ASSERT(sizeof(BE32) == 4) COMPILE_TIME_ASSERT(sizeof(BE64) == 8) COMPILE_TIME_ASSERT(sizeof(LE16) == 2) COMPILE_TIME_ASSERT(sizeof(LE32) == 4) COMPILE_TIME_ASSERT(sizeof(LE64) == 8) COMPILE_TIME_ASSERT_ALIGNED1(BE16) COMPILE_TIME_ASSERT_ALIGNED1(BE32) COMPILE_TIME_ASSERT_ALIGNED1(BE64) COMPILE_TIME_ASSERT_ALIGNED1(LE16) COMPILE_TIME_ASSERT_ALIGNED1(LE32) COMPILE_TIME_ASSERT_ALIGNED1(LE64) CheckIntegral::check(); CheckIntegral::check(); CheckIntegral::check(); CheckIntegral::check(); CheckIntegral::check(); CheckIntegral::check(); CheckIntegral::check(); CheckIntegral::check(); CheckIntegral::check(); CheckIntegral::check(); CheckIntegral::check(); COMPILE_TIME_ASSERT(sizeof(upx_charptr_unit_type) == 1) COMPILE_TIME_ASSERT_ALIGNED1(upx_charptr_unit_type) COMPILE_TIME_ASSERT(sizeof(*((charptr) nullptr)) == 1) COMPILE_TIME_ASSERT(sizeof(UPX_VERSION_STRING4) == 4 + 1) assert_noexcept(strlen(UPX_VERSION_STRING4) == 4); COMPILE_TIME_ASSERT(sizeof(UPX_VERSION_YEAR) == 4 + 1) assert_noexcept(strlen(UPX_VERSION_YEAR) == 4); assert_noexcept(memcmp(UPX_VERSION_DATE_ISO, UPX_VERSION_YEAR, 4) == 0); assert_noexcept( memcmp(&UPX_VERSION_DATE[sizeof(UPX_VERSION_DATE) - 1 - 4], UPX_VERSION_YEAR, 4) == 0); if (gitrev[0]) { size_t revlen = strlen(gitrev); if (strncmp(gitrev, "ERROR", 5) == 0) { assert_noexcept(revlen == 5 || revlen == 6); } else { assert_noexcept(revlen == 12 || revlen == 13); } if (revlen == 6 || revlen == 13) { assert_noexcept(gitrev[revlen - 1] == '+'); } } assert_noexcept(UPX_RSIZE_MAX_MEM == 805306368); #if DEBUG || 1 assert_noexcept(TestBELE::test()); assert_noexcept(TestBELE::test()); assert_noexcept(TestBELE::test()); assert_noexcept(TestBELE::test()); assert_noexcept(TestBELE::test()); assert_noexcept(TestBELE::test()); { alignas(16) static const byte dd[32] = { 0, 0, 0, 0, 0, 0, 0, 0xff, 0xfe, 0xfd, 0xfc, 0xfb, 0xfa, 0xf9, 0xf8, 0, 0, 0, 0, 0x7f, 0x7e, 0x7d, 0x7c, 0x7b, 0x7a, 0x79, 0x78, 0, 0, 0, 0, 0}; const byte *d; const N_BELE_RTP::AbstractPolicy *bele; d = dd + 7; assert_noexcept(upx_adler32(d, 4) == 0x09f003f7); assert_noexcept(upx_adler32(d, 4, 0) == 0x09ec03f6); assert_noexcept(upx_adler32(d, 4, 1) == 0x09f003f7); bele = &N_BELE_RTP::be_policy; assert_noexcept(get_be16(d) == 0xfffe); assert_noexcept(bele->get16(d) == 0xfffe); assert_noexcept(get_be16_signed(d) == -2); assert_noexcept(get_be24(d) == 0xfffefd); assert_noexcept(bele->get24(d) == 0xfffefd); assert_noexcept(get_be24_signed(d) == -259); assert_noexcept(get_be32(d) == 0xfffefdfc); assert_noexcept(bele->get32(d) == 0xfffefdfc); assert_noexcept(get_be32_signed(d) == -66052); bele = &N_BELE_RTP::le_policy; assert_noexcept(get_le16(d) == 0xfeff); assert_noexcept(bele->get16(d) == 0xfeff); assert_noexcept(get_le16_signed(d) == -257); assert_noexcept(get_le24(d) == 0xfdfeff); assert_noexcept(bele->get24(d) == 0xfdfeff); assert_noexcept(get_le24_signed(d) == -131329); assert_noexcept(get_le32(d) == 0xfcfdfeff); assert_noexcept(bele->get32(d) == 0xfcfdfeff); assert_noexcept(get_le32_signed(d) == -50462977); assert_noexcept(get_le64_signed(d) == -506097522914230529LL); assert_noexcept(find_be16(d, 2, 0xfffe) == 0); assert_noexcept(find_le16(d, 2, 0xfeff) == 0); assert_noexcept(find_be32(d, 4, 0xfffefdfc) == 0); assert_noexcept(find_le32(d, 4, 0xfcfdfeff) == 0); d += 12; assert_noexcept(get_be16_signed(d) == 32638); assert_noexcept(get_be24_signed(d) == 8355453); assert_noexcept(get_be32_signed(d) == 2138996092); assert_noexcept(get_be64_signed(d) == 9186918263483431288LL); } { unsigned dd; void *const d = ⅆ dd = ne32_to_le32(0xf7f6f5f4); assert_noexcept(get_le26(d) == 0x03f6f5f4); set_le26(d, 0); assert_noexcept(get_le26(d) == 0); assert_noexcept(dd == ne32_to_le32(0xf4000000)); set_le26(d, 0xff020304); assert_noexcept(get_le26(d) == 0x03020304); assert_noexcept(dd == ne32_to_le32(0xf7020304)); } { upx_uint16_t a = 0; upx_uint32_t b = 0; upx_uint64_t c = 0; set_ne16(&a, 0x04030201); // ignore upper bits set_ne32(&b, 0x04030201); set_ne64(&c, 0x0807060504030201ull); assert_noexcept(a == 0x0201); assert_noexcept(b == 0x04030201); assert_noexcept(c == 0x0807060504030201ull); assert_noexcept(get_ne16(&a) == 0x0201); assert_noexcept(get_ne32(&b) == 0x04030201); assert_noexcept(get_ne64(&c) == 0x0807060504030201ull); } #endif // DEBUG union { short v_short; int v_int; long v_long; long long v_llong; BE16 b16; BE32 b32; BE64 b64; LE16 l16; LE32 l32; LE64 l64; } u; assert_noexcept(testNoAliasing(&u.v_short, &u.b32)); assert_noexcept(testNoAliasing(&u.v_short, &u.l32)); assert_noexcept(testNoAliasing(&u.v_int, &u.b64)); assert_noexcept(testNoAliasing(&u.v_int, &u.l64)); // check working -fno-strict-aliasing assert_noexcept(testNoAliasing(&u.v_short, &u.v_int)); assert_noexcept(testNoAliasing(&u.v_int, &u.v_long)); assert_noexcept(testNoAliasing(&u.v_int, &u.v_llong)); assert_noexcept(testNoAliasing(&u.v_long, &u.v_llong)); assert_noexcept(TestIntegerWrap::inc_gt(0)); assert_noexcept(!TestIntegerWrap::inc_gt(UINT_MAX)); assert_noexcept(TestIntegerWrap::dec_lt(1)); assert_noexcept(!TestIntegerWrap::dec_lt(0)); assert_noexcept(TestIntegerWrap::neg_eq(0)); assert_noexcept(!TestIntegerWrap::neg_eq(1)); assert_noexcept(!TestIntegerWrap::neg_eq(UINT_MAX)); // check working -fno-strict-overflow assert_noexcept(TestIntegerWrap::inc_gt(0)); assert_noexcept(!TestIntegerWrap::inc_gt(INT_MAX)); assert_noexcept(TestIntegerWrap::dec_lt(0)); assert_noexcept(!TestIntegerWrap::dec_lt(INT_MIN)); assert_noexcept(TestIntegerWrap::neg_eq(0)); assert_noexcept(!TestIntegerWrap::neg_eq(1)); assert_noexcept(!TestIntegerWrap::neg_eq(INT_MAX)); assert_noexcept(TestIntegerWrap::neg_eq(INT_MIN)); // special case } /************************************************************************* // some doctest test cases **************************************************************************/ TEST_CASE("assert_noexcept") { // just to make sure that our own assert() macros don't generate any warnings byte dummy = 0; byte *ptr1 = &dummy; const byte *const ptr2 = &dummy; void *ptr3 = nullptr; assert(true); assert(1); assert(ptr1); assert(ptr2); assert(!ptr3); assert_noexcept(true); assert_noexcept(1); assert_noexcept(ptr1); assert_noexcept(ptr2); assert_noexcept(!ptr3); } TEST_CASE("acc_vget") { CHECK_EQ(acc_vget_int(0, 0), 0); CHECK_EQ(acc_vget_long(1, -1), 1); CHECK_EQ(acc_vget_acc_int64l_t(2, 1), 2); CHECK_EQ(acc_vget_acc_hvoid_p(nullptr, 0), nullptr); } TEST_CASE("working -fno-strict-aliasing") { bool ok; long v = 0; short *ps = ACC_STATIC_CAST(short *, acc_vget_acc_hvoid_p(&v, 0)); int *pi = ACC_STATIC_CAST(int *, acc_vget_acc_hvoid_p(&v, 0)); long *pl = ACC_STATIC_CAST(long *, acc_vget_acc_hvoid_p(&v, 0)); *ps = 0; *pl = -1; ok = *ps == -1; CHECK(ok); *pi = 0; *pl = -1; ok = *pi == -1; CHECK(ok); *pl = 0; *ps = -1; ok = *pl != 0; CHECK(ok); *pl = 0; *pi = -1; ok = *pl != 0; CHECK(ok); UNUSED(ok); } TEST_CASE("working -fno-strict-overflow") { CHECK_EQ(acc_vget_int(INT_MAX, 0) + 1, INT_MIN); CHECK_EQ(acc_vget_int(INT_MIN, 0) - 1, INT_MAX); CHECK_EQ(acc_vget_long(LONG_MAX, 0) + 1, LONG_MIN); CHECK_EQ(acc_vget_long(LONG_MIN, 0) - 1, LONG_MAX); bool ok; int i; i = INT_MAX; i += 1; ok = i == INT_MIN; CHECK(ok); i = INT_MIN; i -= 1; ok = i == INT_MAX; CHECK(ok); UNUSED(ok); } TEST_CASE("libc snprintf") { // runtime check that Windows/MinGW works as expected char buf[64]; long long ll = acc_vget_int(-1, 0); unsigned long long llu = (unsigned long long) ll; snprintf(buf, sizeof(buf), "%d.%ld.%lld.%u.%lu.%llu", -3, -2L, ll, 3U, 2LU, llu); CHECK_EQ(strcmp(buf, "-3.-2.-1.3.2.18446744073709551615"), 0); intmax_t im = ll; uintmax_t um = llu; snprintf(buf, sizeof(buf), "%d.%d.%d.%d.%d.%d.%d.%d.%d.%jd", -4, 0, 0, 0, 0, 0, 0, 0, 4, im); CHECK_EQ(strcmp(buf, "-4.0.0.0.0.0.0.0.4.-1"), 0); snprintf(buf, sizeof(buf), "%d.%d.%d.%d.%d.%d.%d.%d.%d.%ju", -5, 0, 0, 0, 0, 0, 0, 0, 5, um); CHECK_EQ(strcmp(buf, "-5.0.0.0.0.0.0.0.5.18446744073709551615"), 0); snprintf(buf, sizeof(buf), "%d.%d.%d.%d.%d.%d.%d.%d.%d.%jx", -6, 0, 0, 0, 0, 0, 0, 0, 6, um); CHECK_EQ(strcmp(buf, "-6.0.0.0.0.0.0.0.6.ffffffffffffffff"), 0); snprintf(buf, sizeof(buf), "%d.%d.%d.%d.%d.%d.%d.%d.%d.%#jx", -7, 0, 0, 0, 0, 0, 0, 0, 7, um); CHECK_EQ(strcmp(buf, "-7.0.0.0.0.0.0.0.7.0xffffffffffffffff"), 0); } TEST_CASE("libc qsort") { // runtime check that libc qsort() never compares identical objects struct Elem { upx_uint16_t id; upx_uint16_t value; static int compare(const void *aa, const void *bb) noexcept { const Elem *a = (const Elem *) aa; const Elem *b = (const Elem *) bb; assert_noexcept(a->id != b->id); // check not IDENTICAL return a->value < b->value ? -1 : (a->value == b->value ? 0 : 1); } static bool qsort_check(Elem *e, size_t n) { upx_uint32_t x = n + 5381; for (size_t i = 0; i < n; i++) { e[i].id = (upx_uint16_t) i; x = x * 33 + (i & 255); e[i].value = (upx_uint16_t) x; } qsort(e, n, sizeof(*e), Elem::compare); bool sorted_ok = true; for (size_t i = 1; i < n; i++) sorted_ok &= e[i - 1].value <= e[i].value; return sorted_ok; } }; constexpr size_t N = 4096; Elem e[N]; for (size_t n = 1; n <= N; n <<= 1) CHECK(Elem::qsort_check(e, n)); } /* vim:set ts=4 sw=4 et: */