// Copyright 2016, VIXL authors
// All rights reserved.
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are met:
//
// * Redistributions of source code must retain the above copyright notice,
// this list of conditions and the following disclaimer.
// * 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.
// * Neither the name of ARM Limited nor the names of its contributors may be
// used to endorse or promote products derived from this software without
// specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS 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 THE COPYRIGHT OWNER OR 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 "code-buffer-vixl.h"
#include "test-runner.h"
namespace vixl {
#define TEST(name) TEST_(CODE_BUFFER_##name)
TEST(align_grow) {
CodeBuffer code_buffer(2);
VIXL_CHECK(code_buffer.GetCapacity() == 2);
VIXL_CHECK(code_buffer.GetRemainingBytes() == 2);
code_buffer.Emit16(0);
VIXL_CHECK(code_buffer.GetCapacity() == 2);
VIXL_CHECK(code_buffer.GetRemainingBytes() == 0);
// Check that the buffer can automatically grow when aligning the cursor.
VIXL_CHECK(IsAligned<2>(code_buffer.GetCursorOffset()));
code_buffer.Align();
VIXL_CHECK(IsWordAligned(code_buffer.GetCursorOffset()));
VIXL_CHECK(code_buffer.GetCapacity() > 2);
code_buffer.SetClean();
}
static void TestDefaultsHelper(const CodeBuffer& buffer) {
VIXL_CHECK(buffer.GetCapacity() == CodeBuffer::kDefaultCapacity);
VIXL_CHECK(buffer.HasSpaceFor(CodeBuffer::kDefaultCapacity));
VIXL_CHECK(!buffer.HasSpaceFor(CodeBuffer::kDefaultCapacity + 1));
VIXL_CHECK(buffer.GetCursorOffset() == 0);
VIXL_CHECK(buffer.GetOffsetFrom(0) == 0);
VIXL_CHECK(buffer.IsManaged());
VIXL_CHECK(!buffer.IsDirty());
VIXL_CHECK(buffer.GetRemainingBytes() == CodeBuffer::kDefaultCapacity);
VIXL_CHECK(buffer.GetSizeInBytes() == 0);
VIXL_CHECK(buffer.GetEndAddress<uintptr_t>() ==
buffer.GetStartAddress<uintptr_t>());
}
TEST(defaults) {
CodeBuffer buffer;
TestDefaultsHelper(buffer);
}
TEST(reset) {
CodeBuffer buffer;
// Update the buffer by writing to it.
buffer.Emit("dummy data");
VIXL_CHECK(buffer.IsDirty());
VIXL_CHECK(buffer.GetSizeInBytes() > 0);
// Calling Reset() should reset it back to its default state. (It does not
// shrink the capacity, but it should not have grown here.)
VIXL_ASSERT(buffer.GetCapacity() == CodeBuffer::kDefaultCapacity);
buffer.Reset();
TestDefaultsHelper(buffer);
}
TEST(ensure_space) {
const size_t initial_capacity = 1234;
CodeBuffer buffer(initial_capacity);
// Requesting less space than we already have should have no effect.
for (size_t space = 0; space < initial_capacity; space++) {
buffer.EnsureSpaceFor(space);
VIXL_CHECK(buffer.GetCapacity() == initial_capacity);
}
// Requesting more memory grows the buffer by an unspecified amount.
buffer.EnsureSpaceFor(initial_capacity + 1);
VIXL_CHECK(buffer.GetCapacity() > initial_capacity);
}
TEST(emit) {
CodeBuffer buffer;
VIXL_ASSERT(buffer.GetSizeInBytes() == 0);
uint64_t base_value = 0x0100001000100101;
const char* test_string = "test string";
size_t expected_size = 0;
// Simple emissions. This should not align or pad in any way.
buffer.EmitData(&base_value, 7);
expected_size += 7;
buffer.EmitString(test_string);
expected_size += strlen(test_string) + 1; // EmitString() emits the '\0'.
buffer.Emit64(static_cast<uint64_t>(base_value * 1));
buffer.Emit(static_cast<uint64_t>(base_value * 2));
expected_size += 16;
buffer.Emit32(static_cast<uint32_t>(base_value * 3));
buffer.Emit(static_cast<uint32_t>(base_value * 4));
expected_size += 8;
buffer.Emit16(static_cast<uint16_t>(base_value * 5));
buffer.Emit(static_cast<uint16_t>(base_value * 6));
expected_size += 4;
buffer.Emit8(static_cast<uint8_t>(base_value * 7));
buffer.Emit(static_cast<uint8_t>(base_value * 8));
expected_size += 2;
VIXL_CHECK(buffer.GetSizeInBytes() == expected_size);
buffer.SetClean();
// clang-format off
uint8_t expected[] = {
// EmitData
0x01, 0x01, 0x10, 0x00, 0x10, 0x00, 0x00,
// EmitString
't', 'e', 's', 't', ' ', 's', 't', 'r', 'i', 'n', 'g', '\0',
// Emit64
0x01, 0x01, 0x10, 0x00, 0x10, 0x00, 0x00, 0x01,
// Emit<uint64_t>
0x02, 0x02, 0x20, 0x00, 0x20, 0x00, 0x00, 0x02,
// Emit32
0x03, 0x03, 0x30, 0x00,
// Emit<uint32_t>
0x04, 0x04, 0x40, 0x00,
// Emit16
0x05, 0x05,
// Emit<uint16_t>
0x06, 0x06,
// Emit8
0x07,
// Emit<uint8_t>
0x08
};
// clang-format on
VIXL_ASSERT(expected_size == sizeof(expected));
VIXL_CHECK(memcmp(buffer.GetStartAddress<const void*>(),
expected,
expected_size) == 0);
}
} // namespace vixl