2023-08-01 17:15:41 +08:00
/*
* Copyright ( c ) 2006 - 2023 , RT - Thread Development Team
*
* SPDX - License - Identifier : Apache - 2.0
*
* Change Logs :
* Date Author Notes
* 2023 - 07 - 29 zmq810150896 first version
*/
# include <rtthread.h>
# include <fcntl.h>
# include <stdint.h>
# include <unistd.h>
# include <dfs_file.h>
# include "sys/epoll.h"
# include "poll.h"
# include <lwp_signal.h>
# define EPOLL_MUTEX_NAME "EVENTEPOLL"
# define EFD_SHARED_EPOLL_TYPE (EPOLL_CTL_ADD | EPOLL_CTL_DEL | EPOLL_CTL_MOD)
# define EPOLLINOUT_BITS (EPOLLIN | EPOLLOUT | EPOLLRDNORM | EPOLLWRNORM)
# define EPOLLEXCLUSIVE_BITS (EPOLLINOUT_BITS | EPOLLERR | EPOLLHUP | \
EPOLLET | EPOLLEXCLUSIVE )
static struct rt_spinlock spinlock ;
struct rt_eventpoll ;
/* Monitor queue */
struct rt_fd_list
{
rt_uint32_t revents ; /* Monitored events */
struct epoll_event epev ;
rt_pollreq_t req ;
struct rt_eventpoll * ep ;
struct rt_wqueue_node wqn ;
struct rt_fd_list * next ;
} ;
struct rt_ready_list
{
int exclusive ; /* If triggered horizontally, a check is made to see if the data has been read, and if there is any data left to read, the readability event is returned in the next epoll_wait */
struct rt_fd_list * rdl_event ; /* rdl: ready list */
struct rt_ready_list * next ;
} ;
struct rt_eventpoll
{
rt_uint32_t tirggered ; /* the waited thread whether triggered */
rt_wqueue_t epoll_read ;
rt_thread_t polling_thread ;
struct rt_mutex lock ;
struct rt_fd_list * fdlist ; /* Monitor list */
int eventpoll_num ; /* Number of ready lists */
rt_pollreq_t req ;
struct rt_ready_list * rdlist ; /* ready list */
} ;
static int epoll_close ( struct dfs_file * file ) ;
static int epoll_poll ( struct dfs_file * file , struct rt_pollreq * req ) ;
static int epoll_get_event ( struct rt_fd_list * fl , rt_pollreq_t * req ) ;
static int epoll_do_ctl ( int epfd , int op , int fd , struct epoll_event * event ) ;
static const struct dfs_file_ops epoll_fops =
{
. close = epoll_close ,
. poll = epoll_poll ,
} ;
static int epoll_close_fdlist ( struct rt_fd_list * fdlist )
{
struct rt_fd_list * fre_node , * list ;
if ( fdlist ! = RT_NULL )
{
list = fdlist ;
while ( list - > next ! = RT_NULL )
{
fre_node = list - > next ;
rt_wqueue_remove ( & fre_node - > wqn ) ;
list - > next = fre_node - > next ;
rt_free ( fre_node ) ;
}
rt_free ( fdlist ) ;
}
return 0 ;
}
static int epoll_close_rdlist ( struct rt_ready_list * rdlist )
{
struct rt_ready_list * list , * fre_node ;
list = rdlist ;
if ( list )
{
while ( list - > next ! = RT_NULL )
{
fre_node = list - > next ;
list - > next = fre_node - > next ;
rt_free ( fre_node ) ;
}
rt_free ( rdlist ) ;
}
return 0 ;
}
static int epoll_close ( struct dfs_file * file )
{
struct rt_eventpoll * ep ;
if ( file - > vnode )
{
if ( file - > vnode - > data )
{
ep = file - > vnode - > data ;
if ( ep )
{
rt_mutex_take ( & ep - > lock , RT_WAITING_FOREVER ) ;
if ( ep - > fdlist )
{
epoll_close_fdlist ( ep - > fdlist ) ;
}
if ( ep - > rdlist )
{
epoll_close_rdlist ( ep - > rdlist ) ;
}
rt_mutex_release ( & ep - > lock ) ;
rt_mutex_detach ( & ep - > lock ) ;
rt_free ( ep ) ;
}
}
}
return 0 ;
}
static int epoll_poll ( struct dfs_file * file , struct rt_pollreq * req )
{
struct rt_eventpoll * ep ;
struct rt_fd_list * fdlist ;
int mask ;
int events = 0 ;
if ( file - > vnode - > data )
{
ep = file - > vnode - > data ;
ep - > req . _key = req - > _key ;
rt_poll_add ( & ep - > epoll_read , req ) ;
fdlist = ep - > fdlist ;
if ( fdlist )
{
while ( fdlist - > next ! = RT_NULL )
{
fdlist = fdlist - > next ;
mask = epoll_get_event ( fdlist , & fdlist - > req ) ;
2023-08-03 21:45:30 +08:00
if ( mask & fdlist - > revents )
2023-08-01 17:15:41 +08:00
{
events | = mask | POLLIN | EPOLLRDNORM ;
break ;
}
}
}
}
return events ;
}
static int epoll_rdlist_add ( struct rt_fd_list * fdl , rt_uint32_t revents )
{
struct rt_ready_list * rdlist = RT_NULL ;
struct rt_eventpoll * ep ;
int isexist = 0 ;
int res = - 1 ;
ep = fdl - > ep ;
if ( revents & ep - > req . _key )
{
rt_wqueue_wakeup ( & ep - > epoll_read , ( void * ) POLLIN ) ;
}
rt_mutex_take ( & ep - > lock , RT_WAITING_FOREVER ) ;
if ( ep - > rdlist = = RT_NULL )
{
ep - > rdlist = ( struct rt_ready_list * ) rt_malloc ( sizeof ( struct rt_ready_list ) ) ;
if ( ep - > rdlist = = RT_NULL )
{
return - 1 ;
}
ep - > rdlist - > next = RT_NULL ;
}
rdlist = ep - > rdlist ;
while ( rdlist - > next ! = RT_NULL )
{
rdlist = rdlist - > next ;
if ( rdlist - > rdl_event - > epev . data . fd = = fdl - > epev . data . fd )
{
isexist = 1 ;
res = 0 ;
break ;
}
}
if ( ! isexist )
{
rdlist = RT_NULL ;
rdlist = ( struct rt_ready_list * ) rt_malloc ( sizeof ( struct rt_ready_list ) ) ;
if ( rdlist ! = RT_NULL )
{
rdlist - > rdl_event = fdl ;
2023-08-03 21:45:30 +08:00
rdlist - > rdl_event - > epev . events = fdl - > revents & revents ;
2023-08-01 17:15:41 +08:00
rdlist - > next = ep - > rdlist - > next ;
rdlist - > exclusive = 0 ;
ep - > rdlist - > next = rdlist ;
ep - > eventpoll_num + + ;
res = 0 ;
if ( rdlist - > rdl_event - > revents & EPOLLONESHOT )
{
rdlist - > rdl_event - > revents = 0 ;
}
}
}
ep - > tirggered = 1 ;
rt_mutex_release ( & ep - > lock ) ;
return res ;
}
static int epoll_wqueue_callback ( struct rt_wqueue_node * wait , void * key )
{
struct rt_fd_list * fdlist ;
if ( key & & ! ( ( rt_ubase_t ) key & wait - > key ) )
return - 1 ;
fdlist = rt_container_of ( wait , struct rt_fd_list , wqn ) ;
2023-08-03 21:45:30 +08:00
if ( fdlist - > revents )
2023-08-01 17:15:41 +08:00
{
epoll_rdlist_add ( fdlist , ( rt_ubase_t ) key ) ;
}
return __wqueue_default_wake ( wait , key ) ;
}
static void epoll_wqueue_add_callback ( rt_wqueue_t * wq , rt_pollreq_t * req )
{
struct rt_fd_list * fdlist ;
struct rt_eventpoll * ep ;
fdlist = rt_container_of ( req , struct rt_fd_list , req ) ;
ep = fdlist - > ep ;
fdlist - > wqn . key = req - > _key ;
rt_list_init ( & ( fdlist - > wqn . list ) ) ;
fdlist - > wqn . polling_thread = ep - > polling_thread ;
fdlist - > wqn . wakeup = epoll_wqueue_callback ;
rt_wqueue_add ( wq , & fdlist - > wqn ) ;
}
static void epoll_ctl_install ( struct rt_fd_list * fdlist , struct rt_eventpoll * ep )
{
rt_uint32_t mask = 0 ;
2023-08-03 21:45:30 +08:00
fdlist - > req . _key = fdlist - > revents ;
2023-08-01 17:15:41 +08:00
mask = epoll_get_event ( fdlist , & fdlist - > req ) ;
2023-08-03 21:45:30 +08:00
if ( mask & fdlist - > revents )
2023-08-01 17:15:41 +08:00
{
epoll_rdlist_add ( fdlist , mask ) ;
}
}
static void epoll_member_init ( struct rt_eventpoll * ep )
{
ep - > tirggered = 0 ;
ep - > eventpoll_num = 0 ;
ep - > polling_thread = rt_thread_self ( ) ;
ep - > rdlist = RT_NULL ;
ep - > fdlist = RT_NULL ;
ep - > req . _key = 0 ;
rt_wqueue_init ( & ep - > epoll_read ) ;
rt_spin_lock_init ( & spinlock ) ;
}
static int epoll_epf_init ( int fd )
{
struct dfs_file * df ;
struct rt_eventpoll * ep ;
rt_err_t ret = 0 ;
df = fd_get ( fd ) ;
if ( df )
{
ep = ( struct rt_eventpoll * ) rt_malloc ( sizeof ( struct rt_eventpoll ) ) ;
if ( ep )
{
epoll_member_init ( ep ) ;
rt_mutex_init ( & ep - > lock , EPOLL_MUTEX_NAME , RT_IPC_FLAG_FIFO ) ;
2023-08-03 21:45:30 +08:00
# ifdef RT_USING_DFS_V2
df - > fops = & epoll_fops ;
# endif
2023-08-01 17:15:41 +08:00
df - > vnode = ( struct dfs_vnode * ) rt_malloc ( sizeof ( struct dfs_vnode ) ) ;
if ( df - > vnode )
{
ep - > fdlist = ( struct rt_fd_list * ) rt_malloc ( sizeof ( struct rt_fd_list ) ) ;
if ( ep - > fdlist )
{
ep - > fdlist - > next = RT_NULL ;
ep - > fdlist - > epev . data . fd = fd ;
ep - > fdlist - > ep = ep ;
dfs_vnode_init ( df - > vnode , FT_REGULAR , & epoll_fops ) ;
df - > vnode - > data = ep ;
}
else
{
ret = - ENOMEM ;
rt_free ( df - > vnode ) ;
rt_free ( ep ) ;
}
}
else
{
ret = - ENOMEM ;
rt_free ( ep ) ;
}
}
else
{
ret = - ENOMEM ;
}
}
return ret ;
}
static int epoll_do_create ( int size )
{
2023-08-03 21:45:30 +08:00
rt_err_t ret = - 1 ;
2023-08-01 17:15:41 +08:00
int status ;
int fd ;
if ( size < 0 )
{
2023-08-03 21:45:30 +08:00
rt_set_errno ( EINVAL ) ;
2023-08-01 17:15:41 +08:00
}
else
{
fd = fd_new ( ) ;
if ( fd > = 0 )
{
ret = fd ;
status = epoll_epf_init ( fd ) ;
if ( status < 0 )
{
fd_release ( fd ) ;
2023-08-03 21:45:30 +08:00
rt_set_errno ( - status ) ;
2023-08-01 17:15:41 +08:00
}
}
else
{
2023-08-03 21:45:30 +08:00
rt_set_errno ( - fd ) ;
2023-08-01 17:15:41 +08:00
}
}
return ret ;
}
static int epoll_ctl_add ( struct dfs_file * df , struct epoll_event * event )
{
struct rt_fd_list * fdlist ;
struct rt_eventpoll * ep ;
rt_err_t ret = - EINVAL ;
if ( df - > vnode - > data )
{
ep = df - > vnode - > data ;
fdlist = ep - > fdlist ;
ret = 0 ;
while ( fdlist - > next ! = RT_NULL )
{
if ( fdlist - > next - > epev . data . fd = = event - > data . fd )
{
return 0 ;
}
fdlist = fdlist - > next ;
}
fdlist = ( struct rt_fd_list * ) rt_malloc ( sizeof ( struct rt_fd_list ) ) ;
if ( fdlist )
{
fdlist - > epev . data . fd = event - > data . fd ;
fdlist - > epev . events = event - > events ;
fdlist - > ep = ep ;
fdlist - > req . _proc = epoll_wqueue_add_callback ;
fdlist - > next = ep - > fdlist - > next ;
fdlist - > revents = event - > events ;
ep - > fdlist - > next = fdlist ;
epoll_ctl_install ( fdlist , ep ) ;
}
else
{
ret = - ENOMEM ;
}
}
return ret ;
}
static int epoll_ctl_del ( struct dfs_file * df , struct epoll_event * event )
{
struct rt_fd_list * fdlist , * fre_fd ;
struct rt_eventpoll * ep = RT_NULL ;
struct rt_ready_list * rdlist , * fre_rdl ;
rt_err_t ret = - EINVAL ;
if ( df - > vnode - > data )
{
ep = df - > vnode - > data ;
fdlist = ep - > fdlist ;
while ( fdlist - > next ! = RT_NULL )
{
if ( fdlist - > next - > epev . data . fd = = event - > data . fd )
{
fre_fd = fdlist - > next ;
fdlist - > next = fdlist - > next - > next ;
if ( fre_fd - > epev . events ! = 0 )
{
rt_wqueue_remove ( & fre_fd - > wqn ) ;
}
rt_free ( fre_fd ) ;
break ;
}
else
{
fdlist = fdlist - > next ;
}
}
if ( ep - > rdlist )
{
rdlist = ep - > rdlist ;
while ( rdlist - > next ! = RT_NULL )
{
if ( rdlist - > next - > rdl_event - > epev . data . fd = = event - > data . fd )
{
fre_rdl = rdlist - > next ;
rdlist - > next = rdlist - > next - > next ;
ep - > eventpoll_num - - ;
rt_free ( fre_rdl ) ;
break ;
}
else
{
rdlist = rdlist - > next ;
}
}
}
ret = 0 ;
}
return ret ;
}
static int epoll_ctl_mod ( struct dfs_file * df , struct epoll_event * event )
{
struct rt_fd_list * fdlist ;
struct rt_eventpoll * ep = RT_NULL ;
rt_err_t ret = - EINVAL ;
if ( df - > vnode - > data )
{
ep = df - > vnode - > data ;
fdlist = ep - > fdlist ;
while ( fdlist - > next ! = RT_NULL )
{
if ( fdlist - > next - > epev . data . fd = = event - > data . fd )
{
fdlist - > next - > epev . events = event - > events ;
fdlist - > next - > revents = event - > events ;
rt_wqueue_remove ( & fdlist - > next - > wqn ) ;
epoll_ctl_install ( fdlist - > next , ep ) ;
break ;
}
fdlist = fdlist - > next ;
}
ret = 0 ;
}
return ret ;
}
static int epoll_do_ctl ( int epfd , int op , int fd , struct epoll_event * event )
{
struct dfs_file * epdf ;
struct rt_eventpoll * ep ;
2023-08-03 21:45:30 +08:00
rt_err_t ret = 0 ;
2023-08-01 17:15:41 +08:00
if ( op & ~ EFD_SHARED_EPOLL_TYPE )
2023-08-03 21:45:30 +08:00
{
rt_set_errno ( EINVAL ) ;
return - 1 ;
}
2023-08-01 17:15:41 +08:00
if ( ( epfd = = fd ) | | ( epfd < 0 ) | | ( fd < 0 ) | | ( event - > data . fd ! = fd ) )
2023-08-03 21:45:30 +08:00
{
rt_set_errno ( EINVAL ) ;
return - 1 ;
}
2023-08-01 17:15:41 +08:00
if ( ! ( event - > events & EPOLLEXCLUSIVE_BITS ) )
2023-08-03 21:45:30 +08:00
{
rt_set_errno ( EINVAL ) ;
return - 1 ;
}
2023-08-01 17:15:41 +08:00
epdf = fd_get ( epfd ) ;
if ( epdf - > vnode - > data )
{
ep = epdf - > vnode - > data ;
rt_mutex_take ( & ep - > lock , RT_WAITING_FOREVER ) ;
switch ( op )
{
case EPOLL_CTL_ADD :
ret = epoll_ctl_add ( epdf , event ) ;
break ;
case EPOLL_CTL_DEL :
ret = epoll_ctl_del ( epdf , event ) ;
break ;
case EPOLL_CTL_MOD :
ret = epoll_ctl_mod ( epdf , event ) ;
break ;
default :
2023-08-03 21:45:30 +08:00
rt_set_errno ( EINVAL ) ;
2023-08-01 17:15:41 +08:00
break ;
}
2023-08-03 21:45:30 +08:00
if ( ret < 0 )
{
rt_set_errno ( - ret ) ;
ret = - 1 ;
}
rt_mutex_release ( & ep - > lock ) ;
}
2023-08-01 17:15:41 +08:00
return ret ;
}
static int epoll_wait_timeout ( struct rt_eventpoll * ep , int msec )
{
rt_int32_t timeout ;
struct rt_thread * thread ;
rt_base_t level ;
int ret = 0 ;
thread = ep - > polling_thread ;
timeout = rt_tick_from_millisecond ( msec ) ;
level = rt_spin_lock_irqsave ( & spinlock ) ;
if ( timeout ! = 0 & & ! ep - > tirggered )
{
if ( rt_thread_suspend_with_flag ( thread , RT_KILLABLE ) = = RT_EOK )
{
if ( timeout > 0 )
{
rt_timer_control ( & ( thread - > thread_timer ) ,
RT_TIMER_CTRL_SET_TIME ,
& timeout ) ;
rt_timer_start ( & ( thread - > thread_timer ) ) ;
}
rt_spin_unlock_irqrestore ( & spinlock , level ) ;
rt_schedule ( ) ;
level = rt_spin_lock_irqsave ( & spinlock ) ;
}
}
ret = ! ep - > tirggered ;
rt_spin_unlock_irqrestore ( & spinlock , level ) ;
return ret ;
}
static int epoll_get_event ( struct rt_fd_list * fl , rt_pollreq_t * req )
{
struct dfs_file * df ;
int mask = 0 ;
int fd = 0 ;
fd = fl - > epev . data . fd ;
if ( fd > = 0 )
{
df = fd_get ( fd ) ;
if ( df )
{
if ( df - > vnode - > fops - > poll )
{
2023-08-03 21:45:30 +08:00
req - > _key = fl - > revents | POLLERR | POLLHUP ;
2023-08-01 17:15:41 +08:00
mask = df - > vnode - > fops - > poll ( df , req ) ;
if ( mask < 0 )
return mask ;
}
2023-08-03 21:45:30 +08:00
mask & = fl - > revents | EPOLLOUT | POLLERR ;
2023-08-01 17:15:41 +08:00
}
}
return mask ;
}
static int epoll_do ( struct rt_eventpoll * ep , struct epoll_event * events , int maxevents , int timeout )
{
struct rt_ready_list * rdlist , * pre_rdlist ;
int event_num = 0 ;
int istimeout = 0 ;
int isn_add = 0 ;
int isfree = 0 ;
int mask = 0 ;
while ( 1 )
{
rt_mutex_take ( & ep - > lock , RT_WAITING_FOREVER ) ;
if ( ep - > eventpoll_num > 0 )
{
rdlist = ep - > rdlist ;
while ( rdlist - > next ! = RT_NULL )
{
isfree = 0 ;
isn_add = 0 ;
pre_rdlist = rdlist ;
rdlist = rdlist - > next ;
if ( event_num < maxevents )
{
if ( rdlist - > rdl_event - > revents = = 0 )
{
isfree = 1 ;
rt_wqueue_remove ( & rdlist - > rdl_event - > wqn ) ;
}
else
{
if ( rdlist - > rdl_event - > revents & EPOLLET )
{
rt_wqueue_remove ( & rdlist - > rdl_event - > wqn ) ;
2023-08-03 21:45:30 +08:00
mask = epoll_get_event ( rdlist - > rdl_event , & rdlist - > rdl_event - > req ) ;
rdlist - > rdl_event - > epev . events = mask & rdlist - > rdl_event - > revents ;
2023-08-01 17:15:41 +08:00
isfree = 1 ;
}
else
{
if ( rdlist - > exclusive )
{
rt_wqueue_remove ( & rdlist - > rdl_event - > wqn ) ;
mask = epoll_get_event ( rdlist - > rdl_event , & rdlist - > rdl_event - > req ) ;
if ( mask & rdlist - > rdl_event - > revents )
{
rdlist - > rdl_event - > epev . events = mask ;
}
else
{
isfree = 1 ;
isn_add = 1 ;
}
}
else
{
rdlist - > exclusive = 1 ;
}
}
}
if ( ! isn_add )
{
events [ event_num ] . data . fd = rdlist - > rdl_event - > epev . data . fd ;
events [ event_num ] . events = rdlist - > rdl_event - > epev . events ;
event_num + + ;
}
if ( isfree )
{
pre_rdlist - > next = rdlist - > next ;
rt_free ( rdlist ) ;
ep - > eventpoll_num - - ;
rdlist = pre_rdlist ;
}
}
else
{
break ;
}
}
}
rt_mutex_release ( & ep - > lock ) ;
if ( event_num | | istimeout )
{
ep - > tirggered = 0 ;
break ;
}
if ( epoll_wait_timeout ( ep , timeout ) )
{
istimeout = 1 ;
}
}
return event_num ;
}
static int epoll_do_wait ( int epfd , struct epoll_event * events , int maxevents , int timeout , const sigset_t * ss )
{
struct rt_eventpoll * ep ;
struct dfs_file * df ;
lwp_sigset_t old_sig , new_sig ;
2023-08-03 21:45:30 +08:00
rt_err_t ret = 0 ;
2023-08-01 17:15:41 +08:00
if ( ss )
{
memcpy ( & new_sig , ss , sizeof ( lwp_sigset_t ) ) ;
lwp_thread_signal_mask ( rt_thread_self ( ) , LWP_SIG_MASK_CMD_BLOCK , & new_sig , & old_sig ) ;
}
if ( ( maxevents > 0 ) & & ( epfd > = 0 ) )
{
df = fd_get ( epfd ) ;
if ( df & & df - > vnode )
{
ep = ( struct rt_eventpoll * ) df - > vnode - > data ;
if ( ep )
{
ret = epoll_do ( ep , events , maxevents , timeout ) ;
}
}
}
if ( ss )
{
lwp_thread_signal_mask ( rt_thread_self ( ) , LWP_SIG_MASK_CMD_SET_MASK , & old_sig , RT_NULL ) ;
}
2023-08-03 21:45:30 +08:00
if ( ret < 0 )
{
rt_set_errno ( - ret ) ;
ret = - 1 ;
}
2023-08-01 17:15:41 +08:00
return ret ;
}
int epoll_create ( int size )
{
return epoll_do_create ( size ) ;
}
int epoll_ctl ( int epfd , int op , int fd , struct epoll_event * event )
{
return epoll_do_ctl ( epfd , op , fd , event ) ;
}
int epoll_wait ( int epfd , struct epoll_event * events , int maxevents , int timeout )
{
return epoll_do_wait ( epfd , events , maxevents , timeout , RT_NULL ) ;
}
int epoll_pwait ( int epfd , struct epoll_event * events , int maxevents , int timeout , const sigset_t * ss )
{
return epoll_do_wait ( epfd , events , maxevents , timeout , ss ) ;
}
int epoll_pwait2 ( int epfd , struct epoll_event * events , int maxevents , int timeout , const sigset_t * ss )
{
return epoll_do_wait ( epfd , events , maxevents , timeout , ss ) ;
}