19762338dSopenharmony_ci/*
29762338dSopenharmony_ci * Copyright (C) 2024 HiHope Open Source Organization.
39762338dSopenharmony_ci * Licensed under the Apache License, Version 2.0 (the "License");
49762338dSopenharmony_ci * you may not use this file except in compliance with the License.
59762338dSopenharmony_ci * You may obtain a copy of the License at
69762338dSopenharmony_ci *
79762338dSopenharmony_ci *     http://www.apache.org/licenses/LICENSE-2.0
89762338dSopenharmony_ci *
99762338dSopenharmony_ci * Unless required by applicable law or agreed to in writing, software
109762338dSopenharmony_ci * distributed under the License is distributed on an "AS IS" BASIS,
119762338dSopenharmony_ci * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
129762338dSopenharmony_ci * See the License for the specific language governing permissions and
139762338dSopenharmony_ci * limitations under the License.
149762338dSopenharmony_ci */
159762338dSopenharmony_ci
169762338dSopenharmony_ci#include <cstring>
179762338dSopenharmony_ci#include <fcntl.h>
189762338dSopenharmony_ci#include <unistd.h>
199762338dSopenharmony_ci#include <gtest/gtest.h>
209762338dSopenharmony_ci#include <sys/types.h>
219762338dSopenharmony_ci#include <sys/uio.h>
229762338dSopenharmony_ci#include "securec.h"
239762338dSopenharmony_ci
249762338dSopenharmony_ciusing namespace testing::ext;
259762338dSopenharmony_ciusing namespace std;
269762338dSopenharmony_ci
279762338dSopenharmony_ciclass PreadvApiTest : public testing::Test {
289762338dSopenharmony_cipublic:
299762338dSopenharmony_ci    static void SetUpTestCase();
309762338dSopenharmony_ci    static void TearDownTestCase();
319762338dSopenharmony_ci    void SetUp();
329762338dSopenharmony_ci    void TearDown();
339762338dSopenharmony_ci
349762338dSopenharmony_ciprivate:
359762338dSopenharmony_ci};
369762338dSopenharmony_ci
379762338dSopenharmony_cistatic const char* TEST_DATA = "Hello, world!";
389762338dSopenharmony_cistatic const size_t TEST_LEN = strlen(TEST_DATA);
399762338dSopenharmony_cistatic const int MAX_LEN = 128;
409762338dSopenharmony_cistatic const char* TEST_FILE = "/data/local/tmp/test_file.txt";
419762338dSopenharmony_cistatic const char* EMPTY_FILE = "/data/local/tmp/empty_file.txt";
429762338dSopenharmony_ci
439762338dSopenharmony_civoid PreadvApiTest::SetUpTestCase()
449762338dSopenharmony_ci{
459762338dSopenharmony_ci}
469762338dSopenharmony_ci
479762338dSopenharmony_civoid PreadvApiTest::TearDownTestCase()
489762338dSopenharmony_ci{
499762338dSopenharmony_ci}
509762338dSopenharmony_ci
519762338dSopenharmony_civoid PreadvApiTest::SetUp()
529762338dSopenharmony_ci{
539762338dSopenharmony_ci    int fd = open(TEST_FILE, O_CREAT | O_RDWR, S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH);
549762338dSopenharmony_ci    write(fd, TEST_DATA, TEST_LEN);
559762338dSopenharmony_ci    close(fd);
569762338dSopenharmony_ci}
579762338dSopenharmony_ci
589762338dSopenharmony_civoid PreadvApiTest::TearDown()
599762338dSopenharmony_ci{
609762338dSopenharmony_ci    unlink(TEST_FILE);
619762338dSopenharmony_ci}
629762338dSopenharmony_ci
639762338dSopenharmony_ci/*
649762338dSopenharmony_ci * @tc.number : SUB_KERNEL_SYSCALL_PREADV_0100
659762338dSopenharmony_ci * @tc.name   : PreadvReadSuccess_0001
669762338dSopenharmony_ci * @tc.desc   : Test basic read functionality of preadv.
679762338dSopenharmony_ci * @tc.size   : MediumTest
689762338dSopenharmony_ci * @tc.type   : Function
699762338dSopenharmony_ci * @tc.level  : Level 1
709762338dSopenharmony_ci */
719762338dSopenharmony_ciHWTEST_F(PreadvApiTest, PreadvReadSuccess_0001, Function | MediumTest | Level1)
729762338dSopenharmony_ci{
739762338dSopenharmony_ci    ssize_t size;
749762338dSopenharmony_ci    int midLen = TEST_LEN / 2;
759762338dSopenharmony_ci    char buffer[MAX_LEN] = { 0 };
769762338dSopenharmony_ci    struct iovec iov[2] = {
779762338dSopenharmony_ci        {
789762338dSopenharmony_ci            .iov_base = buffer,
799762338dSopenharmony_ci            .iov_len = TEST_LEN,
809762338dSopenharmony_ci        }, {
819762338dSopenharmony_ci            .iov_base = &buffer[TEST_LEN],
829762338dSopenharmony_ci            .iov_len = TEST_LEN,
839762338dSopenharmony_ci        }
849762338dSopenharmony_ci    };
859762338dSopenharmony_ci
869762338dSopenharmony_ci    int fd = open(TEST_FILE, O_RDONLY);
879762338dSopenharmony_ci
889762338dSopenharmony_ci    // preadv from file start
899762338dSopenharmony_ci    size = preadv(fd, iov, 2, 0);
909762338dSopenharmony_ci    EXPECT_EQ(size, TEST_LEN);
919762338dSopenharmony_ci    EXPECT_STREQ(static_cast<char *>(iov[0].iov_base), TEST_DATA);
929762338dSopenharmony_ci
939762338dSopenharmony_ci    // preadv from file middle
949762338dSopenharmony_ci    memset_s(buffer, sizeof(buffer), 0, sizeof(buffer));
959762338dSopenharmony_ci    size = preadv(fd, iov, 1, midLen);
969762338dSopenharmony_ci    EXPECT_EQ(size, TEST_LEN - midLen);
979762338dSopenharmony_ci    EXPECT_STREQ(static_cast<char *>(iov[0].iov_base), &TEST_DATA[midLen]);
989762338dSopenharmony_ci
999762338dSopenharmony_ci    // preadv from file end
1009762338dSopenharmony_ci    memset_s(buffer, sizeof(buffer), 0, sizeof(buffer));
1019762338dSopenharmony_ci    size = preadv(fd, iov, 1, TEST_LEN);
1029762338dSopenharmony_ci    EXPECT_EQ(size, 0);
1039762338dSopenharmony_ci
1049762338dSopenharmony_ci    close(fd);
1059762338dSopenharmony_ci}
1069762338dSopenharmony_ci
1079762338dSopenharmony_ci/*
1089762338dSopenharmony_ci * @tc.number : SUB_KERNEL_SYSCALL_PREADV_0200
1099762338dSopenharmony_ci * @tc.name   : PreadvInvalidFdFailed_0002
1109762338dSopenharmony_ci * @tc.desc   : Test using an invalid file descriptor with preadv.
1119762338dSopenharmony_ci * @tc.size   : MediumTest
1129762338dSopenharmony_ci * @tc.type   : Function
1139762338dSopenharmony_ci * @tc.level  : Level 2
1149762338dSopenharmony_ci */
1159762338dSopenharmony_ciHWTEST_F(PreadvApiTest, PreadvInvalidFdFailed_0002, Function | MediumTest | Level2)
1169762338dSopenharmony_ci{
1179762338dSopenharmony_ci    ssize_t size;
1189762338dSopenharmony_ci    char buffer[MAX_LEN] = { 0 };
1199762338dSopenharmony_ci    struct iovec iov = {
1209762338dSopenharmony_ci        .iov_base = buffer,
1219762338dSopenharmony_ci        .iov_len = sizeof(buffer),
1229762338dSopenharmony_ci    };
1239762338dSopenharmony_ci    errno = 0;
1249762338dSopenharmony_ci    size = preadv(-1, &iov, 1, 0);
1259762338dSopenharmony_ci    EXPECT_EQ(size, -1);
1269762338dSopenharmony_ci    EXPECT_EQ(errno, EBADF);
1279762338dSopenharmony_ci}
1289762338dSopenharmony_ci
1299762338dSopenharmony_ci/*
1309762338dSopenharmony_ci * @tc.number : SUB_KERNEL_SYSCALL_PREADV_0300
1319762338dSopenharmony_ci * @tc.name   : PreadvReadFromEmptyFileSuccess_0003
1329762338dSopenharmony_ci * @tc.desc   : Test reading from an empty file with preadv.
1339762338dSopenharmony_ci * @tc.size   : MediumTest
1349762338dSopenharmony_ci * @tc.type   : Function
1359762338dSopenharmony_ci * @tc.level  : Level 1
1369762338dSopenharmony_ci */
1379762338dSopenharmony_ciHWTEST_F(PreadvApiTest, PreadvReadFromEmptyFileSuccess_0003, Function | MediumTest | Level1)
1389762338dSopenharmony_ci{
1399762338dSopenharmony_ci    ssize_t size;
1409762338dSopenharmony_ci    char buffer[MAX_LEN] = { 0 };
1419762338dSopenharmony_ci    struct iovec iov = {
1429762338dSopenharmony_ci        .iov_base = buffer,
1439762338dSopenharmony_ci        .iov_len = sizeof(buffer),
1449762338dSopenharmony_ci    };
1459762338dSopenharmony_ci    int fd = open(EMPTY_FILE, O_CREAT | O_RDWR, 0644);
1469762338dSopenharmony_ci    size = preadv(fd, &iov, 1, 0);
1479762338dSopenharmony_ci    EXPECT_EQ(size, 0);
1489762338dSopenharmony_ci    close(fd);
1499762338dSopenharmony_ci    unlink(EMPTY_FILE);
1509762338dSopenharmony_ci}
1519762338dSopenharmony_ci
1529762338dSopenharmony_ci/*
1539762338dSopenharmony_ci * @tc.number : SUB_KERNEL_SYSCALL_PREADV_0400
1549762338dSopenharmony_ci * @tc.name   : PreadvReadContinuousSuccess_0004
1559762338dSopenharmony_ci * @tc.desc   : Test reading from a file with preadv continuously.
1569762338dSopenharmony_ci * @tc.size   : MediumTest
1579762338dSopenharmony_ci * @tc.type   : Function
1589762338dSopenharmony_ci * @tc.level  : Level 1
1599762338dSopenharmony_ci */
1609762338dSopenharmony_ciHWTEST_F(PreadvApiTest, PreadvReadContinuousSuccess_0004, Function | MediumTest | Level1)
1619762338dSopenharmony_ci{
1629762338dSopenharmony_ci    ssize_t size;
1639762338dSopenharmony_ci    const char *context = "this is a test file.";
1649762338dSopenharmony_ci    char buffer[MAX_LEN] = { 0 };
1659762338dSopenharmony_ci    struct iovec iov[2] = {
1669762338dSopenharmony_ci        {
1679762338dSopenharmony_ci            .iov_base = buffer,
1689762338dSopenharmony_ci            .iov_len = TEST_LEN,
1699762338dSopenharmony_ci        }, {
1709762338dSopenharmony_ci            .iov_base = &buffer[MAX_LEN / 2],
1719762338dSopenharmony_ci            .iov_len = strlen(context),
1729762338dSopenharmony_ci        }
1739762338dSopenharmony_ci    };
1749762338dSopenharmony_ci
1759762338dSopenharmony_ci    int fd = open(TEST_FILE, O_RDWR | O_APPEND, 0644);
1769762338dSopenharmony_ci    write(fd, context, strlen(context));
1779762338dSopenharmony_ci
1789762338dSopenharmony_ci    // preadv context to 2 iovec
1799762338dSopenharmony_ci    size = preadv(fd, iov, 2, 0);
1809762338dSopenharmony_ci    EXPECT_EQ(size, TEST_LEN + strlen(context));
1819762338dSopenharmony_ci    EXPECT_STREQ(static_cast<char *>(iov[0].iov_base), TEST_DATA);
1829762338dSopenharmony_ci    EXPECT_STREQ(static_cast<char *>(iov[1].iov_base), context);
1839762338dSopenharmony_ci
1849762338dSopenharmony_ci    close(fd);
1859762338dSopenharmony_ci}
1869762338dSopenharmony_ci
1879762338dSopenharmony_ci/*
1889762338dSopenharmony_ci * @tc.number : SUB_KERNEL_SYSCALL_PREADV2_0500
1899762338dSopenharmony_ci * @tc.name   : Preadv2ReadSuccess_0002
1909762338dSopenharmony_ci * @tc.desc   : preadv2 read file success.
1919762338dSopenharmony_ci * @tc.size   : MediumTest
1929762338dSopenharmony_ci * @tc.type   : Function
1939762338dSopenharmony_ci * @tc.level  : Level 1
1949762338dSopenharmony_ci */
1959762338dSopenharmony_ciHWTEST_F(PreadvApiTest, Preadv2ReadSuccess_0002, Function | MediumTest | Level1)
1969762338dSopenharmony_ci{
1979762338dSopenharmony_ci    ssize_t size;
1989762338dSopenharmony_ci    int midLen = TEST_LEN / 2;
1999762338dSopenharmony_ci    char buffer[MAX_LEN] = { 0 };
2009762338dSopenharmony_ci    struct iovec iov[2] = {
2019762338dSopenharmony_ci        {
2029762338dSopenharmony_ci            .iov_base = buffer,
2039762338dSopenharmony_ci            .iov_len = TEST_LEN,
2049762338dSopenharmony_ci        }, {
2059762338dSopenharmony_ci            .iov_base = &buffer[TEST_LEN],
2069762338dSopenharmony_ci            .iov_len = TEST_LEN,
2079762338dSopenharmony_ci        }
2089762338dSopenharmony_ci    };
2099762338dSopenharmony_ci
2109762338dSopenharmony_ci    int fd = open(TEST_FILE, O_RDONLY);
2119762338dSopenharmony_ci
2129762338dSopenharmony_ci    // preadv from file start
2139762338dSopenharmony_ci    size = preadv(fd, iov, 2, 0);
2149762338dSopenharmony_ci    EXPECT_EQ(size, TEST_LEN);
2159762338dSopenharmony_ci    EXPECT_STREQ(static_cast<char *>(iov[0].iov_base), TEST_DATA);
2169762338dSopenharmony_ci
2179762338dSopenharmony_ci    // preadv from file middle
2189762338dSopenharmony_ci    memset_s(buffer, sizeof(buffer), 0, sizeof(buffer));
2199762338dSopenharmony_ci    size = preadv2(fd, iov, 1, midLen, 0);
2209762338dSopenharmony_ci    EXPECT_EQ(size, TEST_LEN - midLen);
2219762338dSopenharmony_ci    EXPECT_STREQ(static_cast<char *>(iov[0].iov_base), &TEST_DATA[midLen]);
2229762338dSopenharmony_ci
2239762338dSopenharmony_ci    // preadv from file end
2249762338dSopenharmony_ci    memset_s(buffer, sizeof(buffer), 0, sizeof(buffer));
2259762338dSopenharmony_ci    size = preadv2(fd, iov, 1, TEST_LEN, 0);
2269762338dSopenharmony_ci    EXPECT_EQ(size, 0);
2279762338dSopenharmony_ci
2289762338dSopenharmony_ci    close(fd);
2299762338dSopenharmony_ci}
230