• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /****************************************************************************
2  * fs/vfs/fs_poll.c
3  *
4  * Copyright (c) 2023 Huawei Device Co., Ltd. All rights reserved.
5  * Based on NuttX originally from nuttx source (nuttx/fs/ and nuttx/drivers/)
6  *
7  * Licensed under the Apache License, Version 2.0 (the "License");
8  * you may not use this file except in compliance with the License.
9  * You may obtain a copy of the License at
10  *
11  *     http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing, software
14  * distributed under the License is distributed on an "AS IS" BASIS,
15  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16  * See the License for the specific language governing permissions and
17  * limitations under the License.
18  *
19  ****************************************************************************/
20 
21 /****************************************************************************
22  * Included Files
23  ****************************************************************************/
24 
25 #include "los_hwi.h"
26 #include "vfs_config.h"
27 #include "stdint.h"
28 #include "poll.h"
29 #include "assert.h"
30 #include "errno.h"
31 #include "vnode.h"
32 #include "stdlib.h"
33 #include "stdio.h"
34 #include "console.h"
35 #include "unistd.h"
36 #include "linux/wait.h"
37 #ifdef LOSCFG_NET_LWIP_SACK
38 #include "lwip/sockets.h"
39 #endif
40 #ifndef CONFIG_DISABLE_POLL
41 
42 /****************************************************************************
43  * Pre-processor Definitions
44  ****************************************************************************/
45 
46 #ifndef MSEC_PER_SEC
47 #define MSEC_PER_SEC        1000L
48 #endif
49 
50 #ifndef NSEC_PER_MSEC
51 #define NSEC_PER_MSEC    1000000L
52 #endif
53 
54 #define N_POLL_ITEMS 5
55 #define poll_semgive(sem) sem_post(sem)
56 
57 /****************************************************************************
58  * Data Structures & Macros
59  ****************************************************************************/
60 
61 typedef wait_queue_head_t * poll_wait_queue;
62 
63 typedef struct tag_poll_wait_node
64 {
65   LOS_DL_LIST queue_node;
66   pollevent_t key;
67   struct tag_poll_wait_entry *entry;
68   poll_wait_queue wait_queue;
69 } poll_wait_node;
70 
71 typedef struct tag_poll_wait_entry_table
72 {
73   struct tag_poll_wait_entry_table *next;
74   UINT32 index;
75   poll_wait_node items[N_POLL_ITEMS];
76 } poll_wait_entry_table;
77 
78 typedef struct tag_poll_wait_entry
79 {
80   bool add_queue_flag;
81   sem_t sem;
82   UINT32 inline_index;
83   poll_wait_node inline_items[N_POLL_ITEMS];
84   poll_wait_entry_table *table;
85 } poll_wait_entry;
86 
87 /****************************************************************************
88  * Private Functions
89  ****************************************************************************/
90 
91 /****************************************************************************
92  * Name: poll_semtake
93  ****************************************************************************/
94 
poll_semtake(sem_t * sem)95 static int poll_semtake(sem_t *sem)
96 {
97   /* Take the semaphore (perhaps waiting) */
98 
99   if (sem_wait(sem) < 0)
100     {
101       int err = get_errno();
102 
103       /* The only case that an error should occur here is if the wait were
104        * awakened by a signal.
105        */
106 
107       DEBUGASSERT(err == EINTR);
108       return -err;
109     }
110 
111   return OK;
112 }
113 
set_add_poll_wait_flag(poll_wait_head wait,bool add_queue_flag)114 static void set_add_poll_wait_flag(poll_wait_head wait, bool add_queue_flag)
115 {
116   wait->add_queue_flag = add_queue_flag;
117 }
118 
destroy_poll_wait(poll_wait_head wait)119 static int destroy_poll_wait(poll_wait_head wait)
120 {
121   unsigned int i;
122   unsigned long int_save;
123   poll_wait_node *wait_node = NULL;
124   poll_wait_entry_table *curr_table = NULL;
125 
126   for (i = 0; i < wait->inline_index; ++i)
127     {
128       wait_node = &wait->inline_items[i];
129       spin_lock_irqsave(&wait_node->wait_queue->lock, int_save);
130       LOS_ListDelete(&wait_node->queue_node);
131       spin_unlock_irqrestore(&wait_node->wait_queue->lock, int_save);
132     }
133 
134   while (wait->table)
135     {
136       curr_table = wait->table;
137       wait->table = curr_table->next;
138 
139       for (i = 0; i < curr_table->index; ++i)
140         {
141           wait_node = &curr_table->items[i];
142           spin_lock_irqsave(&wait_node->wait_queue->lock, int_save);
143           LOS_ListDelete(&wait_node->queue_node);
144           spin_unlock_irqrestore(&wait_node->wait_queue->lock, int_save);
145         }
146       free(curr_table);
147     }
148 
149   if (sem_destroy(&wait->sem) < 0)
150     {
151       PRINT_ERR("[%s] sem_destroy failed\n", __FUNCTION__);
152       return -1;
153     }
154 
155   return 0;
156 }
157 
get_poll_item(poll_wait_head wait)158 static poll_wait_node *get_poll_item(poll_wait_head wait)
159 {
160   if (wait->inline_index < N_POLL_ITEMS)
161     {
162       return wait->inline_items + wait->inline_index++;
163     }
164   if (!wait->table || (wait->table && wait->table->index >= N_POLL_ITEMS))
165     {
166       poll_wait_entry_table *new_entry_table = NULL;
167 
168       new_entry_table = (poll_wait_entry_table *)malloc(sizeof(poll_wait_entry_table));
169       if (new_entry_table == NULL)
170         {
171           return (poll_wait_node *)NULL;
172         }
173       new_entry_table->index = 0;
174 
175       new_entry_table->next = wait->table;
176       wait->table = new_entry_table;
177     }
178 
179   return wait->table->items + wait->table->index++;
180 }
181 
add_pollwait_queue(poll_wait_queue queue,poll_table * p)182 static void add_pollwait_queue(poll_wait_queue queue, poll_table *p)
183 {
184   unsigned long int_save;
185   poll_wait_head wait = p->wait;
186   poll_wait_node *new_node = get_poll_item(wait);
187   if (new_node != NULL)
188     {
189       new_node->entry = wait;
190       new_node->key = p->key;
191       new_node->wait_queue = queue;
192       spin_lock_irqsave(&queue->lock, int_save);
193       LOS_ListAdd(&queue->poll_queue, &new_node->queue_node);
194       spin_unlock_irqrestore(&queue->lock, int_save);
195     }
196 }
197 
wait_sem_time(poll_wait_head wait,const struct timespec * time_ptr)198 static int wait_sem_time(poll_wait_head wait, const struct timespec *time_ptr)
199 {
200   if (time_ptr != NULL)
201     {
202       return sem_timedwait(&wait->sem, time_ptr);
203     }
204   else
205     {
206       return poll_semtake(&wait->sem);
207     }
208 }
209 
file_poll(struct file * filep,poll_table * wait)210 static int file_poll(struct file *filep, poll_table *wait)
211 {
212   int ret = -ENOSYS;
213 
214   if (filep->ops != NULL && filep->ops->poll != NULL)
215     {
216       ret = filep->ops->poll(filep, wait);
217     }
218 
219   return ret;
220 }
221 
fdesc_poll(int fd,poll_table * wait)222 static int fdesc_poll(int fd, poll_table *wait)
223 {
224   struct file *filep = NULL;
225 
226   if (fd <= STDERR_FILENO && fd >= STDIN_FILENO) /* fd : [0,2] */
227     {
228       fd = ConsoleUpdateFd();
229       if (fd < 0)
230         {
231           set_errno(EBADF);
232           return VFS_ERROR;
233         }
234     }
235 
236   /* Get the file pointer corresponding to this file descriptor */
237 
238   int ret = fs_getfilep(fd, &filep);
239   if (ret < 0)
240     {
241       /* The errno value has already been set */
242       int errorcode = get_errno();
243       return -errorcode;
244     }
245 
246   /* Let file_poll() do the rest */
247 
248   return file_poll(filep, wait);
249 }
250 
query_fd(int fd,poll_table * wait)251 static int query_fd(int fd, poll_table *wait)
252 {
253   /* Check for a valid file descriptor */
254 
255   if (fd >= CONFIG_NFILE_DESCRIPTORS)
256     {
257       /* Perform the socket ioctl */
258 
259 #if defined(LOSCFG_NET_LWIP_SACK)
260       if (fd < (CONFIG_NFILE_DESCRIPTORS + CONFIG_NSOCKET_DESCRIPTORS))
261         {
262           return socks_poll(fd, wait);
263         }
264       else
265 #endif
266         {
267           return -EBADF;
268         }
269     }
270 
271   return fdesc_poll(fd, wait);
272 }
273 
query_fds(struct pollfd * fds,nfds_t nfds,poll_table * wait)274 static int query_fds(struct pollfd *fds, nfds_t nfds, poll_table *wait)
275 {
276   unsigned int i;
277   int ret;
278   int count = 0;
279 
280   if ((nfds != 0 && fds == NULL) || wait == NULL || wait->wait == NULL)
281     {
282       set_errno(EINVAL);
283       return -1;
284     }
285 
286   for (i = 0; i < nfds; ++i)
287     {
288       struct pollfd *p_fds = &fds[i];
289       if (p_fds->fd < 0)
290         {
291           set_errno(EBADF);
292           return -1;
293         }
294 
295       wait->key = p_fds->events | POLLERR | POLLHUP;
296 
297       ret = query_fd(p_fds->fd, wait);
298       if (ret < 0)
299         {
300           set_errno(-ret);
301           return -1;
302         }
303 
304       p_fds->revents = (p_fds->events | POLLERR | POLLHUP) & (pollevent_t)ret;
305       if (p_fds->revents)
306         {
307           ++count;
308           set_add_poll_wait_flag(wait->wait, false);
309         }
310     }
311 
312   return count;
313 }
314 
315 /****************************************************************************
316  * Public Functions
317  ****************************************************************************/
318 
notify_poll_with_key(wait_queue_head_t * wait_address,pollevent_t key)319 void notify_poll_with_key(wait_queue_head_t *wait_address, pollevent_t key)
320 {
321   unsigned long int_save;
322   int failed_count = 0;
323   poll_wait_node *curr = NULL;
324 
325   spin_lock_irqsave(&wait_address->lock, int_save);
326   LOS_DL_LIST_FOR_EACH_ENTRY(curr, &(wait_address->poll_queue), poll_wait_node, queue_node)
327     {
328       poll_wait_entry *curr_entry = curr->entry;
329       if (!key || (key & curr->key))
330         {
331           if (poll_semgive(&curr_entry->sem) < 0)
332             {
333               failed_count++;
334             }
335         }
336     }
337   spin_unlock_irqrestore(&wait_address->lock, int_save);
338 
339   if (failed_count != 0)
340     {
341       PRINT_ERR("[%s] sem_post failed %d times\n", __FUNCTION__, failed_count);
342     }
343 }
344 
345 /* just for compatible */
346 
notify_poll(wait_queue_head_t * wait_address)347 void notify_poll(wait_queue_head_t *wait_address)
348 {
349   notify_poll_with_key(wait_address, 0);
350 }
351 
poll_wait(struct file * filp,wait_queue_head_t * wait_address,poll_table * p)352 void poll_wait(struct file *filp, wait_queue_head_t *wait_address, poll_table *p)
353 {
354   if (!wait_address || !p || !p->wait)
355     {
356       return;
357     }
358 
359   if (p->wait->add_queue_flag)
360     {
361       add_pollwait_queue(wait_address, p);
362     }
363 }
364 
365 /****************************************************************************
366  * Name: poll
367  *
368  * Description:
369  *   poll() waits for one of a set of file descriptors to become ready to
370  *   perform I/O.  If none of the events requested (and no error) has
371  *   occurred for any of  the  file  descriptors,  then  poll() blocks until
372  *   one of the events occurs.
373  *
374  * Input Parameters:
375  *   fds  - List of structures describing file descriptors to be monitored
376  *   nfds - The number of entries in the list
377  *   timeout - Specifies an upper limit on the time for which poll() will
378  *     block in milliseconds.  A negative value of timeout means an infinite
379  *     timeout.
380  *
381  * Returned Value:
382  *   On success, the number of structures that have non-zero revents fields.
383  *   A value of 0 indicates that the call timed out and no file descriptors
384  *   were ready.  On error, -1 is returned, and errno is set appropriately:
385  *
386  *   EBADF  - An invalid file descriptor was given in one of the sets.
387  *   EFAULT - The fds address is invalid
388  *   EINTR  - A signal occurred before any requested event.
389  *   EINVAL - The nfds value exceeds a system limit.
390  *   ENOMEM - There was no space to allocate internal data structures.
391  *   ENOSYS - One or more of the drivers supporting the file descriptor
392  *     does not support the poll method.
393  *
394  ****************************************************************************/
395 
poll(struct pollfd * fds,nfds_t nfds,int timeout)396 int poll(struct pollfd *fds, nfds_t nfds, int timeout)
397 {
398   poll_table wait_table;
399   poll_wait_entry wait_entry;
400   size_t start_ticks = 0;
401   int millisecs_left;
402   int ret = OK;
403   int count = 0;
404   int err = 0;
405   int temp = 0;
406 
407   wait_table.wait = &wait_entry;
408   wait_table.wait->table = NULL;
409   wait_table.wait->inline_index = 0;
410   if (sem_init(&wait_table.wait->sem, 0, 0) < 0)
411     {
412       set_errno(ENOMEM);
413       return -1;
414     }
415 
416   /* set wait flag */
417 
418   set_add_poll_wait_flag(wait_table.wait, ((timeout == 0) ? false : true));
419 
420   count = query_fds(fds, nfds, &wait_table);
421   if (count > 0)
422     {
423       ret = OK;
424       goto out;
425     }
426   else if (count < 0)
427     {
428       goto out;
429     }
430 
431   /* clear wait flag if no fd got on first query */
432 
433   set_add_poll_wait_flag(wait_table.wait, false);
434 
435   if (timeout > 0)
436     {
437       start_ticks = LOS_TickCountGet();
438     }
439 
440   millisecs_left = timeout;
441   while (count == 0)
442     {
443       if (timeout < 0)
444         {
445           ret = wait_sem_time(wait_table.wait, (const struct timespec *)NULL);
446         }
447       else if (timeout == 0 || millisecs_left <= 0)
448         {
449           ret = OK;
450           goto out;
451         }
452       else if (millisecs_left > 0)
453         {
454           struct timespec wait_time;
455           UINT64 curr_ticks;
456           int millisecs_last;
457 
458           curr_ticks = LOS_TickCountGet();
459           millisecs_last = (curr_ticks - start_ticks) * MSEC_PER_SEC / LOSCFG_BASE_CORE_TICK_PER_SECOND;
460           if (millisecs_last >= timeout)
461             {
462               ret = OK;
463               goto out;
464             }
465           else
466             {
467               millisecs_left = timeout - millisecs_last;
468             }
469 
470           wait_time.tv_sec  = millisecs_left / MSEC_PER_SEC;
471           wait_time.tv_nsec = (millisecs_left - MSEC_PER_SEC * wait_time.tv_sec) * NSEC_PER_MSEC;
472 
473           ret = wait_sem_time(wait_table.wait, &wait_time);
474           if (ret < 0)
475             {
476               err = get_errno();
477 
478               if (err == ETIMEDOUT)
479                 {
480                   ret = OK;
481                 }
482               else
483                 {
484                   ret = -err;
485                 }
486             }
487         }
488 
489       if (ret < 0)
490         {
491           goto out;
492         }
493       count = query_fds(fds, nfds, &wait_table);
494       if (err == ETIMEDOUT)
495         {
496           break;
497         }
498     }
499 
500 out:
501   temp = get_errno();
502   if (destroy_poll_wait(wait_table.wait) < 0)
503     {
504       temp = get_errno();
505     }
506 
507   if (temp != 0)
508     {
509       set_errno(temp);
510     }
511   if (ret < 0)
512     {
513       set_errno(-ret);
514       return -1;
515     }
516 
517   return count;
518 }
519 
520 #endif /* CONFIG_DISABLE_POLL */
521