229 lines
		
	
	
		
			4.8 KiB
		
	
	
	
		
			C
		
	
	
		
		
			
		
	
	
			229 lines
		
	
	
		
			4.8 KiB
		
	
	
	
		
			C
		
	
	
|  | /*
 | ||
|  |  * Copyright (c) 2006-2021, RT-Thread Development Team | ||
|  |  * | ||
|  |  * SPDX-License-Identifier: Apache-2.0 | ||
|  |  * | ||
|  |  * Change Logs: | ||
|  |  * Date           Author       Notes | ||
|  |  * 2016-12-28     Bernard      first version | ||
|  |  * 2018-03-09     Bernard      Add protection for pt->triggered. | ||
|  |  */ | ||
|  | 
 | ||
|  | #include <stdint.h>
 | ||
|  | #include <rthw.h>
 | ||
|  | #include <rtthread.h>
 | ||
|  | #include <dfs_file.h>
 | ||
|  | #include "poll.h"
 | ||
|  | 
 | ||
|  | struct rt_poll_node; | ||
|  | 
 | ||
|  | struct rt_poll_table | ||
|  | { | ||
|  |     rt_pollreq_t req; | ||
|  |     rt_uint32_t triggered; /* the waited thread whether triggered */ | ||
|  |     rt_thread_t polling_thread; | ||
|  |     struct rt_poll_node *nodes; | ||
|  | }; | ||
|  | 
 | ||
|  | struct rt_poll_node | ||
|  | { | ||
|  |     struct rt_wqueue_node wqn; | ||
|  |     struct rt_poll_table *pt; | ||
|  |     struct rt_poll_node *next; | ||
|  | }; | ||
|  | 
 | ||
|  | static int __wqueue_pollwake(struct rt_wqueue_node *wait, void *key) | ||
|  | { | ||
|  |     struct rt_poll_node *pn; | ||
|  | 
 | ||
|  |     if (key && !((rt_ubase_t)key & wait->key)) | ||
|  |         return -1; | ||
|  | 
 | ||
|  |     pn = rt_container_of(wait, struct rt_poll_node, wqn); | ||
|  |     pn->pt->triggered = 1; | ||
|  | 
 | ||
|  |     return __wqueue_default_wake(wait, key); | ||
|  | } | ||
|  | 
 | ||
|  | static void _poll_add(rt_wqueue_t *wq, rt_pollreq_t *req) | ||
|  | { | ||
|  |     struct rt_poll_table *pt; | ||
|  |     struct rt_poll_node *node; | ||
|  | 
 | ||
|  |     node = (struct rt_poll_node *)rt_malloc(sizeof(struct rt_poll_node)); | ||
|  |     if (node == RT_NULL) | ||
|  |         return; | ||
|  | 
 | ||
|  |     pt = rt_container_of(req, struct rt_poll_table, req); | ||
|  | 
 | ||
|  |     node->wqn.key = req->_key; | ||
|  |     rt_list_init(&(node->wqn.list)); | ||
|  |     node->wqn.polling_thread = pt->polling_thread; | ||
|  |     node->wqn.wakeup = __wqueue_pollwake; | ||
|  |     node->next = pt->nodes; | ||
|  |     node->pt = pt; | ||
|  |     pt->nodes = node; | ||
|  |     rt_wqueue_add(wq, &node->wqn); | ||
|  | } | ||
|  | 
 | ||
|  | static void poll_table_init(struct rt_poll_table *pt) | ||
|  | { | ||
|  |     pt->req._proc = _poll_add; | ||
|  |     pt->triggered = 0; | ||
|  |     pt->nodes = RT_NULL; | ||
|  |     pt->polling_thread = rt_thread_self(); | ||
|  | } | ||
|  | 
 | ||
