blob: b5f4538de05d33e598668e6ead5101b5949d0ed2 [file] [log] [blame]
/*
* 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 <android-base/test_utils.h>
#include <fcntl.h>
#include <gtest/gtest.h>
#include <string>
#include <unistd.h>
#include <utils/Log.h>
#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<char*>(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<char*>(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<char[]>(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