277 lines
9.1 KiB
C++
277 lines
9.1 KiB
C++
/*
|
|
* Copyright (C) 2016 The Android Open Source Project
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
#include <string>
|
|
#include <vector>
|
|
|
|
#include <android-base/test_utils.h>
|
|
#include <android-base/file.h>
|
|
#include <gtest/gtest.h>
|
|
|
|
#include <unwindstack/Memory.h>
|
|
|
|
namespace unwindstack {
|
|
|
|
class MemoryFileTest : public ::testing::Test {
|
|
protected:
|
|
void SetUp() override {
|
|
tf_ = new TemporaryFile;
|
|
}
|
|
|
|
void TearDown() override {
|
|
delete tf_;
|
|
}
|
|
|
|
void WriteTestData() {
|
|
ASSERT_TRUE(android::base::WriteStringToFd("0123456789abcdefghijklmnopqrstuvxyz", tf_->fd));
|
|
}
|
|
|
|
MemoryFileAtOffset memory_;
|
|
|
|
TemporaryFile* tf_ = nullptr;
|
|
};
|
|
|
|
TEST_F(MemoryFileTest, init_offset_0) {
|
|
WriteTestData();
|
|
|
|
ASSERT_TRUE(memory_.Init(tf_->path, 0));
|
|
std::vector<char> buffer(11);
|
|
ASSERT_TRUE(memory_.ReadFully(0, buffer.data(), 10));
|
|
buffer[10] = '\0';
|
|
ASSERT_STREQ("0123456789", buffer.data());
|
|
}
|
|
|
|
TEST_F(MemoryFileTest, init_offset_non_zero) {
|
|
WriteTestData();
|
|
|
|
ASSERT_TRUE(memory_.Init(tf_->path, 10));
|
|
std::vector<char> buffer(11);
|
|
ASSERT_TRUE(memory_.ReadFully(0, buffer.data(), 10));
|
|
buffer[10] = '\0';
|
|
ASSERT_STREQ("abcdefghij", buffer.data());
|
|
}
|
|
|
|
TEST_F(MemoryFileTest, init_offset_non_zero_larger_than_pagesize) {
|
|
size_t pagesize = getpagesize();
|
|
std::string large_string;
|
|
for (size_t i = 0; i < pagesize; i++) {
|
|
large_string += '1';
|
|
}
|
|
large_string += "012345678901234abcdefgh";
|
|
ASSERT_TRUE(android::base::WriteStringToFd(large_string, tf_->fd));
|
|
|
|
ASSERT_TRUE(memory_.Init(tf_->path, pagesize + 15));
|
|
std::vector<char> buffer(9);
|
|
ASSERT_TRUE(memory_.ReadFully(0, buffer.data(), 8));
|
|
buffer[8] = '\0';
|
|
ASSERT_STREQ("abcdefgh", buffer.data());
|
|
}
|
|
|
|
TEST_F(MemoryFileTest, init_offset_pagesize_aligned) {
|
|
size_t pagesize = getpagesize();
|
|
std::string data;
|
|
for (size_t i = 0; i < 2 * pagesize; i++) {
|
|
data += static_cast<char>((i / pagesize) + '0');
|
|
data += static_cast<char>((i % 10) + '0');
|
|
}
|
|
ASSERT_TRUE(android::base::WriteStringToFd(data, tf_->fd));
|
|
|
|
ASSERT_TRUE(memory_.Init(tf_->path, 2 * pagesize));
|
|
std::vector<char> buffer(11);
|
|
ASSERT_TRUE(memory_.ReadFully(0, buffer.data(), 10));
|
|
buffer[10] = '\0';
|
|
std::string expected_str;
|
|
for (size_t i = 0; i < 5; i++) {
|
|
expected_str += '1';
|
|
expected_str += static_cast<char>(((i + pagesize) % 10) + '0');
|
|
}
|
|
ASSERT_STREQ(expected_str.c_str(), buffer.data());
|
|
}
|
|
|
|
TEST_F(MemoryFileTest, init_offset_pagesize_aligned_plus_extra) {
|
|
size_t pagesize = getpagesize();
|
|
std::string data;
|
|
for (size_t i = 0; i < 2 * pagesize; i++) {
|
|
data += static_cast<char>((i / pagesize) + '0');
|
|
data += static_cast<char>((i % 10) + '0');
|
|
}
|
|
ASSERT_TRUE(android::base::WriteStringToFd(data, tf_->fd));
|
|
|
|
ASSERT_TRUE(memory_.Init(tf_->path, 2 * pagesize + 10));
|
|
std::vector<char> buffer(11);
|
|
ASSERT_TRUE(memory_.ReadFully(0, buffer.data(), 10));
|
|
buffer[10] = '\0';
|
|
std::string expected_str;
|
|
for (size_t i = 0; i < 5; i++) {
|
|
expected_str += '1';
|
|
expected_str += static_cast<char>(((i + pagesize + 5) % 10) + '0');
|
|
}
|
|
ASSERT_STREQ(expected_str.c_str(), buffer.data());
|
|
}
|
|
|
|
TEST_F(MemoryFileTest, init_offset_greater_than_filesize) {
|
|
size_t pagesize = getpagesize();
|
|
std::string data;
|
|
uint64_t file_size = 2 * pagesize + pagesize / 2;
|
|
for (size_t i = 0; i < file_size; i++) {
|
|
data += static_cast<char>((i / pagesize) + '0');
|
|
}
|
|
ASSERT_TRUE(android::base::WriteStringToFd(data, tf_->fd));
|
|
|
|
// Check offset > file size fails and aligned_offset > file size.
|
|
ASSERT_FALSE(memory_.Init(tf_->path, file_size + 2 * pagesize));
|
|
// Check offset == filesize fails.
|
|
ASSERT_FALSE(memory_.Init(tf_->path, file_size));
|
|
// Check aligned_offset < filesize, but offset > filesize fails.
|
|
ASSERT_FALSE(memory_.Init(tf_->path, 2 * pagesize + pagesize / 2 + pagesize / 4));
|
|
}
|
|
|
|
TEST_F(MemoryFileTest, read_error) {
|
|
std::string data;
|
|
for (size_t i = 0; i < 5000; i++) {
|
|
data += static_cast<char>((i % 10) + '0');
|
|
}
|
|
ASSERT_TRUE(android::base::WriteStringToFd(data, tf_->fd));
|
|
|
|
std::vector<char> buffer(100);
|
|
|
|
// Read before init.
|
|
ASSERT_FALSE(memory_.ReadFully(0, buffer.data(), 10));
|
|
|
|
ASSERT_TRUE(memory_.Init(tf_->path, 0));
|
|
|
|
ASSERT_FALSE(memory_.ReadFully(10000, buffer.data(), 10));
|
|
ASSERT_FALSE(memory_.ReadFully(5000, buffer.data(), 10));
|
|
ASSERT_FALSE(memory_.ReadFully(4990, buffer.data(), 11));
|
|
ASSERT_TRUE(memory_.ReadFully(4990, buffer.data(), 10));
|
|
ASSERT_FALSE(memory_.ReadFully(4999, buffer.data(), 2));
|
|
ASSERT_TRUE(memory_.ReadFully(4999, buffer.data(), 1));
|
|
|
|
// Check that overflow fails properly.
|
|
ASSERT_FALSE(memory_.ReadFully(UINT64_MAX - 100, buffer.data(), 200));
|
|
}
|
|
|
|
TEST_F(MemoryFileTest, read_past_file_within_mapping) {
|
|
size_t pagesize = getpagesize();
|
|
|
|
ASSERT_TRUE(pagesize > 100);
|
|
std::vector<uint8_t> buffer(pagesize - 100);
|
|
for (size_t i = 0; i < pagesize - 100; i++) {
|
|
buffer[i] = static_cast<uint8_t>((i % 0x5e) + 0x20);
|
|
}
|
|
ASSERT_TRUE(android::base::WriteFully(tf_->fd, buffer.data(), buffer.size()));
|
|
|
|
ASSERT_TRUE(memory_.Init(tf_->path, 0));
|
|
|
|
for (size_t i = 0; i < 100; i++) {
|
|
uint8_t value;
|
|
ASSERT_FALSE(memory_.ReadFully(buffer.size() + i, &value, 1))
|
|
<< "Should have failed at value " << i;
|
|
}
|
|
}
|
|
|
|
TEST_F(MemoryFileTest, map_partial_offset_aligned) {
|
|
size_t pagesize = getpagesize();
|
|
std::vector<uint8_t> buffer(pagesize * 10);
|
|
for (size_t i = 0; i < pagesize * 10; i++) {
|
|
buffer[i] = i / pagesize + 1;
|
|
}
|
|
ASSERT_TRUE(android::base::WriteFully(tf_->fd, buffer.data(), buffer.size()));
|
|
|
|
// Map in only two pages of the data, and after the first page.
|
|
ASSERT_TRUE(memory_.Init(tf_->path, pagesize, pagesize * 2));
|
|
|
|
std::vector<uint8_t> read_buffer(pagesize * 2);
|
|
// Make sure that reading after mapped data is a failure.
|
|
ASSERT_FALSE(memory_.ReadFully(pagesize * 2, read_buffer.data(), 1));
|
|
ASSERT_TRUE(memory_.ReadFully(0, read_buffer.data(), pagesize * 2));
|
|
for (size_t i = 0; i < pagesize; i++) {
|
|
ASSERT_EQ(2, read_buffer[i]) << "Failed at byte " << i;
|
|
}
|
|
for (size_t i = pagesize; i < pagesize * 2; i++) {
|
|
ASSERT_EQ(3, read_buffer[i]) << "Failed at byte " << i;
|
|
}
|
|
}
|
|
|
|
TEST_F(MemoryFileTest, map_partial_offset_unaligned) {
|
|
size_t pagesize = getpagesize();
|
|
ASSERT_TRUE(pagesize > 0x100);
|
|
std::vector<uint8_t> buffer(pagesize * 10);
|
|
for (size_t i = 0; i < buffer.size(); i++) {
|
|
buffer[i] = i / pagesize + 1;
|
|
}
|
|
ASSERT_TRUE(android::base::WriteFully(tf_->fd, buffer.data(), buffer.size()));
|
|
|
|
// Map in only two pages of the data, and after the first page.
|
|
ASSERT_TRUE(memory_.Init(tf_->path, pagesize + 0x100, pagesize * 2));
|
|
|
|
std::vector<uint8_t> read_buffer(pagesize * 2);
|
|
// Make sure that reading after mapped data is a failure.
|
|
ASSERT_FALSE(memory_.ReadFully(pagesize * 2, read_buffer.data(), 1));
|
|
ASSERT_TRUE(memory_.ReadFully(0, read_buffer.data(), pagesize * 2));
|
|
for (size_t i = 0; i < pagesize - 0x100; i++) {
|
|
ASSERT_EQ(2, read_buffer[i]) << "Failed at byte " << i;
|
|
}
|
|
for (size_t i = pagesize - 0x100; i < 2 * pagesize - 0x100; i++) {
|
|
ASSERT_EQ(3, read_buffer[i]) << "Failed at byte " << i;
|
|
}
|
|
for (size_t i = 2 * pagesize - 0x100; i < pagesize * 2; i++) {
|
|
ASSERT_EQ(4, read_buffer[i]) << "Failed at byte " << i;
|
|
}
|
|
}
|
|
|
|
TEST_F(MemoryFileTest, map_overflow) {
|
|
size_t pagesize = getpagesize();
|
|
ASSERT_TRUE(pagesize > 0x100);
|
|
std::vector<uint8_t> buffer(pagesize * 10);
|
|
for (size_t i = 0; i < buffer.size(); i++) {
|
|
buffer[i] = i / pagesize + 1;
|
|
}
|
|
ASSERT_TRUE(android::base::WriteFully(tf_->fd, buffer.data(), buffer.size()));
|
|
|
|
// Map in only two pages of the data, and after the first page.
|
|
ASSERT_TRUE(memory_.Init(tf_->path, pagesize + 0x100, UINT64_MAX));
|
|
|
|
std::vector<uint8_t> read_buffer(pagesize * 10);
|
|
ASSERT_FALSE(memory_.ReadFully(pagesize * 9 - 0x100 + 1, read_buffer.data(), 1));
|
|
ASSERT_TRUE(memory_.ReadFully(0, read_buffer.data(), pagesize * 9 - 0x100));
|
|
}
|
|
|
|
TEST_F(MemoryFileTest, init_reinit) {
|
|
size_t pagesize = getpagesize();
|
|
std::vector<uint8_t> buffer(pagesize * 2);
|
|
for (size_t i = 0; i < buffer.size(); i++) {
|
|
buffer[i] = i / pagesize + 1;
|
|
}
|
|
ASSERT_TRUE(android::base::WriteFully(tf_->fd, buffer.data(), buffer.size()));
|
|
|
|
ASSERT_TRUE(memory_.Init(tf_->path, 0));
|
|
std::vector<uint8_t> read_buffer(buffer.size());
|
|
ASSERT_TRUE(memory_.ReadFully(0, read_buffer.data(), pagesize));
|
|
for (size_t i = 0; i < pagesize; i++) {
|
|
ASSERT_EQ(1, read_buffer[i]) << "Failed at byte " << i;
|
|
}
|
|
|
|
// Now reinit.
|
|
ASSERT_TRUE(memory_.Init(tf_->path, pagesize));
|
|
ASSERT_TRUE(memory_.ReadFully(0, read_buffer.data(), pagesize));
|
|
for (size_t i = 0; i < pagesize; i++) {
|
|
ASSERT_EQ(2, read_buffer[i]) << "Failed at byte " << i;
|
|
}
|
|
}
|
|
|
|
} // namespace unwindstack
|