C++程序  |  193行  |  6.32 KB

/*
 * 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