普通文本  |  167行  |  5.58 KB

// 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