blob: 2e1618c7b8abbce21cad53b5e10af68a25929e7e [file] [log] [blame]
/*
* Copyright (C) 2019-2021 Apple Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
* THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "config.h"
#include <wtf/Packed.h>
#include <wtf/HashMap.h>
#include <wtf/MathExtras.h>
#include <wtf/Vector.h>
#if OS(DARWIN)
#include <mach/vm_param.h>
#endif
namespace TestWebKitAPI {
struct PackedPair {
PackedPtr<uint8_t> key { nullptr };
PackedPtr<uint8_t> value { nullptr };
};
TEST(WTF_Packed, StructSize)
{
EXPECT_EQ(alignof(PackedPair), 1U);
#if CPU(X86_64)
EXPECT_EQ(sizeof(PackedPair), 12U);
#endif
{
Packed<double> value;
value = 4.2;
EXPECT_EQ(value.get(), 4.2);
}
{
uint64_t originalValue = 0xff00ff00dd00dd00UL;
Packed<uint64_t> value;
value = originalValue;
EXPECT_EQ(value.get(), originalValue);
EXPECT_EQ(alignof(Packed<uint64_t>), 1U);
EXPECT_EQ(sizeof(Packed<uint64_t>), sizeof(uint64_t));
}
}
TEST(WTF_Packed, AssignAndGet)
{
{
PackedPtr<uint8_t> key { nullptr };
static_assert(OS_CONSTANT(EFFECTIVE_ADDRESS_WIDTH) != 64, "");
uint8_t* candidates[] = {
0,
bitwise_cast<uint8_t*>(static_cast<uintptr_t>((1ULL << (OS_CONSTANT(EFFECTIVE_ADDRESS_WIDTH) / 2)) - 1)),
bitwise_cast<uint8_t*>(static_cast<uintptr_t>((1ULL << (OS_CONSTANT(EFFECTIVE_ADDRESS_WIDTH) - 1)) - 1)),
#if !CPU(X86_64) || OS(DARWIN) || OS(LINUX) || OS(WINDOWS)
// These OSes will never allocate user space addresses with
// bit 47 (i.e. OS_CONSTANT(EFFECTIVE_ADDRESS_WIDTH) - 1) set.
bitwise_cast<uint8_t*>(static_cast<uintptr_t>((1ULL << OS_CONSTANT(EFFECTIVE_ADDRESS_WIDTH)) - 1)),
#else
bitwise_cast<uint8_t*>(static_cast<uintptr_t>(~((1ULL << (OS_CONSTANT(EFFECTIVE_ADDRESS_WIDTH) - 1)) - 1))), // min higher half
bitwise_cast<uint8_t*>(std::numeric_limits<uintptr_t>::max()), // max higher half
#endif
};
int count = sizeof(candidates) / sizeof(uint8_t*);
for (int i = 0; i < count; i++) {
key = candidates[i];
EXPECT_EQ(key.get(), candidates[i]);
}
}
}
TEST(WTF_Packed, PackedAlignedPtr)
{
{
PackedAlignedPtr<uint8_t, 256> key { nullptr };
EXPECT_LE(sizeof(key), 5U);
}
{
PackedAlignedPtr<uint8_t, 16> key { nullptr };
#if (OS(IOS) || OS(TVOS) || OS(WATCHOS)) && CPU(ARM64)
EXPECT_EQ(sizeof(key), 4U);
#else
EXPECT_LE(sizeof(key), 6U);
#endif
}
}
struct PackingTarget {
unsigned m_value { 0 };
};
TEST(WTF_Packed, HashMap)
{
Vector<PackingTarget> vector;
HashMap<PackedPtr<PackingTarget>, unsigned> map;
vector.reserveCapacity(10000);
for (unsigned i = 0; i < 10000; ++i)
vector.uncheckedAppend(PackingTarget { i });
for (auto& target : vector)
map.add(&target, target.m_value);
for (auto& target : vector) {
EXPECT_TRUE(map.contains(&target));
EXPECT_EQ(map.get(&target), target.m_value);
}
}
} // namespace TestWebKitAPI