fix: 增加内核epoll系统调用

【背景】增加内核epoll系统调用适配musl接口并增加测试用例

【修改方案】
1, 删除musl仓的porting
2,内核实现epoll对应接口及注册系统调用
3,testsuit目录添加对应的用例

【影响】
对现有的产品编译不会有影响。

re #I4FXPT
Signed-off-by: wangchen <253227059@qq.com>

Change-Id: Ia97ea49067aa1ff47b8c1c83675ac75e75d25955
This commit is contained in:
lnlan 2021-10-20 03:46:41 +00:00
parent 38163de8f3
commit 2251b8a2d1
14 changed files with 849 additions and 0 deletions

View File

@ -96,6 +96,7 @@ extern "C" {
#define CONFIG_NEXPANED_DESCRIPTORS (CONFIG_NTIME_DESCRIPTORS + CONFIG_NQUEUE_DESCRIPTORS) #define CONFIG_NEXPANED_DESCRIPTORS (CONFIG_NTIME_DESCRIPTORS + CONFIG_NQUEUE_DESCRIPTORS)
#define TIMER_FD_OFFSET FD_SETSIZE #define TIMER_FD_OFFSET FD_SETSIZE
#define MQUEUE_FD_OFFSET (FD_SETSIZE + CONFIG_NTIME_DESCRIPTORS) #define MQUEUE_FD_OFFSET (FD_SETSIZE + CONFIG_NTIME_DESCRIPTORS)
#define EPOLL_FD_OFFSET (FD_SETSIZE + CONFIG_NTIME_DESCRIPTORS + CONFIG_NQUEUE_DESCRIPTORS)
/* net configure */ /* net configure */
@ -129,6 +130,8 @@ extern "C" {
#define VFS_USING_WORKDIR // enable current working directory #define VFS_USING_WORKDIR // enable current working directory
#define CONFIG_EPOLL_DESCRIPTORS 32
/* permission configure */ /* permission configure */
#define DEFAULT_DIR_MODE 0777 #define DEFAULT_DIR_MODE 0777
#define DEFAULT_FILE_MODE 0666 #define DEFAULT_FILE_MODE 0666

View File

@ -70,6 +70,8 @@ static void FillFdInfo(struct SeqBuf *seqBuf, struct filelist *fileList, unsigne
name = "(timer)"; name = "(timer)";
} else if (sysFd < (MQUEUE_FD_OFFSET + CONFIG_NQUEUE_DESCRIPTORS)) { } else if (sysFd < (MQUEUE_FD_OFFSET + CONFIG_NQUEUE_DESCRIPTORS)) {
name = "(mqueue)"; name = "(mqueue)";
} else if (sysFd < (EPOLL_FD_OFFSET + CONFIG_EPOLL_DESCRIPTORS)) {
name = "(epoll)";
} else { } else {
name = "(unknown)"; name = "(unknown)";
} }

View File

@ -104,6 +104,7 @@ kernel_module(module_name) {
"$LITEOSTHIRDPARTY/NuttX/fs/vfs/fs_truncate64.c", "$LITEOSTHIRDPARTY/NuttX/fs/vfs/fs_truncate64.c",
"$LITEOSTHIRDPARTY/NuttX/fs/vfs/fs_unlink.c", "$LITEOSTHIRDPARTY/NuttX/fs/vfs/fs_unlink.c",
"$LITEOSTHIRDPARTY/NuttX/fs/vfs/fs_write.c", "$LITEOSTHIRDPARTY/NuttX/fs/vfs/fs_write.c",
"$LITEOSTOPDIR/fs/vfs/epoll/fs_epoll.c",
"$LITEOSTOPDIR/fs/vfs/mount.c", "$LITEOSTOPDIR/fs/vfs/mount.c",
"$LITEOSTOPDIR/fs/vfs/path_cache.c", "$LITEOSTOPDIR/fs/vfs/path_cache.c",
"$LITEOSTOPDIR/fs/vfs/vnode.c", "$LITEOSTOPDIR/fs/vfs/vnode.c",

View File

@ -35,6 +35,7 @@ $(LITEOSTOPDIR)/fs/vfs/mount.c \
$(LITEOSTOPDIR)/fs/vfs/vnode.c \ $(LITEOSTOPDIR)/fs/vfs/vnode.c \
$(LITEOSTOPDIR)/fs/vfs/path_cache.c \ $(LITEOSTOPDIR)/fs/vfs/path_cache.c \
$(LITEOSTOPDIR)/fs/vfs/vnode_hash.c \ $(LITEOSTOPDIR)/fs/vfs/vnode_hash.c \
$(LITEOSTOPDIR)/fs/vfs/epoll/fs_epoll.c \
$(LITEOSTHIRDPARTY)/NuttX/fs/vfs/fs_close.c \ $(LITEOSTHIRDPARTY)/NuttX/fs/vfs/fs_close.c \
$(LITEOSTHIRDPARTY)/NuttX/fs/vfs/fs_dup2.c \ $(LITEOSTHIRDPARTY)/NuttX/fs/vfs/fs_dup2.c \
$(LITEOSTHIRDPARTY)/NuttX/fs/vfs/fs_dup.c \ $(LITEOSTHIRDPARTY)/NuttX/fs/vfs/fs_dup.c \

344
fs/vfs/epoll/fs_epoll.c Executable file
View File

@ -0,0 +1,344 @@
/*
* Copyright (c) 2021-2021 Huawei Device Co., Ltd. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without modification,
* are permitted provided that the following conditions are met:
*
* 1. Redistributions of source code must retain the above copyright notice, this list of
* conditions and the following disclaimer.
*
* 2. Redistributions in binary form must reproduce the above copyright notice, this list
* of conditions and the following disclaimer in the documentation and/or other materials
* provided with the distribution.
*
* 3. Neither the name of the copyright holder nor the names of its contributors may be used
* to endorse or promote products derived from this software without specific prior written
* permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
* OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
* OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
* ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "epoll.h"
#include <stdint.h>
#include <poll.h>
#include <errno.h>
#include <string.h>
#include "pthread.h"
/* 100, the number of fd one epollfd can control */
#define EPOLL_DEFAULT_SIZE 100
/* Internal data, used to manage each epoll fd */
struct epoll_head {
int size;
int nodeCount;
struct epoll_event *evs;
};
STATIC pthread_mutex_t g_epollMutex = PTHREAD_RECURSIVE_MUTEX_INITIALIZER_NP;
#ifndef MAX_EPOLL_FD
#define MAX_EPOLL_FD CONFIG_EPOLL_DESCRIPTORS
#endif
/* Record the kernel fd of epoll */
STATIC fd_set g_epollFdSet;
/* Record the private data of epoll */
STATIC struct epoll_head *g_epPrivBuf[MAX_EPOLL_FD];
/**
* Alloc sysFd, storage epoll private data, set using bit.
*
* @param maxfdp: Maximum allowed application of sysFd.
* @param head: Private data.
* @return the index of the new fd; -1 on error
*/
static int EpollAllocSysFd(int maxfdp, struct epoll_head *head)
{
int i;
fd_set *fdset = &g_epollFdSet;
for (i = 0; i < maxfdp; i++) {
if (fdset && !(FD_ISSET(i + EPOLL_FD_OFFSET, fdset))) {
FD_SET(i + EPOLL_FD_OFFSET, fdset);
if (!g_epPrivBuf[i]) {
g_epPrivBuf[i] = head;
return i + EPOLL_FD_OFFSET;
}
}
}
set_errno(EMFILE);
return -1;
}
/**
* free sysFd, delete epoll private data, clear using bit.
*
* @param fd: epoll fd.
* @return 0 or -1
*/
static int EpollFreeSysFd(int fd)
{
int id = fd - EPOLL_FD_OFFSET;
if ((id < 0) || (id >= MAX_EPOLL_FD)) {
set_errno(EMFILE);
return -1;
}
fd_set *fdset = &g_epollFdSet;
if (fdset && FD_ISSET(fd, fdset)) {
FD_CLR(id, fdset);
g_epPrivBuf[id] = NULL;
}
return 0;
}
/**
* get private data by epoll fd
*
* @param fd: epoll fd.
* @return point to epoll_head
*/
static struct epoll_head *EpollGetDataBuff(int fd)
{
int id = fd - EPOLL_FD_OFFSET;
if ((id < 0) || (id >= MAX_EPOLL_FD)) {
return NULL;
}
return g_epPrivBuf[id];
}
/**
* when do EPOLL_CTL_ADD, need check if fd exist
*
* @param epHead: epoll control head, find by epoll id .
* @param fd: ctl add fd.
* @return 0 or -1
*/
static int CheckFdExist(struct epoll_head *epHead, int fd)
{
int i;
for (i = 0; i < epHead->nodeCount; i++) {
if (epHead->evs[i].data.fd == fd) {
return -1;
}
}
return 0;
}
/**
* close epoll
*
* @param epHead: epoll control head.
* @return void
*/
static VOID DoEpollClose(struct epoll_head *epHead)
{
if (epHead != NULL) {
if (epHead->evs != NULL) {
free(epHead->evs);
}
free(epHead);
}
return ;
}
/**
* epoll_create,
* The simple version of epoll does not use red-black trees,
* so when fd is normal value (greater than 0),
* actually allocated epoll can manage num of EPOLL_DEFAULT_SIZE
*
* @param size: not actually used
* @return epoll fd
*/
int epoll_create(int size)
{
int fd = -1;
if (size <= 0) {
set_errno(EINVAL);
return fd;
}
struct epoll_head *epHead = (struct epoll_head *)malloc(sizeof(struct epoll_head));
if (epHead == NULL) {
set_errno(ENOMEM);
return fd;
}
/* actually allocated epoll can manage num is EPOLL_DEFAULT_SIZE */
epHead->size = EPOLL_DEFAULT_SIZE;
epHead->nodeCount = 0;
epHead->evs = malloc(sizeof(struct epoll_event) * EPOLL_DEFAULT_SIZE);
if (epHead->evs == NULL) {
free(epHead);
set_errno(ENOMEM);
return fd;
}
/* fd set, get sysfd, for close */
(VOID)pthread_mutex_lock(&g_epollMutex);
fd = EpollAllocSysFd(MAX_EPOLL_FD, epHead);
if (fd == -1) {
(VOID)pthread_mutex_unlock(&g_epollMutex);
DoEpollClose(epHead);
set_errno(EMFILE);
return fd;
}
(VOID)pthread_mutex_unlock(&g_epollMutex);
return fd;
}
/**
* epoll_close,
* called by close
* @param epfd: epoll fd
* @return 0 or -1
*/
int epoll_close(int epfd)
{
struct epoll_head *epHead = NULL;
epHead = EpollGetDataBuff(epfd);
if (epHead == NULL) {
set_errno(EBADF);
return -1;
}
DoEpollClose(epHead);
return EpollFreeSysFd(epfd);
}
int epoll_ctl(int epfd, int op, int fd, struct epoll_event *ev)
{
struct epoll_head *epHead = NULL;
int i;
int ret = -1;
epHead = EpollGetDataBuff(epfd);
if (epHead== NULL) {
set_errno(EBADF);
return ret;
}
switch (op) {
case EPOLL_CTL_ADD:
ret = CheckFdExist(epHead, fd);
if (ret == -1) {
set_errno(EEXIST);
return -1;
}
if (epHead->nodeCount == EPOLL_DEFAULT_SIZE) {
set_errno(ENOMEM);
return -1;
}
epHead->evs[epHead->nodeCount].events = ev->events | POLLERR | POLLHUP;
epHead->evs[epHead->nodeCount].data.fd = fd;
epHead->nodeCount++;
return 0;
case EPOLL_CTL_DEL:
for (i = 0; i < epHead->nodeCount; i++) {
if (epHead->evs[i].data.fd != fd) {
continue;
}
if (i != epHead->nodeCount - 1) {
memmove_s(&epHead->evs[i], epHead->nodeCount - i, &epHead->evs[i + 1],
epHead->nodeCount - i);
}
epHead->nodeCount--;
return 0;
}
set_errno(ENOENT);
return -1;
case EPOLL_CTL_MOD:
for (i = 0; i < epHead->nodeCount; i++) {
if (epHead->evs[i].data.fd == fd) {
epHead->evs[i].events = ev->events | POLLERR | POLLHUP;
return 0;
}
}
set_errno(ENOENT);
return -1;
default:
set_errno(EINVAL);
return -1;
}
}
int epoll_wait(int epfd, FAR struct epoll_event *evs, int maxevents, int timeout)
{
struct epoll_head *epHead = NULL;
int ret;
int counter;
int i;
struct pollfd *pFd = NULL;
int pollSize;
epHead = EpollGetDataBuff(epfd);
if (epHead== NULL) {
set_errno(EBADF);
return -1;
}
if (maxevents <= 0) {
set_errno(EINVAL);
return -1;
}
if (maxevents > epHead->nodeCount) {
pollSize = epHead->nodeCount;
} else {
pollSize = maxevents;
}
pFd = malloc(sizeof(struct pollfd) * pollSize);
if (pFd == NULL) {
set_errno(EINVAL);
return -1;
}
for (i = 0; i < epHead->nodeCount; i++) {
pFd[i].fd = epHead->evs[i].data.fd;
pFd[i].events = (short)epHead->evs[i].events;
}
ret = poll(pFd, pollSize, timeout);
if (ret <= 0) {
return 0;
}
for (i = 0, counter = 0; i < ret && counter < pollSize; counter++) {
if (pFd[counter].revents != 0) {
evs[i].data.fd = pFd[counter].fd;
evs[i].events = pFd[counter].revents;
i++;
}
}
return i;
}

84
fs/vfs/include/epoll.h Executable file
View File

@ -0,0 +1,84 @@
/*
* Copyright (c) 2021-2021 Huawei Device Co., Ltd. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without modification,
* are permitted provided that the following conditions are met:
*
* 1. Redistributions of source code must retain the above copyright notice, this list of
* conditions and the following disclaimer.
*
* 2. Redistributions in binary form must reproduce the above copyright notice, this list
* of conditions and the following disclaimer in the documentation and/or other materials
* provided with the distribution.
*
* 3. Neither the name of the copyright holder nor the names of its contributors may be used
* to endorse or promote products derived from this software without specific prior written
* permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
* OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
* OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
* ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#ifndef _FS_EPOLL_H_
#define _FS_EPOLL_H_
#include "los_typedef.h"
#ifdef __cplusplus
extern "C" {
#endif
#ifndef FAR
#define FAR
#endif
#define EPOLL_CLOEXEC O_CLOEXEC
#define EPOLL_NONBLOCK O_NONBLOCK
#define EPOLLIN 0x001
#define EPOLLPRI 0x002
#define EPOLLOUT 0x004
#define EPOLLRDNORM 0x040
#define EPOLLNVAL 0x020
#define EPOLLRDBAND 0x080
#define EPOLLWRNORM 0x100
#define EPOLLWRBAND 0x200
#define EPOLLMSG 0x400
#define EPOLLERR 0x008
#define EPOLLHUP 0x010
#define EPOLL_CTL_ADD 1
#define EPOLL_CTL_DEL 2
#define EPOLL_CTL_MOD 3
typedef union epoll_data {
void *ptr;
int fd;
UINT32 u32;
UINT64 u64;
} epoll_data_t;
struct epoll_event {
UINT32 events;
epoll_data_t data;
};
int epoll_create(int size);
int epoll_close(int epfd);
int epoll_ctl(int epfd, int op, int fd, struct epoll_event *ev);
int epoll_wait(int epfd, FAR struct epoll_event *evs, int maxevents, int timeout);
#ifdef __cplusplus
}
#endif
#endif /* sys/epoll.h */

129
syscall/fs_syscall.c Normal file → Executable file
View File

@ -43,6 +43,7 @@
#include "sys/uio.h" #include "sys/uio.h"
#include "poll.h" #include "poll.h"
#include "sys/prctl.h" #include "sys/prctl.h"
#include "epoll.h"
#ifdef LOSCFG_KERNEL_DYNLOAD #ifdef LOSCFG_KERNEL_DYNLOAD
#include "los_exec_elf.h" #include "los_exec_elf.h"
#endif #endif
@ -2594,4 +2595,132 @@ int SysPselect6(int nfds, fd_set *readfds, fd_set *writefds, fd_set *exceptfds,
return ret; return ret;
} }
int SysEpollCreate(int size)
{
int ret;
int procFd;
if (size <= 0) {
return -EINVAL;
}
ret = epoll_create(size);
if (ret < 0) {
ret = -get_errno();
}
procFd = AllocAndAssocProcessFd((INTPTR)(ret), MIN_START_FD);
if (procFd == -1) {
epoll_close(ret);
return -EMFILE;
}
return procFd;
}
int SysEpollCreate1(int flags)
{
int ret;
int procFd;
ret = epoll_create(flags);
if (ret < 0) {
ret = -get_errno();
}
procFd = AllocAndAssocProcessFd((INTPTR)(ret), MIN_START_FD);
if (procFd == -1) {
epoll_close(ret);
return -EMFILE;
}
return procFd;
}
int SysEpollCtl(int epfd, int op, int fd, struct epoll_event *ev)
{
int ret;
CHECK_ASPACE(ev, sizeof(struct epoll_event));
CPY_FROM_USER(ev);
fd = GetAssociatedSystemFd(fd);
epfd = GetAssociatedSystemFd(epfd);
if ((fd < 0) || (epfd < 0)) {
ret = -EBADF;
goto OUT;
}
ret = epoll_ctl(epfd, op, fd, ev);
if (ret < 0) {
ret =-EBADF;
goto OUT;
}
CPY_TO_USER(ev);
OUT:
return (ret == -1)? -get_errno():ret;
}
int SysEpollWait(int epfd, struct epoll_event *evs, int maxevents, int timeout)
{
int ret = 0;
CHECK_ASPACE(evs, sizeof(struct epoll_event));
CPY_FROM_USER(evs);
epfd = GetAssociatedSystemFd(epfd);
if (epfd < 0) {
ret = -EBADF;
goto OUT;
}
ret = epoll_wait(epfd, evs, maxevents, timeout);
if (ret < 0) {
ret = -get_errno();
}
CPY_TO_USER(evs);
OUT:
return (ret == -1) ? -get_errno() : ret;
}
int SysEpollPwait(int epfd, struct epoll_event *evs, int maxevents, int timeout, const sigset_t *mask)
{
sigset_t_l origMask;
sigset_t_l setl;
int ret = 0;
CHECK_ASPACE(mask, sizeof(sigset_t));
if (mask != NULL) {
ret = LOS_ArchCopyFromUser(&setl, mask, sizeof(sigset_t));
if (ret != 0) {
return -EFAULT;
}
}
CHECK_ASPACE(evs, sizeof(struct epoll_event));
CPY_FROM_USER(evs);
epfd = GetAssociatedSystemFd(epfd);
if (epfd < 0) {
ret = -EBADF;
goto OUT;
}
OsSigprocMask(SIG_SETMASK, &setl, &origMask);
ret = epoll_wait(epfd, evs, maxevents, timeout);
if (ret < 0) {
ret = -get_errno();
}
OsSigprocMask(SIG_SETMASK, &origMask, NULL);
CPY_TO_USER(evs);
OUT:
return (ret == -1) ? -get_errno() : ret;
}
#endif #endif

6
syscall/los_syscall.h Normal file → Executable file
View File

@ -60,6 +60,7 @@
#include "sys/socket.h" #include "sys/socket.h"
#include "dirent.h" #include "dirent.h"
#include "fs/file.h" #include "fs/file.h"
#include "epoll.h"
#endif #endif
#include <sys/wait.h> #include <sys/wait.h>
#include "sys/resource.h" #include "sys/resource.h"
@ -284,6 +285,11 @@ extern int SysPpoll(struct pollfd *fds, nfds_t nfds, const struct timespec *tmo_
extern int SysPrctl(int option, ...); extern int SysPrctl(int option, ...);
extern ssize_t SysPread64(int fd, void *buf, size_t nbytes, off64_t offset); extern ssize_t SysPread64(int fd, void *buf, size_t nbytes, off64_t offset);
extern ssize_t SysPwrite64(int fd, const void *buf, size_t nbytes, off64_t offset); extern ssize_t SysPwrite64(int fd, const void *buf, size_t nbytes, off64_t offset);
extern int SysEpollCreate(int size);
extern int SysEpollCreate1(int size);
extern int SysEpollCtl(int epfd, int op, int fd, struct epoll_event *ev);
extern int SysEpollWait(int epfd, struct epoll_event *evs, int maxevents, int timeout);
extern int SysEpollPwait(int epfd, struct epoll_event *evs, int maxevents, int timeout, const sigset_t *mask);
extern char *SysGetcwd(char *buf, size_t n); extern char *SysGetcwd(char *buf, size_t n);
extern ssize_t SysSendFile(int outfd, int infd, off_t *offset, size_t count); extern ssize_t SysSendFile(int outfd, int infd, off_t *offset, size_t count);
extern int SysTruncate(const char *path, off_t length); extern int SysTruncate(const char *path, off_t length);

5
syscall/syscall_lookup.h Normal file → Executable file
View File

@ -91,6 +91,11 @@ SYSCALL_HAND_DEF(__NR_ppoll, SysPpoll, int, ARG_NUM_5)
SYSCALL_HAND_DEF(__NR_prctl, SysPrctl, int, ARG_NUM_7) SYSCALL_HAND_DEF(__NR_prctl, SysPrctl, int, ARG_NUM_7)
SYSCALL_HAND_DEF(__NR_pread64, SysPread64, ssize_t, ARG_NUM_7) SYSCALL_HAND_DEF(__NR_pread64, SysPread64, ssize_t, ARG_NUM_7)
SYSCALL_HAND_DEF(__NR_pwrite64, SysPwrite64, ssize_t, ARG_NUM_7) SYSCALL_HAND_DEF(__NR_pwrite64, SysPwrite64, ssize_t, ARG_NUM_7)
SYSCALL_HAND_DEF(__NR_epoll_create, SysEpollCreate, int, ARG_NUM_1)
SYSCALL_HAND_DEF(__NR_epoll_create1, SysEpollCreate1, int, ARG_NUM_1)
SYSCALL_HAND_DEF(__NR_epoll_ctl, SysEpollCtl, int, ARG_NUM_4)
SYSCALL_HAND_DEF(__NR_epoll_wait, SysEpollWait, int, ARG_NUM_4)
SYSCALL_HAND_DEF(__NR_epoll_pwait, SysEpollPwait, int, ARG_NUM_5)
SYSCALL_HAND_DEF(__NR_getcwd, SysGetcwd, char *, ARG_NUM_2) SYSCALL_HAND_DEF(__NR_getcwd, SysGetcwd, char *, ARG_NUM_2)
SYSCALL_HAND_DEF(__NR_sendfile, SysSendFile, ssize_t, ARG_NUM_4) SYSCALL_HAND_DEF(__NR_sendfile, SysSendFile, ssize_t, ARG_NUM_4)
SYSCALL_HAND_DEF(__NR_truncate64, SysTruncate64, int, ARG_NUM_7) SYSCALL_HAND_DEF(__NR_truncate64, SysTruncate64, int, ARG_NUM_7)

2
testsuites/unittest/IO/BUILD.gn Normal file → Executable file
View File

@ -88,6 +88,8 @@ sources_full = [
"full/IO_test_ppoll_003.cpp", "full/IO_test_ppoll_003.cpp",
"full/IO_test_pselect_001.cpp", "full/IO_test_pselect_001.cpp",
"full/IO_test_pselect_002.cpp", "full/IO_test_pselect_002.cpp",
"full/IO_test_epoll_001.cpp",
"full/IO_test_epoll_002.cpp",
] ]
if (LOSCFG_USER_TEST_LEVEL >= TEST_LEVEL_LOW) { if (LOSCFG_USER_TEST_LEVEL >= TEST_LEVEL_LOW) {

View File

@ -55,6 +55,7 @@
#include "pwd.h" #include "pwd.h"
#include "sys/uio.h" #include "sys/uio.h"
#include "syslog.h" #include "syslog.h"
#include "sys/epoll.h"
extern int CloseRmAllFile(int fd[], char filePathName[][50], int cnt); extern int CloseRmAllFile(int fd[], char filePathName[][50], int cnt);
extern char *g_ioTestPath; extern char *g_ioTestPath;
@ -134,5 +135,7 @@ extern VOID IO_TEST_STRFMON_L_002(VOID);
extern VOID IO_TEST_PPOLL_001(VOID); extern VOID IO_TEST_PPOLL_001(VOID);
extern VOID IO_TEST_PPOLL_002(VOID); extern VOID IO_TEST_PPOLL_002(VOID);
extern VOID IO_TEST_PPOLL_003(VOID); extern VOID IO_TEST_PPOLL_003(VOID);
extern VOID IO_TEST_EPOLL_001(VOID);
extern VOID IO_TEST_EPOLL_002(VOID);
#endif #endif

View File

@ -0,0 +1,107 @@
/*
* Copyright (c) 2013-2019 Huawei Technologies Co., Ltd. All rights reserved.
* Copyright (c) 2020-2021 Huawei Device Co., Ltd. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without modification,
* are permitted provided that the following conditions are met:
*
* 1. Redistributions of source code must retain the above copyright notice, this list of
* conditions and the following disclaimer.
*
* 2. Redistributions in binary form must reproduce the above copyright notice, this list
* of conditions and the following disclaimer in the documentation and/or other materials
* provided with the distribution.
*
* 3. Neither the name of the copyright holder nor the names of its contributors may be used
* to endorse or promote products derived from this software without specific prior written
* permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
* OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
* OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
* ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "It_test_IO.h"
#include <sys/epoll.h>
#include "signal.h"
#include "pthread.h"
#include <sys/time.h>
#include <sys/types.h>
#include <unistd.h>
static UINT32 testcase(VOID)
{
fd_set rfds;
int retval;
pid_t pid;
int pipeFd[2]; /* 2, pipe id num */
char buffer[40]; /* 40, buffer size */
int i = 0;
int status;
int epFd;
sigset_t mask;
struct epoll_event ev;
struct epoll_event evWait[2]; /* 2, evs num */
retval = pipe(pipeFd);
ICUNIT_GOTO_EQUAL(retval, 0, retval, OUT);
/* Watch fd to see when it has input. */
FD_ZERO(&rfds);
FD_SET(pipeFd[0], &rfds);
/* Wait up to three seconds. */
epFd = epoll_create1(100); /* 100, cretae input, */
ICUNIT_GOTO_NOT_EQUAL(epFd, -1, epFd, OUT);
ev.events = EPOLLIN | EPOLLOUT | EPOLLRDNORM | EPOLLWRNORM;
retval = epoll_ctl(epFd, EPOLL_CTL_ADD, pipeFd[0], &ev);
ICUNIT_GOTO_NOT_EQUAL(retval, -1, retval, OUT);
pid = fork();
if (pid == 0) {
close(pipeFd[1]);
memset(evWait, 0, sizeof(struct epoll_event) * 2); /* 2, evs num */
retval = epoll_wait(epFd, evWait, 2, 3000); /* 2, num of wait fd. 3000, wait time */
close(pipeFd[0]);
if (retval) {
exit(LOS_OK);
} else {
exit(LOS_NOK);
}
} else {
sleep(1);
close(pipeFd[0]);
retval = write(pipeFd[1], "0123456789012345678901234567890123456789", 40); /* write 40 bytes to stdin(fd 0) */
ICUNIT_GOTO_EQUAL(retval, 40, retval, OUT);
close(pipeFd[1]);
wait(&status);
status = WEXITSTATUS(status);
ICUNIT_ASSERT_EQUAL(status, LOS_OK, status);
}
return LOS_OK;
OUT:
close(pipeFd[0]);
close(pipeFd[1]);
close(epFd);
return LOS_NOK;
}
VOID IO_TEST_EPOLL_001(VOID)
{
TEST_ADD_CASE(__FUNCTION__, testcase, TEST_LIB, TEST_LIBC, TEST_LEVEL1, TEST_FUNCTION);
}

View File

@ -0,0 +1,139 @@
/*
* Copyright (c) 2013-2019 Huawei Technologies Co., Ltd. All rights reserved.
* Copyright (c) 2020-2021 Huawei Device Co., Ltd. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without modification,
* are permitted provided that the following conditions are met:
*
* 1. Redistributions of source code must retain the above copyright notice, this list of
* conditions and the following disclaimer.
*
* 2. Redistributions in binary form must reproduce the above copyright notice, this list
* of conditions and the following disclaimer in the documentation and/or other materials
* provided with the distribution.
*
* 3. Neither the name of the copyright holder nor the names of its contributors may be used
* to endorse or promote products derived from this software without specific prior written
* permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
* OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
* OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
* ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "It_test_IO.h"
#include <sys/epoll.h>
#include "signal.h"
#include "pthread.h"
#include <sys/time.h>
#include <sys/types.h>
#include <unistd.h>
static void SigPrint(int sig)
{
return;
}
static UINT32 testcase(VOID)
{
fd_set rfds;
struct timespec tv;
int retval;
pid_t pid;
int pipeFd[2]; /* 2, pipe id num */
char buffer[40]; /* 40, buffer size */
int i = 0;
int status;
int epFd;
sigset_t mask;
void (*retSig)(int);
struct epoll_event ev;
struct epoll_event evWait[2];
retSig = signal(SIGALRM, SigPrint);
ICUNIT_ASSERT_NOT_EQUAL(retSig, NULL, retSig);
retSig = signal(SIGUSR1, SigPrint);
ICUNIT_ASSERT_NOT_EQUAL(retSig, NULL, retSig);
retval = sigemptyset(&mask);
ICUNIT_ASSERT_EQUAL(retval, 0, retval);
retval = sigaddset(&mask, SIGALRM);
ICUNIT_ASSERT_EQUAL(retval, 0, retval);
retval = sigaddset(&mask, SIGUSR1);
ICUNIT_ASSERT_EQUAL(retval, 0, retval);
retval = pipe(pipeFd);
ICUNIT_GOTO_EQUAL(retval, 0, retval, OUT);
/* Watch fd to see when it has input. */
FD_ZERO(&rfds);
FD_SET(pipeFd[0], &rfds);
/* Wait up to three seconds. */
tv.tv_sec = 3; /* 3, wait timer, second */
tv.tv_nsec = 5; /* 5, wait timer, nano second */
epFd = epoll_create1(100); /* 100, cretae input, */
ICUNIT_GOTO_NOT_EQUAL(epFd, -1, epFd, OUT);
ev.events = EPOLLRDNORM;
retval = epoll_ctl(epFd, EPOLL_CTL_ADD, pipeFd[0], &ev);
ICUNIT_GOTO_NOT_EQUAL(retval, -1, retval, OUT);
pid = fork();
if (pid == 0) {
close(pipeFd[1]);
memset(evWait, 0, sizeof(struct epoll_event) * 2); /* 2, evs num */
evWait[0].data.fd = pipeFd[0];
retval = epoll_pwait(epFd, evWait, 2, 3000, &mask); /* 2, num of wait fd. 3000, wait time */
close(pipeFd[0]);
if (retval) {
exit(LOS_OK);
} else {
exit(LOS_NOK);
}
} else {
sleep(1);
close(pipeFd[0]);
retval = kill(pid, SIGALRM);
ICUNIT_ASSERT_EQUAL(retval, 0, retval);
retval = kill(pid, SIGUSR1);
ICUNIT_ASSERT_EQUAL(retval, 0, retval);
close(pipeFd[1]);
wait(&status);
status = WEXITSTATUS(status);
ICUNIT_ASSERT_EQUAL(status, LOS_OK, status);
}
return LOS_OK;
OUT:
close(pipeFd[0]);
close(pipeFd[1]);
close(epFd);
return LOS_NOK;
}
VOID IO_TEST_EPOLL_002(VOID)
{
TEST_ADD_CASE(__FUNCTION__, testcase, TEST_LIB, TEST_LIBC, TEST_LEVEL1, TEST_FUNCTION);
}

View File

@ -148,6 +148,29 @@ HWTEST_F(IoTest, IO_TEST_PPOLL_003, TestSize.Level0)
IO_TEST_PPOLL_003(); IO_TEST_PPOLL_003();
} }
/* *
* @tc.name: IO_TEST_EPOLL_001
* @tc.desc: function for IoTest
* @tc.type: FUNC
* @tc.require: AR000EEMQ9
*/
HWTEST_F(IoTest, IO_TEST_EPOLL_001, TestSize.Level0)
{
IO_TEST_EPOLL_001();
}
/* *
* @tc.name: IO_TEST_EPOLL_002
* @tc.desc: function for IoTest
* @tc.type: FUNC
* @tc.require: AR000EEMQ9
*/
HWTEST_F(IoTest, IO_TEST_EPOLL_002, TestSize.Level0)
{
IO_TEST_EPOLL_002();
}
/* * /* *
* @tc.name: IT_STDLIB_POLL_002 * @tc.name: IT_STDLIB_POLL_002
* @tc.desc: function for IoTest * @tc.desc: function for IoTest