/* * Copyright (c) 2022, David Tuin * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include using namespace JS; template static void test_nullptr_input() { Type* ptr = nullptr; JS::Value val { ptr }; EXPECT(val.is_null()); EXPECT(!val.is_object()); EXPECT(!val.is_string()); EXPECT(!val.is_bigint()); EXPECT(!val.is_symbol()); EXPECT(!val.is_accessor()); EXPECT(!val.is_cell()); EXPECT(!val.is_number()); EXPECT(!val.is_undefined()); } #define TEST_NULLPTR_INPUT(type) \ TEST_CASE(value_nullptr_input_##type) \ { \ test_nullptr_input(); \ } TEST_NULLPTR_INPUT(Object); TEST_NULLPTR_INPUT(PrimitiveString); TEST_NULLPTR_INPUT(Symbol); TEST_NULLPTR_INPUT(BigInt); TEST_NULLPTR_INPUT(Accessor); #undef TEST_NULLPTR_INPUT TEST_CASE(valid_pointer_in_gives_same_pointer_out) { if (sizeof(void*) < sizeof(double)) return; #define EXPECT_POINTER_TO_SURVIVE(input) \ { \ JS::Value value(reinterpret_cast(static_cast(input))); \ EXPECT(value.is_object()); \ EXPECT(!value.is_null()); \ auto extracted_pointer = JS::Value::extract_pointer_bits(value.encoded()); \ EXPECT_EQ(static_cast(input), extracted_pointer); \ } EXPECT_POINTER_TO_SURVIVE(0x1); EXPECT_POINTER_TO_SURVIVE(0x10); EXPECT_POINTER_TO_SURVIVE(0x100); EXPECT_POINTER_TO_SURVIVE(0x00007fffffffffff); EXPECT_POINTER_TO_SURVIVE(0x0000700000000000); EXPECT_POINTER_TO_SURVIVE(0x0000100000000000); #if ARCH(X86_64) // On x86-64, the top 16 bits of pointers are equal to bit 47. EXPECT_POINTER_TO_SURVIVE(0xffff800000000000); EXPECT_POINTER_TO_SURVIVE(0xffff800000000001); EXPECT_POINTER_TO_SURVIVE(0xffff800000000010); #elif ARCH(AARCH64) // ... but they should contain zeroes on AArch64. EXPECT_POINTER_TO_SURVIVE(0x0000800000000000); EXPECT_POINTER_TO_SURVIVE(0x0000800000000001); EXPECT_POINTER_TO_SURVIVE(0x0000800000000010); #endif #undef EXPECT_POINTER_TO_SURVIVE } TEST_CASE(non_canon_nans) { #define EXPECT_TO_BE_NAN(input) \ { \ Value val { bit_cast(input) }; \ EXPECT(val.is_nan()); \ EXPECT(val.is_number()); \ EXPECT(!val.is_integral_number()); \ EXPECT(!val.is_finite_number()); \ EXPECT(!val.is_infinity()); \ EXPECT(!val.is_empty()); \ EXPECT(!val.is_nullish()); \ } EXPECT_TO_BE_NAN(CANON_NAN_BITS | 0x1); EXPECT_TO_BE_NAN(CANON_NAN_BITS | 0x10); EXPECT_TO_BE_NAN(CANON_NAN_BITS | (NULL_TAG << TAG_SHIFT)); EXPECT_TO_BE_NAN(CANON_NAN_BITS | (UNDEFINED_TAG << TAG_SHIFT)); EXPECT_TO_BE_NAN(CANON_NAN_BITS | (INT32_TAG << TAG_SHIFT) | 0x88); EXPECT_TO_BE_NAN(CANON_NAN_BITS | (OBJECT_TAG << TAG_SHIFT)); EXPECT_TO_BE_NAN(CANON_NAN_BITS | (OBJECT_TAG << TAG_SHIFT) | 0x1230); EXPECT_TO_BE_NAN(CANON_NAN_BITS | (STRING_TAG << TAG_SHIFT)); EXPECT_TO_BE_NAN(CANON_NAN_BITS | (STRING_TAG << TAG_SHIFT) | 0x1230); u64 sign_bit = 1ULL << 63; EXPECT_TO_BE_NAN(CANON_NAN_BITS | sign_bit | 0x1); EXPECT_TO_BE_NAN(CANON_NAN_BITS | sign_bit | 0x10); EXPECT_TO_BE_NAN(CANON_NAN_BITS | sign_bit | (NULL_TAG << TAG_SHIFT)); EXPECT_TO_BE_NAN(CANON_NAN_BITS | sign_bit | (UNDEFINED_TAG << TAG_SHIFT)); EXPECT_TO_BE_NAN(CANON_NAN_BITS | sign_bit | (INT32_TAG << TAG_SHIFT) | 0x88); EXPECT_TO_BE_NAN(CANON_NAN_BITS | sign_bit | (OBJECT_TAG << TAG_SHIFT)); EXPECT_TO_BE_NAN(CANON_NAN_BITS | sign_bit | (OBJECT_TAG << TAG_SHIFT) | 0x1230); EXPECT_TO_BE_NAN(CANON_NAN_BITS | sign_bit | (STRING_TAG << TAG_SHIFT)); EXPECT_TO_BE_NAN(CANON_NAN_BITS | sign_bit | (STRING_TAG << TAG_SHIFT) | 0x1230); #undef EXPECT_TO_BE_NAN }