|  | static int poll_wait_timeout(struct rt_poll_table *pt, int msec) | ||
|  | { | ||
|  |     rt_int32_t timeout; | ||
|  |     int ret = 0; | ||
|  |     struct rt_thread *thread; | ||
|  |     rt_base_t level; | ||
|  | 
 | ||
|  |     thread = pt->polling_thread; | ||
|  | 
 | ||
|  |     timeout = rt_tick_from_millisecond(msec); | ||
|  | 
 | ||
|  |     level = rt_hw_interrupt_disable(); | ||
|  | 
 | ||
|  |     if (timeout != 0 && !pt->triggered) | ||
|  |     { | ||
|  |         if (rt_thread_suspend_with_flag(thread, RT_INTERRUPTIBLE) == RT_EOK) | ||
|  |         { | ||
|  |             if (timeout > 0) | ||
|  |             { | ||
|  |                 rt_timer_control(&(thread->thread_timer), | ||
|  |                         RT_TIMER_CTRL_SET_TIME, | ||
|  |                         &timeout); | ||
|  |                 rt_timer_start(&(thread->thread_timer)); | ||
|  |             } | ||
|  | 
 | ||
|  |             rt_hw_interrupt_enable(level); | ||
|  | 
 | ||
|  |             rt_schedule(); | ||
|  | 
 | ||
|  |             level = rt_hw_interrupt_disable(); | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     ret = !pt->triggered; | ||
|  |     rt_hw_interrupt_enable(level); | ||
|  | 
 | ||
|  |     return ret; | ||
|  | } | ||
|  | 
 | ||
|  | static int do_pollfd(struct pollfd *pollfd, rt_pollreq_t *req) | ||
|  | { | ||
|  |     int mask = 0; | ||
|  |     int fd; | ||
|  | 
 | ||
|  |     fd = pollfd->fd; | ||
|  | 
 | ||
|  |     if (fd >= 0) | ||
|  |     { | ||
|  |         struct dfs_file *f = fd_get(fd); | ||
|  |         mask = POLLNVAL; | ||
|  | 
 | ||
|  |         if (f) | ||
|  |         { | ||
|  |             mask = POLLMASK_DEFAULT; | ||
|  |             if (f->vnode->fops->poll) | ||
|  |             { | ||
|  |                 req->_key = pollfd->events | POLLERR | POLLHUP; | ||
|  | 
 | ||
|  |                 mask = f->vnode->fops->poll(f, req); | ||
|  | 
 | ||
|  |                 /* dealwith the device return error -1*/ | ||
|  |                 if (mask < 0) | ||
|  |                 { | ||
|  |                     pollfd->revents = 0; | ||
|  |                     return mask; | ||
|  |                 } | ||
|  |             } | ||
|  |             /* Mask out unneeded events. */ | ||
|  |             mask &= pollfd->events | POLLERR | POLLHUP; | ||
|  |         } | ||
|  |     } | ||
|  |     pollfd->revents = mask; | ||
|  | 
 | ||
|  |     return mask; | ||
|  | } | ||
|  | 
 | ||
|  | static int poll_do(struct pollfd *fds, nfds_t nfds, struct rt_poll_table *pt, int msec) | ||
|  | { | ||
|  |     int num; | ||
|  |     int istimeout = 0; | ||
|  |     nfds_t n; | ||
|  |     struct pollfd *pf; | ||
|  |     int  ret = 0; | ||
|  | 
 | ||
|  |     if (msec == 0) | ||
|  |     { | ||
|  |         pt->req._proc = RT_NULL; | ||
|  |         istimeout = 1; | ||
|  |     } | ||
|  | 
 | ||
|  |     while (1) | ||
|  |     { | ||
|  |         pf = fds; | ||
|  |         num = 0; | ||
|  |         pt->triggered = 0; | ||
|  | 
 | ||
|  |         for (n = 0; n < nfds; n ++) | ||
|  |         { | ||
|  |             ret = do_pollfd(pf, &pt->req); | ||
|  |             if(ret < 0) | ||
|  |             { | ||
|  |                 /*dealwith the device return error -1  */ | ||
|  |                 pt->req._proc = RT_NULL; | ||
|  |                 return ret; | ||
|  |             } | ||
|  |             else if(ret > 0) | ||
|  |             { | ||
|  |                 num ++; | ||
|  |                 pt->req._proc = RT_NULL; | ||
|  |             } | ||
|  |             pf ++; | ||
|  |         } | ||
|  | 
 | ||
|  |         pt->req._proc = RT_NULL; | ||
|  | 
 | ||
|  |         if (num || istimeout) | ||
|  |             break; | ||
|  | 
 | ||
|  |         if (poll_wait_timeout(pt, msec)) | ||
|  |             istimeout = 1; | ||
|  |     } | ||
|  | 
 | ||
|  |     return num; | ||
|  | } | ||
|  | 
 | ||
|  | static void poll_teardown(struct rt_poll_table *pt) | ||
|  | { | ||
|  |     struct rt_poll_node *node, *next; | ||
|  | 
 | ||
|  |     next = pt->nodes; | ||
|  |     while (next) | ||
|  |     { | ||
|  |         node = next; | ||
|  |         rt_wqueue_remove(&node->wqn); | ||
|  |         next = node->next; | ||
|  |         rt_free(node); | ||
|  |     } | ||
|  | } | ||
|  | 
 | ||
|  | int poll(struct pollfd *fds, nfds_t nfds, int timeout) | ||
|  | { | ||
|  |     int num; | ||
|  |     struct rt_poll_table table; | ||
|  | 
 | ||
|  |     poll_table_init(&table); | ||
|  | 
 | ||
|  |     num = poll_do(fds, nfds, &table, timeout); | ||
|  | 
 | ||
|  |     poll_teardown(&table); | ||
|  | 
 | ||
|  |     return num; | ||
|  | } |