10d163575Sopenharmony_ci/*
20d163575Sopenharmony_ci * Copyright (c) 2021-2022 Huawei Device Co., Ltd. All rights reserved.
30d163575Sopenharmony_ci *
40d163575Sopenharmony_ci * Redistribution and use in source and binary forms, with or without modification,
50d163575Sopenharmony_ci * are permitted provided that the following conditions are met:
60d163575Sopenharmony_ci *
70d163575Sopenharmony_ci * 1. Redistributions of source code must retain the above copyright notice, this list of
80d163575Sopenharmony_ci *    conditions and the following disclaimer.
90d163575Sopenharmony_ci *
100d163575Sopenharmony_ci * 2. Redistributions in binary form must reproduce the above copyright notice, this list
110d163575Sopenharmony_ci *    of conditions and the following disclaimer in the documentation and/or other materials
120d163575Sopenharmony_ci *    provided with the distribution.
130d163575Sopenharmony_ci *
140d163575Sopenharmony_ci * 3. Neither the name of the copyright holder nor the names of its contributors may be used
150d163575Sopenharmony_ci *    to endorse or promote products derived from this software without specific prior written
160d163575Sopenharmony_ci *    permission.
170d163575Sopenharmony_ci *
180d163575Sopenharmony_ci * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
190d163575Sopenharmony_ci * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
200d163575Sopenharmony_ci * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
210d163575Sopenharmony_ci * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
220d163575Sopenharmony_ci * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
230d163575Sopenharmony_ci * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
240d163575Sopenharmony_ci * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
250d163575Sopenharmony_ci * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
260d163575Sopenharmony_ci * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
270d163575Sopenharmony_ci * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
280d163575Sopenharmony_ci * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
290d163575Sopenharmony_ci */
300d163575Sopenharmony_ci
310d163575Sopenharmony_ci#include "epoll.h"
320d163575Sopenharmony_ci#include <stdint.h>
330d163575Sopenharmony_ci#include <poll.h>
340d163575Sopenharmony_ci#include <errno.h>
350d163575Sopenharmony_ci#include <string.h>
360d163575Sopenharmony_ci#include "pthread.h"
370d163575Sopenharmony_ci
380d163575Sopenharmony_ci/* 100, the number of fd one epollfd can control */
390d163575Sopenharmony_ci#define EPOLL_DEFAULT_SIZE 100
400d163575Sopenharmony_ci
410d163575Sopenharmony_ci/* Internal data, used to manage each epoll fd */
420d163575Sopenharmony_cistruct epoll_head {
430d163575Sopenharmony_ci    int size;
440d163575Sopenharmony_ci    int nodeCount;
450d163575Sopenharmony_ci    struct epoll_event *evs;
460d163575Sopenharmony_ci};
470d163575Sopenharmony_ci
480d163575Sopenharmony_ciSTATIC pthread_mutex_t g_epollMutex = PTHREAD_RECURSIVE_MUTEX_INITIALIZER_NP;
490d163575Sopenharmony_ci
500d163575Sopenharmony_ci#ifndef MAX_EPOLL_FD
510d163575Sopenharmony_ci#define MAX_EPOLL_FD CONFIG_EPOLL_DESCRIPTORS
520d163575Sopenharmony_ci#endif
530d163575Sopenharmony_ci
540d163575Sopenharmony_ci/* Record the kernel fd of epoll  */
550d163575Sopenharmony_ciSTATIC fd_set g_epollFdSet;
560d163575Sopenharmony_ci
570d163575Sopenharmony_ci/* Record the private data of epoll  */
580d163575Sopenharmony_ciSTATIC struct epoll_head *g_epPrivBuf[MAX_EPOLL_FD];
590d163575Sopenharmony_ci
600d163575Sopenharmony_ci/**
610d163575Sopenharmony_ci * Alloc sysFd, storage epoll private data, set using bit.
620d163575Sopenharmony_ci *
630d163575Sopenharmony_ci * @param maxfdp: Maximum allowed application of sysFd.
640d163575Sopenharmony_ci * @param head: Private data.
650d163575Sopenharmony_ci * @return the index of the new fd; -1 on error
660d163575Sopenharmony_ci */
670d163575Sopenharmony_cistatic int EpollAllocSysFd(int maxfdp, struct epoll_head *head)
680d163575Sopenharmony_ci{
690d163575Sopenharmony_ci    int i;
700d163575Sopenharmony_ci
710d163575Sopenharmony_ci    fd_set *fdset = &g_epollFdSet;
720d163575Sopenharmony_ci
730d163575Sopenharmony_ci    for (i = 0; i < maxfdp; i++) {
740d163575Sopenharmony_ci        if (fdset && !(FD_ISSET(i, fdset))) {
750d163575Sopenharmony_ci            FD_SET(i, fdset);
760d163575Sopenharmony_ci            if (!g_epPrivBuf[i]) {
770d163575Sopenharmony_ci                g_epPrivBuf[i] = head;
780d163575Sopenharmony_ci                return i + EPOLL_FD_OFFSET;
790d163575Sopenharmony_ci            }
800d163575Sopenharmony_ci        }
810d163575Sopenharmony_ci    }
820d163575Sopenharmony_ci
830d163575Sopenharmony_ci    set_errno(EMFILE);
840d163575Sopenharmony_ci    return -1;
850d163575Sopenharmony_ci}
860d163575Sopenharmony_ci
870d163575Sopenharmony_ci/**
880d163575Sopenharmony_ci * free sysFd, delete epoll private data, clear using bit.
890d163575Sopenharmony_ci *
900d163575Sopenharmony_ci * @param fd: epoll fd.
910d163575Sopenharmony_ci * @return 0 or -1
920d163575Sopenharmony_ci */
930d163575Sopenharmony_cistatic int EpollFreeSysFd(int fd)
940d163575Sopenharmony_ci{
950d163575Sopenharmony_ci    int efd = fd - EPOLL_FD_OFFSET;
960d163575Sopenharmony_ci
970d163575Sopenharmony_ci    if ((efd < 0) || (efd >= MAX_EPOLL_FD)) {
980d163575Sopenharmony_ci        set_errno(EMFILE);
990d163575Sopenharmony_ci        return -1;
1000d163575Sopenharmony_ci    }
1010d163575Sopenharmony_ci
1020d163575Sopenharmony_ci    fd_set *fdset = &g_epollFdSet;
1030d163575Sopenharmony_ci    if (fdset && FD_ISSET(efd, fdset)) {
1040d163575Sopenharmony_ci        FD_CLR(efd, fdset);
1050d163575Sopenharmony_ci        g_epPrivBuf[efd] = NULL;
1060d163575Sopenharmony_ci    }
1070d163575Sopenharmony_ci
1080d163575Sopenharmony_ci    return 0;
1090d163575Sopenharmony_ci}
1100d163575Sopenharmony_ci
1110d163575Sopenharmony_ci/**
1120d163575Sopenharmony_ci * get private data by epoll fd
1130d163575Sopenharmony_ci *
1140d163575Sopenharmony_ci * @param fd: epoll fd.
1150d163575Sopenharmony_ci * @return point to epoll_head
1160d163575Sopenharmony_ci */
1170d163575Sopenharmony_cistatic struct epoll_head *EpollGetDataBuff(int fd)
1180d163575Sopenharmony_ci{
1190d163575Sopenharmony_ci    int id = fd - EPOLL_FD_OFFSET;
1200d163575Sopenharmony_ci
1210d163575Sopenharmony_ci    if ((id < 0) || (id >= MAX_EPOLL_FD)) {
1220d163575Sopenharmony_ci        return NULL;
1230d163575Sopenharmony_ci    }
1240d163575Sopenharmony_ci
1250d163575Sopenharmony_ci    return g_epPrivBuf[id];
1260d163575Sopenharmony_ci}
1270d163575Sopenharmony_ci
1280d163575Sopenharmony_ci/**
1290d163575Sopenharmony_ci * when do EPOLL_CTL_ADD, need check if fd exist
1300d163575Sopenharmony_ci *
1310d163575Sopenharmony_ci * @param epHead: epoll control head, find by epoll id .
1320d163575Sopenharmony_ci * @param fd: ctl add fd.
1330d163575Sopenharmony_ci * @return 0 or -1
1340d163575Sopenharmony_ci */
1350d163575Sopenharmony_cistatic int CheckFdExist(struct epoll_head *epHead, int fd)
1360d163575Sopenharmony_ci{
1370d163575Sopenharmony_ci    int i;
1380d163575Sopenharmony_ci    for (i = 0; i < epHead->nodeCount; i++) {
1390d163575Sopenharmony_ci        if (epHead->evs[i].data.fd == fd) {
1400d163575Sopenharmony_ci            return -1;
1410d163575Sopenharmony_ci        }
1420d163575Sopenharmony_ci    }
1430d163575Sopenharmony_ci
1440d163575Sopenharmony_ci    return 0;
1450d163575Sopenharmony_ci}
1460d163575Sopenharmony_ci
1470d163575Sopenharmony_ci/**
1480d163575Sopenharmony_ci * close epoll
1490d163575Sopenharmony_ci *
1500d163575Sopenharmony_ci * @param epHead: epoll control head.
1510d163575Sopenharmony_ci * @return void
1520d163575Sopenharmony_ci */
1530d163575Sopenharmony_cistatic VOID DoEpollClose(struct epoll_head *epHead)
1540d163575Sopenharmony_ci{
1550d163575Sopenharmony_ci    if (epHead != NULL) {
1560d163575Sopenharmony_ci        if (epHead->evs != NULL) {
1570d163575Sopenharmony_ci            free(epHead->evs);
1580d163575Sopenharmony_ci        }
1590d163575Sopenharmony_ci
1600d163575Sopenharmony_ci        free(epHead);
1610d163575Sopenharmony_ci    }
1620d163575Sopenharmony_ci
1630d163575Sopenharmony_ci    return;
1640d163575Sopenharmony_ci}
1650d163575Sopenharmony_ci
1660d163575Sopenharmony_ci/**
1670d163575Sopenharmony_ci * epoll_create unsupported api
1680d163575Sopenharmony_ci *
1690d163575Sopenharmony_ci * epoll_create is implemented by calling epoll_create1, it's parameter 'size' is useless.
1700d163575Sopenharmony_ci *
1710d163575Sopenharmony_ci * epoll_create1,
1720d163575Sopenharmony_ci * The simple version of epoll does not use red-black trees,
1730d163575Sopenharmony_ci * so when fd is normal value (greater than 0),
1740d163575Sopenharmony_ci * actually allocated epoll can manage num of EPOLL_DEFAULT_SIZE
1750d163575Sopenharmony_ci *
1760d163575Sopenharmony_ci * @param flags: not actually used
1770d163575Sopenharmony_ci * @return epoll fd
1780d163575Sopenharmony_ci */
1790d163575Sopenharmony_ciint epoll_create1(int flags)
1800d163575Sopenharmony_ci{
1810d163575Sopenharmony_ci    (void)flags;
1820d163575Sopenharmony_ci    int fd = -1;
1830d163575Sopenharmony_ci
1840d163575Sopenharmony_ci    struct epoll_head *epHead = (struct epoll_head *)malloc(sizeof(struct epoll_head));
1850d163575Sopenharmony_ci    if (epHead == NULL) {
1860d163575Sopenharmony_ci        set_errno(ENOMEM);
1870d163575Sopenharmony_ci        return fd;
1880d163575Sopenharmony_ci    }
1890d163575Sopenharmony_ci
1900d163575Sopenharmony_ci    /* actually allocated epoll can manage num is EPOLL_DEFAULT_SIZE */
1910d163575Sopenharmony_ci    epHead->size = EPOLL_DEFAULT_SIZE;
1920d163575Sopenharmony_ci    epHead->nodeCount = 0;
1930d163575Sopenharmony_ci    epHead->evs = malloc(sizeof(struct epoll_event) * EPOLL_DEFAULT_SIZE);
1940d163575Sopenharmony_ci    if (epHead->evs == NULL) {
1950d163575Sopenharmony_ci        free(epHead);
1960d163575Sopenharmony_ci        set_errno(ENOMEM);
1970d163575Sopenharmony_ci        return fd;
1980d163575Sopenharmony_ci    }
1990d163575Sopenharmony_ci
2000d163575Sopenharmony_ci    /* fd set, get sysfd, for close */
2010d163575Sopenharmony_ci    (VOID)pthread_mutex_lock(&g_epollMutex);
2020d163575Sopenharmony_ci    fd = EpollAllocSysFd(MAX_EPOLL_FD, epHead);
2030d163575Sopenharmony_ci    if (fd == -1) {
2040d163575Sopenharmony_ci        (VOID)pthread_mutex_unlock(&g_epollMutex);
2050d163575Sopenharmony_ci        DoEpollClose(epHead);
2060d163575Sopenharmony_ci        set_errno(EMFILE);
2070d163575Sopenharmony_ci        return fd;
2080d163575Sopenharmony_ci    }
2090d163575Sopenharmony_ci    (VOID)pthread_mutex_unlock(&g_epollMutex);
2100d163575Sopenharmony_ci    return fd;
2110d163575Sopenharmony_ci}
2120d163575Sopenharmony_ci
2130d163575Sopenharmony_ci/**
2140d163575Sopenharmony_ci * epoll_close,
2150d163575Sopenharmony_ci * called by close
2160d163575Sopenharmony_ci * @param epfd: epoll fd
2170d163575Sopenharmony_ci * @return 0 or -1
2180d163575Sopenharmony_ci */
2190d163575Sopenharmony_ciint epoll_close(int epfd)
2200d163575Sopenharmony_ci{
2210d163575Sopenharmony_ci    struct epoll_head *epHead = NULL;
2220d163575Sopenharmony_ci
2230d163575Sopenharmony_ci    (VOID)pthread_mutex_lock(&g_epollMutex);
2240d163575Sopenharmony_ci    epHead = EpollGetDataBuff(epfd);
2250d163575Sopenharmony_ci    if (epHead == NULL) {
2260d163575Sopenharmony_ci        (VOID)pthread_mutex_unlock(&g_epollMutex);
2270d163575Sopenharmony_ci        set_errno(EBADF);
2280d163575Sopenharmony_ci        return -1;
2290d163575Sopenharmony_ci    }
2300d163575Sopenharmony_ci
2310d163575Sopenharmony_ci    DoEpollClose(epHead);
2320d163575Sopenharmony_ci    int ret = EpollFreeSysFd(epfd);
2330d163575Sopenharmony_ci    (VOID)pthread_mutex_unlock(&g_epollMutex);
2340d163575Sopenharmony_ci    return ret;
2350d163575Sopenharmony_ci}
2360d163575Sopenharmony_ci
2370d163575Sopenharmony_ciint epoll_ctl(int epfd, int op, int fd, struct epoll_event *ev)
2380d163575Sopenharmony_ci{
2390d163575Sopenharmony_ci    struct epoll_head *epHead = NULL;
2400d163575Sopenharmony_ci    int i;
2410d163575Sopenharmony_ci    int ret = -1;
2420d163575Sopenharmony_ci
2430d163575Sopenharmony_ci    (VOID)pthread_mutex_lock(&g_epollMutex);
2440d163575Sopenharmony_ci    epHead = EpollGetDataBuff(epfd);
2450d163575Sopenharmony_ci    if (epHead == NULL) {
2460d163575Sopenharmony_ci        set_errno(EBADF);
2470d163575Sopenharmony_ci        goto OUT_RELEASE;
2480d163575Sopenharmony_ci    }
2490d163575Sopenharmony_ci
2500d163575Sopenharmony_ci    if (ev == NULL) {
2510d163575Sopenharmony_ci        set_errno(EINVAL);
2520d163575Sopenharmony_ci        goto OUT_RELEASE;
2530d163575Sopenharmony_ci    }
2540d163575Sopenharmony_ci
2550d163575Sopenharmony_ci    switch (op) {
2560d163575Sopenharmony_ci        case EPOLL_CTL_ADD:
2570d163575Sopenharmony_ci            ret = CheckFdExist(epHead, fd);
2580d163575Sopenharmony_ci            if (ret == -1) {
2590d163575Sopenharmony_ci                set_errno(EEXIST);
2600d163575Sopenharmony_ci                goto OUT_RELEASE;
2610d163575Sopenharmony_ci            }
2620d163575Sopenharmony_ci
2630d163575Sopenharmony_ci            if (epHead->nodeCount == EPOLL_DEFAULT_SIZE) {
2640d163575Sopenharmony_ci                set_errno(ENOMEM);
2650d163575Sopenharmony_ci                goto OUT_RELEASE;
2660d163575Sopenharmony_ci            }
2670d163575Sopenharmony_ci
2680d163575Sopenharmony_ci            epHead->evs[epHead->nodeCount].events = ev->events | POLLERR | POLLHUP;
2690d163575Sopenharmony_ci            epHead->evs[epHead->nodeCount].data.fd = fd;
2700d163575Sopenharmony_ci            epHead->nodeCount++;
2710d163575Sopenharmony_ci            ret = 0;
2720d163575Sopenharmony_ci            break;
2730d163575Sopenharmony_ci        case EPOLL_CTL_DEL:
2740d163575Sopenharmony_ci            for (i = 0; i < epHead->nodeCount; i++) {
2750d163575Sopenharmony_ci                if (epHead->evs[i].data.fd != fd) {
2760d163575Sopenharmony_ci                    continue;
2770d163575Sopenharmony_ci                }
2780d163575Sopenharmony_ci
2790d163575Sopenharmony_ci                if (i != epHead->nodeCount - 1) {
2800d163575Sopenharmony_ci                    memmove_s(&epHead->evs[i], epHead->nodeCount - i, &epHead->evs[i + 1],
2810d163575Sopenharmony_ci                              epHead->nodeCount - i);
2820d163575Sopenharmony_ci                }
2830d163575Sopenharmony_ci                epHead->nodeCount--;
2840d163575Sopenharmony_ci                ret = 0;
2850d163575Sopenharmony_ci                goto OUT_RELEASE;
2860d163575Sopenharmony_ci            }
2870d163575Sopenharmony_ci            set_errno(ENOENT);
2880d163575Sopenharmony_ci            break;
2890d163575Sopenharmony_ci        case EPOLL_CTL_MOD:
2900d163575Sopenharmony_ci            for (i = 0; i < epHead->nodeCount; i++) {
2910d163575Sopenharmony_ci                if (epHead->evs[i].data.fd == fd) {
2920d163575Sopenharmony_ci                    epHead->evs[i].events = ev->events | POLLERR | POLLHUP;
2930d163575Sopenharmony_ci                    ret = 0;
2940d163575Sopenharmony_ci                    goto OUT_RELEASE;
2950d163575Sopenharmony_ci                }
2960d163575Sopenharmony_ci            }
2970d163575Sopenharmony_ci            set_errno(ENOENT);
2980d163575Sopenharmony_ci            break;
2990d163575Sopenharmony_ci        default:
3000d163575Sopenharmony_ci            set_errno(EINVAL);
3010d163575Sopenharmony_ci            break;
3020d163575Sopenharmony_ci    }
3030d163575Sopenharmony_ci
3040d163575Sopenharmony_ciOUT_RELEASE:
3050d163575Sopenharmony_ci    (VOID)pthread_mutex_unlock(&g_epollMutex);
3060d163575Sopenharmony_ci    return ret;
3070d163575Sopenharmony_ci}
3080d163575Sopenharmony_ci
3090d163575Sopenharmony_ciint epoll_wait(int epfd, FAR struct epoll_event *evs, int maxevents, int timeout)
3100d163575Sopenharmony_ci{
3110d163575Sopenharmony_ci    struct epoll_head *epHead = NULL;
3120d163575Sopenharmony_ci    int ret;
3130d163575Sopenharmony_ci    int counter;
3140d163575Sopenharmony_ci    int i;
3150d163575Sopenharmony_ci    struct pollfd *pFd = NULL;
3160d163575Sopenharmony_ci    int pollSize;
3170d163575Sopenharmony_ci
3180d163575Sopenharmony_ci    epHead = EpollGetDataBuff(epfd);
3190d163575Sopenharmony_ci    if (epHead == NULL) {
3200d163575Sopenharmony_ci        set_errno(EBADF);
3210d163575Sopenharmony_ci        return -1;
3220d163575Sopenharmony_ci    }
3230d163575Sopenharmony_ci
3240d163575Sopenharmony_ci    if ((maxevents <= 0) || (evs == NULL)) {
3250d163575Sopenharmony_ci        set_errno(EINVAL);
3260d163575Sopenharmony_ci        return -1;
3270d163575Sopenharmony_ci    }
3280d163575Sopenharmony_ci
3290d163575Sopenharmony_ci    if (maxevents > epHead->nodeCount) {
3300d163575Sopenharmony_ci        pollSize = epHead->nodeCount;
3310d163575Sopenharmony_ci    } else {
3320d163575Sopenharmony_ci        pollSize = maxevents;
3330d163575Sopenharmony_ci    }
3340d163575Sopenharmony_ci
3350d163575Sopenharmony_ci    pFd = malloc(sizeof(struct pollfd) * pollSize);
3360d163575Sopenharmony_ci    if (pFd == NULL) {
3370d163575Sopenharmony_ci        set_errno(EINVAL);
3380d163575Sopenharmony_ci        return -1;
3390d163575Sopenharmony_ci    }
3400d163575Sopenharmony_ci
3410d163575Sopenharmony_ci    for (i = 0; i < pollSize; i++) {
3420d163575Sopenharmony_ci        pFd[i].fd = epHead->evs[i].data.fd;
3430d163575Sopenharmony_ci        pFd[i].events = (short)epHead->evs[i].events;
3440d163575Sopenharmony_ci    }
3450d163575Sopenharmony_ci
3460d163575Sopenharmony_ci
3470d163575Sopenharmony_ci    ret = poll(pFd, pollSize, timeout);
3480d163575Sopenharmony_ci    if (ret <= 0) {
3490d163575Sopenharmony_ci        free(pFd);
3500d163575Sopenharmony_ci        return 0;
3510d163575Sopenharmony_ci    }
3520d163575Sopenharmony_ci
3530d163575Sopenharmony_ci    for (i = 0, counter = 0; i < ret && counter < pollSize; counter++) {
3540d163575Sopenharmony_ci        if (pFd[counter].revents != 0) {
3550d163575Sopenharmony_ci            evs[i].data.fd = pFd[counter].fd;
3560d163575Sopenharmony_ci            evs[i].events  = pFd[counter].revents;
3570d163575Sopenharmony_ci            i++;
3580d163575Sopenharmony_ci        }
3590d163575Sopenharmony_ci    }
3600d163575Sopenharmony_ci
3610d163575Sopenharmony_ci    free(pFd);
3620d163575Sopenharmony_ci    return i;
3630d163575Sopenharmony_ci}
3640d163575Sopenharmony_ci
365