1 /* 2 * Copyright (C) 2009 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 #ifndef ART_RUNTIME_BASE_UNIX_FILE_RANDOM_ACCESS_FILE_TEST_H_ 18 #define ART_RUNTIME_BASE_UNIX_FILE_RANDOM_ACCESS_FILE_TEST_H_ 19 20 #include <errno.h> 21 #include <memory> 22 #include <string> 23 24 #include "common_runtime_test.h" 25 26 namespace unix_file { 27 28 class RandomAccessFileTest : public testing::Test { 29 protected: ~RandomAccessFileTest()30 virtual ~RandomAccessFileTest() { 31 } 32 33 // Override this to return an instance of the subclass under test that's 34 // backed by a temporary file. 35 virtual RandomAccessFile* MakeTestFile() = 0; 36 SetUp()37 virtual void SetUp() { 38 art::CommonRuntimeTest::SetUpAndroidData(android_data_); 39 } 40 TearDown()41 virtual void TearDown() { 42 art::CommonRuntimeTest::TearDownAndroidData(android_data_, true); 43 } 44 GetTmpPath(const std::string & name)45 std::string GetTmpPath(const std::string& name) { 46 std::string path; 47 path = android_data_; 48 path += "/"; 49 path += name; 50 return path; 51 } 52 53 // TODO(enh): ReadString (and WriteString) might be generally useful. ReadString(RandomAccessFile * f,std::string * s)54 static bool ReadString(RandomAccessFile* f, std::string* s) { 55 s->clear(); 56 char buf[256]; 57 int64_t n = 0; 58 int64_t offset = 0; 59 while ((n = f->Read(buf, sizeof(buf), offset)) > 0) { 60 s->append(buf, n); 61 offset += n; 62 } 63 return n != -1; 64 } 65 TestRead()66 void TestRead() { 67 char buf[256]; 68 std::unique_ptr<RandomAccessFile> file(MakeTestFile()); 69 70 // Reading from the start of an empty file gets you zero bytes, however many 71 // you ask for. 72 ASSERT_EQ(0, file->Read(buf, 0, 0)); 73 ASSERT_EQ(0, file->Read(buf, 123, 0)); 74 75 const std::string content("hello"); 76 ASSERT_EQ(content.size(), static_cast<uint64_t>(file->Write(content.data(), content.size(), 0))); 77 78 TestReadContent(content, file.get()); 79 80 CleanUp(file.get()); 81 } 82 TestReadContent(const std::string & content,RandomAccessFile * file)83 void TestReadContent(const std::string& content, RandomAccessFile* file) { 84 const int buf_size = content.size() + 10; 85 std::unique_ptr<char[]> buf(new char[buf_size]); 86 // Can't read from a negative offset. 87 ASSERT_EQ(-EINVAL, file->Read(buf.get(), 0, -123)); 88 89 // Reading too much gets us just what's in the file. 90 ASSERT_EQ(content.size(), static_cast<uint64_t>(file->Read(buf.get(), buf_size, 0))); 91 ASSERT_EQ(std::string(buf.get(), content.size()), content); 92 93 // We only get as much as we ask for. 94 const size_t short_request = 2; 95 ASSERT_LT(short_request, content.size()); 96 ASSERT_EQ(short_request, static_cast<uint64_t>(file->Read(buf.get(), short_request, 0))); 97 ASSERT_EQ(std::string(buf.get(), short_request), 98 content.substr(0, short_request)); 99 100 // We don't have to start at the beginning. 101 const int non_zero_offset = 2; 102 ASSERT_GT(non_zero_offset, 0); 103 ASSERT_EQ(short_request, static_cast<uint64_t>(file->Read(buf.get(), short_request, 104 non_zero_offset))); 105 ASSERT_EQ(std::string(buf.get(), short_request), 106 content.substr(non_zero_offset, short_request)); 107 108 // Reading past the end gets us nothing. 109 ASSERT_EQ(0, file->Read(buf.get(), buf_size, file->GetLength())); 110 ASSERT_EQ(0, file->Read(buf.get(), buf_size, file->GetLength() + 1)); 111 } 112 TestSetLength()113 void TestSetLength() { 114 const std::string content("hello"); 115 std::unique_ptr<RandomAccessFile> file(MakeTestFile()); 116 ASSERT_EQ(content.size(), static_cast<uint64_t>(file->Write(content.data(), content.size(), 0))); 117 ASSERT_EQ(content.size(), static_cast<uint64_t>(file->GetLength())); 118 119 // Can't give a file a negative length. 120 ASSERT_EQ(-EINVAL, file->SetLength(-123)); 121 122 // Can truncate the file. 123 int64_t new_length = 2; 124 ASSERT_EQ(0, file->SetLength(new_length)); 125 ASSERT_EQ(new_length, file->GetLength()); 126 std::string new_content; 127 ASSERT_TRUE(ReadString(file.get(), &new_content)); 128 ASSERT_EQ(content.substr(0, 2), new_content); 129 130 // Expanding the file appends zero bytes. 131 new_length = file->GetLength() + 1; 132 ASSERT_EQ(0, file->SetLength(new_length)); 133 ASSERT_EQ(new_length, file->GetLength()); 134 ASSERT_TRUE(ReadString(file.get(), &new_content)); 135 ASSERT_EQ('\0', new_content[new_length - 1]); 136 137 CleanUp(file.get()); 138 } 139 TestWrite()140 void TestWrite() { 141 const std::string content("hello"); 142 std::unique_ptr<RandomAccessFile> file(MakeTestFile()); 143 144 // Can't write to a negative offset. 145 ASSERT_EQ(-EINVAL, file->Write(content.data(), 0, -123)); 146 147 // Writing zero bytes of data is a no-op. 148 ASSERT_EQ(0, file->Write(content.data(), 0, 0)); 149 ASSERT_EQ(0, file->GetLength()); 150 151 // We can write data. 152 ASSERT_EQ(content.size(), static_cast<uint64_t>(file->Write(content.data(), content.size(), 0))); 153 ASSERT_EQ(content.size(), static_cast<uint64_t>(file->GetLength())); 154 std::string new_content; 155 ASSERT_TRUE(ReadString(file.get(), &new_content)); 156 ASSERT_EQ(new_content, content); 157 158 // We can read it back. 159 char buf[256]; 160 ASSERT_EQ(content.size(), static_cast<uint64_t>(file->Read(buf, sizeof(buf), 0))); 161 ASSERT_EQ(std::string(buf, content.size()), content); 162 163 // We can append data past the end. 164 ASSERT_EQ(content.size(), static_cast<uint64_t>(file->Write(content.data(), content.size(), 165 file->GetLength() + 1))); 166 int64_t new_length = 2*content.size() + 1; 167 ASSERT_EQ(file->GetLength(), new_length); 168 ASSERT_TRUE(ReadString(file.get(), &new_content)); 169 ASSERT_EQ(std::string("hello\0hello", new_length), new_content); 170 171 CleanUp(file.get()); 172 } 173 CleanUp(RandomAccessFile * file ATTRIBUTE_UNUSED)174 virtual void CleanUp(RandomAccessFile* file ATTRIBUTE_UNUSED) { 175 } 176 177 protected: 178 std::string android_data_; 179 }; 180 181 } // namespace unix_file 182 183 #endif // ART_RUNTIME_BASE_UNIX_FILE_RANDOM_ACCESS_FILE_TEST_H_ 184