2023-04-03 20:31:51 +02:00
|
|
|
/*
|
|
|
|
* SPDX-FileCopyrightText: 2023 Espressif Systems (Shanghai) CO LTD
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
*/
|
|
|
|
#include <pthread.h>
|
2023-11-27 14:32:52 +01:00
|
|
|
#include <dlfcn.h>
|
|
|
|
#include <sys/types.h>
|
2023-04-03 20:31:51 +02:00
|
|
|
#include "esp_err.h"
|
|
|
|
#include "errno.h"
|
|
|
|
|
|
|
|
/** This module addresses the FreeRTOS simulator's coexistence with linux system calls from user apps.
|
|
|
|
* It's only included when building without lwIP, so we need to use linux system's select() which would receive
|
|
|
|
* EINTR event on every FreeRTOS interrupt; we workaround this problem by wrapping select()
|
|
|
|
* to bypass and silence these events.
|
|
|
|
*/
|
2023-11-27 14:32:52 +01:00
|
|
|
typedef int (*select_func_t) (int fd, fd_set * rfds, fd_set * wfds, fd_set *efds, struct timeval *tval);
|
2023-04-03 20:31:51 +02:00
|
|
|
|
|
|
|
static inline int64_t get_us(void)
|
|
|
|
{
|
|
|
|
struct timespec spec;
|
|
|
|
clock_gettime(CLOCK_REALTIME, &spec);
|
|
|
|
return spec.tv_nsec / 1000 + spec.tv_sec * 1000000;
|
|
|
|
}
|
|
|
|
|
2023-11-27 14:32:52 +01:00
|
|
|
int select (int fd, fd_set * rfds, fd_set * wfds, fd_set *efds, struct timeval *tval)
|
2023-04-03 20:31:51 +02:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
struct timeval *tv = tval;
|
2023-05-15 10:28:54 +02:00
|
|
|
struct timeval timeval_local = {};
|
2023-04-03 20:31:51 +02:00
|
|
|
int64_t start = 0;
|
|
|
|
int64_t timeout_us = 0;
|
2023-11-27 14:32:52 +01:00
|
|
|
select_func_t real_select = (select_func_t) dlsym(RTLD_NEXT, "select");
|
2023-04-03 20:31:51 +02:00
|
|
|
if (tv != NULL) {
|
|
|
|
start = get_us();
|
|
|
|
timeout_us = tval->tv_sec * 1000000 + tval->tv_usec;
|
2023-05-15 10:28:54 +02:00
|
|
|
timeval_local.tv_sec = tval->tv_sec;
|
|
|
|
timeval_local.tv_usec = tval->tv_usec;
|
2023-04-03 20:31:51 +02:00
|
|
|
tv = &timeval_local; // this (tv != NULL) indicates that we should handle timeouts
|
|
|
|
}
|
2023-11-27 14:32:52 +01:00
|
|
|
while ((ret = real_select(fd, rfds, wfds, efds, tv)) < 0 && errno == EINTR) {
|
2023-04-03 20:31:51 +02:00
|
|
|
if (tv != NULL) {
|
|
|
|
int64_t now = get_us();
|
|
|
|
timeout_us -= now - start;
|
|
|
|
if (timeout_us < 0) {
|
|
|
|
errno = 0;
|
|
|
|
ret = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
start = now;
|
|
|
|
tv->tv_usec = timeout_us % 1000000;
|
|
|
|
tv->tv_sec = timeout_us / 1000000;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|