/* * Copyright 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. */ #define LOG_TAG "AsyncIO_test.cpp" #include #include #include #include #include #include #include "AsyncIO.h" namespace android { constexpr int TEST_PACKET_SIZE = 512; constexpr int POOL_COUNT = 10; static const std::string dummyDataStr = "/*\n * Copyright 2015 The Android Open Source Project\n *\n * Licensed un" "der the Apache License, Version 2.0 (the \"License\");\n * you may not us" "e this file except in compliance with the License.\n * You may obtain a c" "opy of the License at\n *\n * http://www.apache.org/licenses/LICENSE" "-2.0\n *\n * Unless required by applicable law or agreed to in writing, s" "oftware\n * distributed under the License is distributed on an \"AS IS\" " "BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express o" "r implied.\n * Se"; class AsyncIOTest : public ::testing::Test { protected: TemporaryFile dummy_file; AsyncIOTest() {} ~AsyncIOTest() {} }; TEST_F(AsyncIOTest, testRead) { char buf[TEST_PACKET_SIZE + 1]; buf[TEST_PACKET_SIZE] = '\0'; EXPECT_EQ(write(dummy_file.fd, dummyDataStr.c_str(), TEST_PACKET_SIZE), TEST_PACKET_SIZE); struct aiocb aio; struct aiocb *aiol[] = {&aio}; aio.aio_fildes = dummy_file.fd; aio.aio_buf = buf; aio.aio_offset = 0; aio.aio_nbytes = TEST_PACKET_SIZE; EXPECT_EQ(aio_read(&aio), 0); EXPECT_EQ(aio_suspend(aiol, 1, nullptr), 0); EXPECT_EQ(aio_return(&aio), TEST_PACKET_SIZE); EXPECT_STREQ(buf, dummyDataStr.c_str()); } TEST_F(AsyncIOTest, testWrite) { char buf[TEST_PACKET_SIZE + 1]; buf[TEST_PACKET_SIZE] = '\0'; struct aiocb aio; struct aiocb *aiol[] = {&aio}; aio.aio_fildes = dummy_file.fd; aio.aio_buf = const_cast(dummyDataStr.c_str()); aio.aio_offset = 0; aio.aio_nbytes = TEST_PACKET_SIZE; EXPECT_EQ(aio_write(&aio), 0); EXPECT_EQ(aio_suspend(aiol, 1, nullptr), 0); EXPECT_EQ(aio_return(&aio), TEST_PACKET_SIZE); EXPECT_EQ(read(dummy_file.fd, buf, TEST_PACKET_SIZE), TEST_PACKET_SIZE); EXPECT_STREQ(buf, dummyDataStr.c_str()); } TEST_F(AsyncIOTest, testError) { char buf[TEST_PACKET_SIZE + 1]; buf[TEST_PACKET_SIZE] = '\0'; struct aiocb aio; struct aiocb *aiol[] = {&aio}; aio.aio_fildes = -1; aio.aio_buf = const_cast(dummyDataStr.c_str()); aio.aio_offset = 0; aio.aio_nbytes = TEST_PACKET_SIZE; EXPECT_EQ(aio_write(&aio), 0); EXPECT_EQ(aio_suspend(aiol, 1, nullptr), 0); EXPECT_EQ(aio_return(&aio), -1); EXPECT_EQ(aio_error(&aio), EBADF); } TEST_F(AsyncIOTest, testSpliceRead) { char buf[TEST_PACKET_SIZE + 1]; buf[TEST_PACKET_SIZE] = '\0'; int pipeFd[2]; EXPECT_EQ(pipe(pipeFd), 0); EXPECT_EQ(write(dummy_file.fd, dummyDataStr.c_str(), TEST_PACKET_SIZE), TEST_PACKET_SIZE); struct aiocb aio; struct aiocb *aiol[] = {&aio}; aio.aio_fildes = dummy_file.fd; aio.aio_sink = pipeFd[1]; aio.aio_offset = 0; aio.aio_nbytes = TEST_PACKET_SIZE; EXPECT_EQ(aio_splice_read(&aio), 0); EXPECT_EQ(aio_suspend(aiol, 1, nullptr), 0); EXPECT_EQ(aio_return(&aio), TEST_PACKET_SIZE); EXPECT_EQ(read(pipeFd[0], buf, TEST_PACKET_SIZE), TEST_PACKET_SIZE); EXPECT_STREQ(buf, dummyDataStr.c_str()); } TEST_F(AsyncIOTest, testSpliceWrite) { char buf[TEST_PACKET_SIZE + 1]; buf[TEST_PACKET_SIZE] = '\0'; int pipeFd[2]; EXPECT_EQ(pipe(pipeFd), 0); EXPECT_EQ(write(pipeFd[1], dummyDataStr.c_str(), TEST_PACKET_SIZE), TEST_PACKET_SIZE); struct aiocb aio; struct aiocb *aiol[] = {&aio}; aio.aio_fildes = pipeFd[0]; aio.aio_sink = dummy_file.fd; aio.aio_offset = 0; aio.aio_nbytes = TEST_PACKET_SIZE; EXPECT_EQ(aio_splice_write(&aio), 0); EXPECT_EQ(aio_suspend(aiol, 1, nullptr), 0); EXPECT_EQ(aio_return(&aio), TEST_PACKET_SIZE); EXPECT_EQ(read(dummy_file.fd, buf, TEST_PACKET_SIZE), TEST_PACKET_SIZE); EXPECT_STREQ(buf, dummyDataStr.c_str()); } TEST_F(AsyncIOTest, testPoolWrite) { aio_pool_write_init(); char buf[TEST_PACKET_SIZE * POOL_COUNT + 1]; buf[TEST_PACKET_SIZE * POOL_COUNT] = '\0'; for (int i = 0; i < POOL_COUNT; i++) { struct aiocb *aiop = new struct aiocb; aiop->aio_fildes = dummy_file.fd; aiop->aio_pool_buf = std::unique_ptr(new char[TEST_PACKET_SIZE]); memcpy(aiop->aio_pool_buf.get(), dummyDataStr.c_str(), TEST_PACKET_SIZE); aiop->aio_offset = i * TEST_PACKET_SIZE; aiop->aio_nbytes = TEST_PACKET_SIZE; EXPECT_EQ(aio_pool_write(aiop), 0); } aio_pool_end(); EXPECT_EQ(read(dummy_file.fd, buf, TEST_PACKET_SIZE * POOL_COUNT), TEST_PACKET_SIZE * POOL_COUNT); std::stringstream ss; for (int i = 0; i < POOL_COUNT; i++) ss << dummyDataStr; EXPECT_STREQ(buf, ss.str().c_str()); } TEST_F(AsyncIOTest, testSplicePoolWrite) { aio_pool_splice_init(); char buf[TEST_PACKET_SIZE * POOL_COUNT + 1]; buf[TEST_PACKET_SIZE * POOL_COUNT] = '\0'; for (int i = 0; i < POOL_COUNT; i++) { int pipeFd[2]; EXPECT_EQ(pipe(pipeFd), 0); EXPECT_EQ(write(pipeFd[1], dummyDataStr.c_str(), TEST_PACKET_SIZE), TEST_PACKET_SIZE); struct aiocb *aiop = new struct aiocb; aiop->aio_fildes = pipeFd[0]; aiop->aio_sink = dummy_file.fd; aiop->aio_offset = i * TEST_PACKET_SIZE; aiop->aio_nbytes = TEST_PACKET_SIZE; EXPECT_EQ(aio_pool_write(aiop), 0); } aio_pool_end(); EXPECT_EQ(read(dummy_file.fd, buf, TEST_PACKET_SIZE * POOL_COUNT), TEST_PACKET_SIZE * POOL_COUNT); std::stringstream ss; for (int i = 0; i < POOL_COUNT; i++) ss << dummyDataStr; EXPECT_STREQ(buf, ss.str().c_str()); } } // namespace android