/* * Copyright (C) 2009 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 "base/unix_file/fd_file.h" #include "base/unix_file/random_access_file_test.h" #include "common_runtime_test.h" // For ScratchFile #include "gtest/gtest.h" namespace unix_file { class FdFileTest : public RandomAccessFileTest { protected: virtual RandomAccessFile* MakeTestFile() { return new FdFile(fileno(tmpfile()), false); } }; TEST_F(FdFileTest, Read) { TestRead(); } TEST_F(FdFileTest, SetLength) { TestSetLength(); } TEST_F(FdFileTest, Write) { TestWrite(); } TEST_F(FdFileTest, UnopenedFile) { FdFile file; EXPECT_EQ(-1, file.Fd()); EXPECT_FALSE(file.IsOpened()); EXPECT_TRUE(file.GetPath().empty()); } TEST_F(FdFileTest, OpenClose) { std::string good_path(GetTmpPath("some-file.txt")); FdFile file; ASSERT_TRUE(file.Open(good_path, O_CREAT | O_WRONLY)); EXPECT_GE(file.Fd(), 0); EXPECT_TRUE(file.IsOpened()); EXPECT_EQ(0, file.Flush()); EXPECT_EQ(0, file.Close()); EXPECT_EQ(-1, file.Fd()); EXPECT_FALSE(file.IsOpened()); EXPECT_TRUE(file.Open(good_path, O_RDONLY)); EXPECT_GE(file.Fd(), 0); EXPECT_TRUE(file.IsOpened()); ASSERT_EQ(file.Close(), 0); ASSERT_EQ(unlink(good_path.c_str()), 0); } TEST_F(FdFileTest, ReadFullyEmptyFile) { // New scratch file, zero-length. art::ScratchFile tmp; FdFile file; ASSERT_TRUE(file.Open(tmp.GetFilename(), O_RDONLY)); EXPECT_GE(file.Fd(), 0); EXPECT_TRUE(file.IsOpened()); uint8_t buffer[16]; EXPECT_FALSE(file.ReadFully(&buffer, 4)); } template static void NullTerminateCharArray(char (&array)[Size]) { array[Size - 1] = '\0'; } TEST_F(FdFileTest, ReadFullyWithOffset) { // New scratch file, zero-length. art::ScratchFile tmp; FdFile file; ASSERT_TRUE(file.Open(tmp.GetFilename(), O_RDWR)); EXPECT_GE(file.Fd(), 0); EXPECT_TRUE(file.IsOpened()); char ignore_prefix[20] = {'a', }; NullTerminateCharArray(ignore_prefix); char read_suffix[10] = {'b', }; NullTerminateCharArray(read_suffix); off_t offset = 0; // Write scratch data to file that we can read back into. EXPECT_TRUE(file.Write(ignore_prefix, sizeof(ignore_prefix), offset)); offset += sizeof(ignore_prefix); EXPECT_TRUE(file.Write(read_suffix, sizeof(read_suffix), offset)); ASSERT_EQ(file.Flush(), 0); // Reading at an offset should only produce 'bbbb...', since we ignore the 'aaa...' prefix. char buffer[sizeof(read_suffix)]; EXPECT_TRUE(file.PreadFully(buffer, sizeof(read_suffix), offset)); EXPECT_STREQ(&read_suffix[0], &buffer[0]); ASSERT_EQ(file.Close(), 0); } TEST_F(FdFileTest, ReadWriteFullyWithOffset) { // New scratch file, zero-length. art::ScratchFile tmp; FdFile file; ASSERT_TRUE(file.Open(tmp.GetFilename(), O_RDWR)); EXPECT_GE(file.Fd(), 0); EXPECT_TRUE(file.IsOpened()); const char* test_string = "This is a test string"; size_t length = strlen(test_string) + 1; const size_t offset = 12; std::unique_ptr offset_read_string(new char[length]); std::unique_ptr read_string(new char[length]); // Write scratch data to file that we can read back into. EXPECT_TRUE(file.PwriteFully(test_string, length, offset)); ASSERT_EQ(file.Flush(), 0); // Test reading both the offsets. EXPECT_TRUE(file.PreadFully(&offset_read_string[0], length, offset)); EXPECT_STREQ(test_string, &offset_read_string[0]); EXPECT_TRUE(file.PreadFully(&read_string[0], length, 0u)); EXPECT_NE(memcmp(&read_string[0], test_string, length), 0); ASSERT_EQ(file.Close(), 0); } TEST_F(FdFileTest, Copy) { art::ScratchFile src_tmp; FdFile src; ASSERT_TRUE(src.Open(src_tmp.GetFilename(), O_RDWR)); ASSERT_GE(src.Fd(), 0); ASSERT_TRUE(src.IsOpened()); char src_data[] = "Some test data."; ASSERT_TRUE(src.WriteFully(src_data, sizeof(src_data))); // Including the zero terminator. ASSERT_EQ(0, src.Flush()); ASSERT_EQ(static_cast(sizeof(src_data)), src.GetLength()); art::ScratchFile dest_tmp; FdFile dest; ASSERT_TRUE(dest.Open(src_tmp.GetFilename(), O_RDWR)); ASSERT_GE(dest.Fd(), 0); ASSERT_TRUE(dest.IsOpened()); ASSERT_TRUE(dest.Copy(&src, 0, sizeof(src_data))); ASSERT_EQ(0, dest.Flush()); ASSERT_EQ(static_cast(sizeof(src_data)), dest.GetLength()); char check_data[sizeof(src_data)]; ASSERT_TRUE(dest.PreadFully(check_data, sizeof(src_data), 0u)); CHECK_EQ(0, memcmp(check_data, src_data, sizeof(src_data))); ASSERT_EQ(0, dest.Close()); ASSERT_EQ(0, src.Close()); } } // namespace unix_file