|
Packit |
b5b901 |
/* Copyright libuv project contributors. All rights reserved.
|
|
Packit |
b5b901 |
*
|
|
Packit |
b5b901 |
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
Packit |
b5b901 |
* of this software and associated documentation files (the "Software"), to
|
|
Packit |
b5b901 |
* deal in the Software without restriction, including without limitation the
|
|
Packit |
b5b901 |
* rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
|
|
Packit |
b5b901 |
* sell copies of the Software, and to permit persons to whom the Software is
|
|
Packit |
b5b901 |
* furnished to do so, subject to the following conditions:
|
|
Packit |
b5b901 |
*
|
|
Packit |
b5b901 |
* The above copyright notice and this permission notice shall be included in
|
|
Packit |
b5b901 |
* all copies or substantial portions of the Software.
|
|
Packit |
b5b901 |
*
|
|
Packit |
b5b901 |
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
Packit |
b5b901 |
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
Packit |
b5b901 |
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
Packit |
b5b901 |
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
Packit |
b5b901 |
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
Packit |
b5b901 |
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
|
|
Packit |
b5b901 |
* IN THE SOFTWARE.
|
|
Packit |
b5b901 |
*/
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
#include "os390-syscalls.h"
|
|
Packit |
b5b901 |
#include <errno.h>
|
|
Packit |
b5b901 |
#include <stdlib.h>
|
|
Packit |
b5b901 |
#include <search.h>
|
|
Packit |
b5b901 |
#include <termios.h>
|
|
Packit |
b5b901 |
#include <sys/msg.h>
|
|
Packit |
b5b901 |
|
|
Packit Service |
e08953 |
#define CW_INTRPT 1
|
|
Packit |
b5b901 |
#define CW_CONDVAR 32
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
#pragma linkage(BPX4CTW, OS)
|
|
Packit |
b5b901 |
#pragma linkage(BPX1CTW, OS)
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
static int number_of_epolls;
|
|
Packit |
b5b901 |
static QUEUE global_epoll_queue;
|
|
Packit |
b5b901 |
static uv_mutex_t global_epoll_lock;
|
|
Packit |
b5b901 |
static uv_once_t once = UV_ONCE_INIT;
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
int scandir(const char* maindir, struct dirent*** namelist,
|
|
Packit |
b5b901 |
int (*filter)(const struct dirent*),
|
|
Packit |
b5b901 |
int (*compar)(const struct dirent**,
|
|
Packit |
b5b901 |
const struct dirent **)) {
|
|
Packit |
b5b901 |
struct dirent** nl;
|
|
Packit Service |
e08953 |
struct dirent** nl_copy;
|
|
Packit |
b5b901 |
struct dirent* dirent;
|
|
Packit |
b5b901 |
unsigned count;
|
|
Packit |
b5b901 |
size_t allocated;
|
|
Packit |
b5b901 |
DIR* mdir;
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
nl = NULL;
|
|
Packit |
b5b901 |
count = 0;
|
|
Packit |
b5b901 |
allocated = 0;
|
|
Packit |
b5b901 |
mdir = opendir(maindir);
|
|
Packit |
b5b901 |
if (!mdir)
|
|
Packit |
b5b901 |
return -1;
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
while (1) {
|
|
Packit |
b5b901 |
dirent = readdir(mdir);
|
|
Packit |
b5b901 |
if (!dirent)
|
|
Packit |
b5b901 |
break;
|
|
Packit |
b5b901 |
if (!filter || filter(dirent)) {
|
|
Packit |
b5b901 |
struct dirent* copy;
|
|
Packit |
b5b901 |
copy = uv__malloc(sizeof(*copy));
|
|
Packit Service |
e08953 |
if (!copy)
|
|
Packit Service |
e08953 |
goto error;
|
|
Packit |
b5b901 |
memcpy(copy, dirent, sizeof(*copy));
|
|
Packit |
b5b901 |
|
|
Packit Service |
e08953 |
nl_copy = uv__realloc(nl, sizeof(*copy) * (count + 1));
|
|
Packit Service |
e08953 |
if (nl_copy == NULL) {
|
|
Packit Service |
e08953 |
uv__free(copy);
|
|
Packit Service |
e08953 |
goto error;
|
|
Packit Service |
e08953 |
}
|
|
Packit Service |
e08953 |
|
|
Packit Service |
e08953 |
nl = nl_copy;
|
|
Packit |
b5b901 |
nl[count++] = copy;
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
qsort(nl, count, sizeof(struct dirent *),
|
|
Packit |
b5b901 |
(int (*)(const void *, const void *)) compar);
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
closedir(mdir);
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
*namelist = nl;
|
|
Packit |
b5b901 |
return count;
|
|
Packit Service |
e08953 |
|
|
Packit Service |
e08953 |
error:
|
|
Packit Service |
e08953 |
while (count > 0) {
|
|
Packit Service |
e08953 |
dirent = nl[--count];
|
|
Packit Service |
e08953 |
uv__free(dirent);
|
|
Packit Service |
e08953 |
}
|
|
Packit Service |
e08953 |
uv__free(nl);
|
|
Packit Service |
e08953 |
closedir(mdir);
|
|
Packit Service |
e08953 |
errno = ENOMEM;
|
|
Packit Service |
e08953 |
return -1;
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
static unsigned int next_power_of_two(unsigned int val) {
|
|
Packit |
b5b901 |
val -= 1;
|
|
Packit |
b5b901 |
val |= val >> 1;
|
|
Packit |
b5b901 |
val |= val >> 2;
|
|
Packit |
b5b901 |
val |= val >> 4;
|
|
Packit |
b5b901 |
val |= val >> 8;
|
|
Packit |
b5b901 |
val |= val >> 16;
|
|
Packit |
b5b901 |
val += 1;
|
|
Packit |
b5b901 |
return val;
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
static void maybe_resize(uv__os390_epoll* lst, unsigned int len) {
|
|
Packit |
b5b901 |
unsigned int newsize;
|
|
Packit |
b5b901 |
unsigned int i;
|
|
Packit |
b5b901 |
struct pollfd* newlst;
|
|
Packit |
b5b901 |
struct pollfd event;
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
if (len <= lst->size)
|
|
Packit |
b5b901 |
return;
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
if (lst->size == 0)
|
|
Packit |
b5b901 |
event.fd = -1;
|
|
Packit |
b5b901 |
else {
|
|
Packit |
b5b901 |
/* Extract the message queue at the end. */
|
|
Packit |
b5b901 |
event = lst->items[lst->size - 1];
|
|
Packit |
b5b901 |
lst->items[lst->size - 1].fd = -1;
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
newsize = next_power_of_two(len);
|
|
Packit Service |
e08953 |
newlst = uv__reallocf(lst->items, newsize * sizeof(lst->items[0]));
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
if (newlst == NULL)
|
|
Packit |
b5b901 |
abort();
|
|
Packit |
b5b901 |
for (i = lst->size; i < newsize; ++i)
|
|
Packit |
b5b901 |
newlst[i].fd = -1;
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
/* Restore the message queue at the end */
|
|
Packit |
b5b901 |
newlst[newsize - 1] = event;
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
lst->items = newlst;
|
|
Packit |
b5b901 |
lst->size = newsize;
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
static void init_message_queue(uv__os390_epoll* lst) {
|
|
Packit |
b5b901 |
struct {
|
|
Packit |
b5b901 |
long int header;
|
|
Packit |
b5b901 |
char body;
|
|
Packit |
b5b901 |
} msg;
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
/* initialize message queue */
|
|
Packit |
b5b901 |
lst->msg_queue = msgget(IPC_PRIVATE, 0600 | IPC_CREAT);
|
|
Packit |
b5b901 |
if (lst->msg_queue == -1)
|
|
Packit |
b5b901 |
abort();
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
/*
|
|
Packit |
b5b901 |
On z/OS, the message queue will be affiliated with the process only
|
|
Packit |
b5b901 |
when a send is performed on it. Once this is done, the system
|
|
Packit |
b5b901 |
can be queried for all message queues belonging to our process id.
|
|
Packit |
b5b901 |
*/
|
|
Packit |
b5b901 |
msg.header = 1;
|
|
Packit |
b5b901 |
if (msgsnd(lst->msg_queue, &msg, sizeof(msg.body), 0) != 0)
|
|
Packit |
b5b901 |
abort();
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
/* Clean up the dummy message sent above */
|
|
Packit |
b5b901 |
if (msgrcv(lst->msg_queue, &msg, sizeof(msg.body), 0, 0) != sizeof(msg.body))
|
|
Packit |
b5b901 |
abort();
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
static void before_fork(void) {
|
|
Packit |
b5b901 |
uv_mutex_lock(&global_epoll_lock);
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
static void after_fork(void) {
|
|
Packit |
b5b901 |
uv_mutex_unlock(&global_epoll_lock);
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
static void child_fork(void) {
|
|
Packit |
b5b901 |
QUEUE* q;
|
|
Packit |
b5b901 |
uv_once_t child_once = UV_ONCE_INIT;
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
/* reset once */
|
|
Packit |
b5b901 |
memcpy(&once, &child_once, sizeof(child_once));
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
/* reset epoll list */
|
|
Packit |
b5b901 |
while (!QUEUE_EMPTY(&global_epoll_queue)) {
|
|
Packit |
b5b901 |
uv__os390_epoll* lst;
|
|
Packit |
b5b901 |
q = QUEUE_HEAD(&global_epoll_queue);
|
|
Packit |
b5b901 |
QUEUE_REMOVE(q);
|
|
Packit |
b5b901 |
lst = QUEUE_DATA(q, uv__os390_epoll, member);
|
|
Packit |
b5b901 |
uv__free(lst->items);
|
|
Packit |
b5b901 |
lst->items = NULL;
|
|
Packit |
b5b901 |
lst->size = 0;
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
uv_mutex_unlock(&global_epoll_lock);
|
|
Packit |
b5b901 |
uv_mutex_destroy(&global_epoll_lock);
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
static void epoll_init(void) {
|
|
Packit |
b5b901 |
QUEUE_INIT(&global_epoll_queue);
|
|
Packit |
b5b901 |
if (uv_mutex_init(&global_epoll_lock))
|
|
Packit |
b5b901 |
abort();
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
if (pthread_atfork(&before_fork, &after_fork, &child_fork))
|
|
Packit |
b5b901 |
abort();
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
uv__os390_epoll* epoll_create1(int flags) {
|
|
Packit |
b5b901 |
uv__os390_epoll* lst;
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
lst = uv__malloc(sizeof(*lst));
|
|
Packit |
b5b901 |
if (lst != NULL) {
|
|
Packit |
b5b901 |
/* initialize list */
|
|
Packit |
b5b901 |
lst->size = 0;
|
|
Packit |
b5b901 |
lst->items = NULL;
|
|
Packit |
b5b901 |
init_message_queue(lst);
|
|
Packit |
b5b901 |
maybe_resize(lst, 1);
|
|
Packit |
b5b901 |
lst->items[lst->size - 1].fd = lst->msg_queue;
|
|
Packit |
b5b901 |
lst->items[lst->size - 1].events = POLLIN;
|
|
Packit |
b5b901 |
lst->items[lst->size - 1].revents = 0;
|
|
Packit |
b5b901 |
uv_once(&once, epoll_init);
|
|
Packit |
b5b901 |
uv_mutex_lock(&global_epoll_lock);
|
|
Packit |
b5b901 |
QUEUE_INSERT_TAIL(&global_epoll_queue, &lst->member);
|
|
Packit |
b5b901 |
uv_mutex_unlock(&global_epoll_lock);
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
return lst;
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
int epoll_ctl(uv__os390_epoll* lst,
|
|
Packit |
b5b901 |
int op,
|
|
Packit |
b5b901 |
int fd,
|
|
Packit |
b5b901 |
struct epoll_event *event) {
|
|
Packit |
b5b901 |
uv_mutex_lock(&global_epoll_lock);
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
if (op == EPOLL_CTL_DEL) {
|
|
Packit |
b5b901 |
if (fd >= lst->size || lst->items[fd].fd == -1) {
|
|
Packit |
b5b901 |
uv_mutex_unlock(&global_epoll_lock);
|
|
Packit |
b5b901 |
errno = ENOENT;
|
|
Packit |
b5b901 |
return -1;
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
lst->items[fd].fd = -1;
|
|
Packit |
b5b901 |
} else if (op == EPOLL_CTL_ADD) {
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
/* Resizing to 'fd + 1' would expand the list to contain at least
|
|
Packit |
b5b901 |
* 'fd'. But we need to guarantee that the last index on the list
|
|
Packit |
b5b901 |
* is reserved for the message queue. So specify 'fd + 2' instead.
|
|
Packit |
b5b901 |
*/
|
|
Packit |
b5b901 |
maybe_resize(lst, fd + 2);
|
|
Packit |
b5b901 |
if (lst->items[fd].fd != -1) {
|
|
Packit |
b5b901 |
uv_mutex_unlock(&global_epoll_lock);
|
|
Packit |
b5b901 |
errno = EEXIST;
|
|
Packit |
b5b901 |
return -1;
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
lst->items[fd].fd = fd;
|
|
Packit |
b5b901 |
lst->items[fd].events = event->events;
|
|
Packit |
b5b901 |
lst->items[fd].revents = 0;
|
|
Packit |
b5b901 |
} else if (op == EPOLL_CTL_MOD) {
|
|
Packit |
b5b901 |
if (fd >= lst->size - 1 || lst->items[fd].fd == -1) {
|
|
Packit |
b5b901 |
uv_mutex_unlock(&global_epoll_lock);
|
|
Packit |
b5b901 |
errno = ENOENT;
|
|
Packit |
b5b901 |
return -1;
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
lst->items[fd].events = event->events;
|
|
Packit |
b5b901 |
lst->items[fd].revents = 0;
|
|
Packit |
b5b901 |
} else
|
|
Packit |
b5b901 |
abort();
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
uv_mutex_unlock(&global_epoll_lock);
|
|
Packit |
b5b901 |
return 0;
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
|
|
Packit Service |
e08953 |
#define EP_MAX_PFDS (ULONG_MAX / sizeof(struct pollfd))
|
|
Packit Service |
e08953 |
#define EP_MAX_EVENTS (INT_MAX / sizeof(struct epoll_event))
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
int epoll_wait(uv__os390_epoll* lst, struct epoll_event* events,
|
|
Packit |
b5b901 |
int maxevents, int timeout) {
|
|
Packit |
b5b901 |
nmsgsfds_t size;
|
|
Packit |
b5b901 |
struct pollfd* pfds;
|
|
Packit |
b5b901 |
int pollret;
|
|
Packit |
b5b901 |
int reventcount;
|
|
Packit |
b5b901 |
int nevents;
|
|
Packit Service |
e08953 |
struct pollfd msg_fd;
|
|
Packit Service |
e08953 |
int i;
|
|
Packit |
b5b901 |
|
|
Packit Service |
e08953 |
if (!lst || !lst->items || !events) {
|
|
Packit Service |
e08953 |
errno = EFAULT;
|
|
Packit Service |
e08953 |
return -1;
|
|
Packit Service |
e08953 |
}
|
|
Packit Service |
e08953 |
|
|
Packit Service |
e08953 |
if (lst->size > EP_MAX_PFDS) {
|
|
Packit Service |
e08953 |
errno = EINVAL;
|
|
Packit Service |
e08953 |
return -1;
|
|
Packit Service |
e08953 |
}
|
|
Packit Service |
e08953 |
|
|
Packit Service |
e08953 |
if (maxevents <= 0 || maxevents > EP_MAX_EVENTS) {
|
|
Packit Service |
e08953 |
errno = EINVAL;
|
|
Packit Service |
e08953 |
return -1;
|
|
Packit Service |
e08953 |
}
|
|
Packit Service |
e08953 |
|
|
Packit Service |
e08953 |
if (lst->size > 0)
|
|
Packit Service |
e08953 |
_SET_FDS_MSGS(size, 1, lst->size - 1);
|
|
Packit Service |
e08953 |
else
|
|
Packit Service |
e08953 |
_SET_FDS_MSGS(size, 0, 0);
|
|
Packit |
b5b901 |
pfds = lst->items;
|
|
Packit |
b5b901 |
pollret = poll(pfds, size, timeout);
|
|
Packit |
b5b901 |
if (pollret <= 0)
|
|
Packit |
b5b901 |
return pollret;
|
|
Packit |
b5b901 |
|
|
Packit Service |
e08953 |
assert(lst->size > 0);
|
|
Packit Service |
e08953 |
|
|
Packit |
b5b901 |
pollret = _NFDS(pollret) + _NMSGS(pollret);
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
reventcount = 0;
|
|
Packit |
b5b901 |
nevents = 0;
|
|
Packit Service |
e08953 |
msg_fd = pfds[lst->size - 1];
|
|
Packit Service |
e08953 |
for (i = 0;
|
|
Packit |
b5b901 |
i < lst->size && i < maxevents && reventcount < pollret; ++i) {
|
|
Packit |
b5b901 |
struct epoll_event ev;
|
|
Packit |
b5b901 |
struct pollfd* pfd;
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
pfd = &pfds[i];
|
|
Packit |
b5b901 |
if (pfd->fd == -1 || pfd->revents == 0)
|
|
Packit |
b5b901 |
continue;
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
ev.fd = pfd->fd;
|
|
Packit |
b5b901 |
ev.events = pfd->revents;
|
|
Packit Service |
e08953 |
ev.is_msg = 0;
|
|
Packit |
b5b901 |
if (pfd->revents & POLLIN && pfd->revents & POLLOUT)
|
|
Packit |
b5b901 |
reventcount += 2;
|
|
Packit |
b5b901 |
else if (pfd->revents & (POLLIN | POLLOUT))
|
|
Packit |
b5b901 |
++reventcount;
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
pfd->revents = 0;
|
|
Packit |
b5b901 |
events[nevents++] = ev;
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
|
|
Packit Service |
e08953 |
if (msg_fd.revents != 0 && msg_fd.fd != -1)
|
|
Packit Service |
e08953 |
if (i == lst->size)
|
|
Packit Service |
e08953 |
events[nevents - 1].is_msg = 1;
|
|
Packit Service |
e08953 |
|
|
Packit |
b5b901 |
return nevents;
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
int epoll_file_close(int fd) {
|
|
Packit |
b5b901 |
QUEUE* q;
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
uv_once(&once, epoll_init);
|
|
Packit |
b5b901 |
uv_mutex_lock(&global_epoll_lock);
|
|
Packit |
b5b901 |
QUEUE_FOREACH(q, &global_epoll_queue) {
|
|
Packit |
b5b901 |
uv__os390_epoll* lst;
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
lst = QUEUE_DATA(q, uv__os390_epoll, member);
|
|
Packit |
b5b901 |
if (fd < lst->size && lst->items != NULL && lst->items[fd].fd != -1)
|
|
Packit |
b5b901 |
lst->items[fd].fd = -1;
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
uv_mutex_unlock(&global_epoll_lock);
|
|
Packit |
b5b901 |
return 0;
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
void epoll_queue_close(uv__os390_epoll* lst) {
|
|
Packit |
b5b901 |
/* Remove epoll instance from global queue */
|
|
Packit |
b5b901 |
uv_mutex_lock(&global_epoll_lock);
|
|
Packit |
b5b901 |
QUEUE_REMOVE(&lst->member);
|
|
Packit |
b5b901 |
uv_mutex_unlock(&global_epoll_lock);
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
/* Free resources */
|
|
Packit |
b5b901 |
msgctl(lst->msg_queue, IPC_RMID, NULL);
|
|
Packit |
b5b901 |
lst->msg_queue = -1;
|
|
Packit |
b5b901 |
uv__free(lst->items);
|
|
Packit |
b5b901 |
lst->items = NULL;
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
int nanosleep(const struct timespec* req, struct timespec* rem) {
|
|
Packit |
b5b901 |
unsigned nano;
|
|
Packit |
b5b901 |
unsigned seconds;
|
|
Packit |
b5b901 |
unsigned events;
|
|
Packit |
b5b901 |
unsigned secrem;
|
|
Packit |
b5b901 |
unsigned nanorem;
|
|
Packit |
b5b901 |
int rv;
|
|
Packit Service |
e08953 |
int err;
|
|
Packit |
b5b901 |
int rsn;
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
nano = (int)req->tv_nsec;
|
|
Packit |
b5b901 |
seconds = req->tv_sec;
|
|
Packit Service |
e08953 |
events = CW_CONDVAR | CW_INTRPT;
|
|
Packit Service |
e08953 |
secrem = 0;
|
|
Packit Service |
e08953 |
nanorem = 0;
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
#if defined(_LP64)
|
|
Packit Service |
e08953 |
BPX4CTW(&seconds, &nano, &events, &secrem, &nanorem, &rv, &err, &rsn;;
|
|
Packit |
b5b901 |
#else
|
|
Packit Service |
e08953 |
BPX1CTW(&seconds, &nano, &events, &secrem, &nanorem, &rv, &err, &rsn;;
|
|
Packit |
b5b901 |
#endif
|
|
Packit |
b5b901 |
|
|
Packit Service |
e08953 |
/* Don't clobber errno unless BPX1CTW/BPX4CTW errored.
|
|
Packit Service |
e08953 |
* Don't leak EAGAIN, that just means the timeout expired.
|
|
Packit Service |
e08953 |
*/
|
|
Packit Service |
e08953 |
if (rv == -1)
|
|
Packit Service |
e08953 |
if (err == EAGAIN)
|
|
Packit Service |
e08953 |
rv = 0;
|
|
Packit Service |
e08953 |
else
|
|
Packit Service |
e08953 |
errno = err;
|
|
Packit |
b5b901 |
|
|
Packit Service |
e08953 |
if (rem != NULL && (rv == 0 || err == EINTR)) {
|
|
Packit |
b5b901 |
rem->tv_nsec = nanorem;
|
|
Packit |
b5b901 |
rem->tv_sec = secrem;
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
|
|
Packit Service |
e08953 |
return rv;
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
char* mkdtemp(char* path) {
|
|
Packit |
b5b901 |
static const char* tempchars =
|
|
Packit |
b5b901 |
"abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789";
|
|
Packit |
b5b901 |
static const size_t num_chars = 62;
|
|
Packit |
b5b901 |
static const size_t num_x = 6;
|
|
Packit |
b5b901 |
char *ep, *cp;
|
|
Packit |
b5b901 |
unsigned int tries, i;
|
|
Packit |
b5b901 |
size_t len;
|
|
Packit |
b5b901 |
uint64_t v;
|
|
Packit |
b5b901 |
int fd;
|
|
Packit |
b5b901 |
int retval;
|
|
Packit |
b5b901 |
int saved_errno;
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
len = strlen(path);
|
|
Packit |
b5b901 |
ep = path + len;
|
|
Packit |
b5b901 |
if (len < num_x || strncmp(ep - num_x, "XXXXXX", num_x)) {
|
|
Packit |
b5b901 |
errno = EINVAL;
|
|
Packit |
b5b901 |
return NULL;
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
fd = open("/dev/urandom", O_RDONLY);
|
|
Packit |
b5b901 |
if (fd == -1)
|
|
Packit |
b5b901 |
return NULL;
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
tries = TMP_MAX;
|
|
Packit |
b5b901 |
retval = -1;
|
|
Packit |
b5b901 |
do {
|
|
Packit |
b5b901 |
if (read(fd, &v, sizeof(v)) != sizeof(v))
|
|
Packit |
b5b901 |
break;
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
cp = ep - num_x;
|
|
Packit |
b5b901 |
for (i = 0; i < num_x; i++) {
|
|
Packit |
b5b901 |
*cp++ = tempchars[v % num_chars];
|
|
Packit |
b5b901 |
v /= num_chars;
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
if (mkdir(path, S_IRWXU) == 0) {
|
|
Packit |
b5b901 |
retval = 0;
|
|
Packit |
b5b901 |
break;
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
else if (errno != EEXIST)
|
|
Packit |
b5b901 |
break;
|
|
Packit |
b5b901 |
} while (--tries);
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
saved_errno = errno;
|
|
Packit |
b5b901 |
uv__close(fd);
|
|
Packit |
b5b901 |
if (tries == 0) {
|
|
Packit |
b5b901 |
errno = EEXIST;
|
|
Packit |
b5b901 |
return NULL;
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
if (retval == -1) {
|
|
Packit |
b5b901 |
errno = saved_errno;
|
|
Packit |
b5b901 |
return NULL;
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
return path;
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
ssize_t os390_readlink(const char* path, char* buf, size_t len) {
|
|
Packit |
b5b901 |
ssize_t rlen;
|
|
Packit |
b5b901 |
ssize_t vlen;
|
|
Packit |
b5b901 |
ssize_t plen;
|
|
Packit |
b5b901 |
char* delimiter;
|
|
Packit |
b5b901 |
char old_delim;
|
|
Packit |
b5b901 |
char* tmpbuf;
|
|
Packit |
b5b901 |
char realpathstr[PATH_MAX + 1];
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
tmpbuf = uv__malloc(len + 1);
|
|
Packit |
b5b901 |
if (tmpbuf == NULL) {
|
|
Packit |
b5b901 |
errno = ENOMEM;
|
|
Packit |
b5b901 |
return -1;
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
rlen = readlink(path, tmpbuf, len);
|
|
Packit |
b5b901 |
if (rlen < 0) {
|
|
Packit |
b5b901 |
uv__free(tmpbuf);
|
|
Packit |
b5b901 |
return rlen;
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
if (rlen < 3 || strncmp("/$", tmpbuf, 2) != 0) {
|
|
Packit |
b5b901 |
/* Straightforward readlink. */
|
|
Packit |
b5b901 |
memcpy(buf, tmpbuf, rlen);
|
|
Packit |
b5b901 |
uv__free(tmpbuf);
|
|
Packit |
b5b901 |
return rlen;
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
/*
|
|
Packit |
b5b901 |
* There is a parmlib variable at the beginning
|
|
Packit |
b5b901 |
* which needs interpretation.
|
|
Packit |
b5b901 |
*/
|
|
Packit |
b5b901 |
tmpbuf[rlen] = '\0';
|
|
Packit |
b5b901 |
delimiter = strchr(tmpbuf + 2, '/');
|
|
Packit |
b5b901 |
if (delimiter == NULL)
|
|
Packit |
b5b901 |
/* No slash at the end */
|
|
Packit |
b5b901 |
delimiter = strchr(tmpbuf + 2, '\0');
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
/* Read real path of the variable. */
|
|
Packit |
b5b901 |
old_delim = *delimiter;
|
|
Packit |
b5b901 |
*delimiter = '\0';
|
|
Packit |
b5b901 |
if (realpath(tmpbuf, realpathstr) == NULL) {
|
|
Packit |
b5b901 |
uv__free(tmpbuf);
|
|
Packit |
b5b901 |
return -1;
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
/* realpathstr is not guaranteed to end with null byte.*/
|
|
Packit |
b5b901 |
realpathstr[PATH_MAX] = '\0';
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
/* Reset the delimiter and fill up the buffer. */
|
|
Packit |
b5b901 |
*delimiter = old_delim;
|
|
Packit |
b5b901 |
plen = strlen(delimiter);
|
|
Packit |
b5b901 |
vlen = strlen(realpathstr);
|
|
Packit |
b5b901 |
rlen = plen + vlen;
|
|
Packit |
b5b901 |
if (rlen > len) {
|
|
Packit |
b5b901 |
uv__free(tmpbuf);
|
|
Packit |
b5b901 |
errno = ENAMETOOLONG;
|
|
Packit |
b5b901 |
return -1;
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
memcpy(buf, realpathstr, vlen);
|
|
Packit |
b5b901 |
memcpy(buf + vlen, delimiter, plen);
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
/* Done using temporary buffer. */
|
|
Packit |
b5b901 |
uv__free(tmpbuf);
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
return rlen;
|
|
Packit |
b5b901 |
}
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
|
|
Packit |
b5b901 |
size_t strnlen(const char* str, size_t maxlen) {
|
|
Packit |
b5b901 |
char* p = memchr(str, 0, maxlen);
|
|
Packit |
b5b901 |
if (p == NULL)
|
|
Packit |
b5b901 |
return maxlen;
|
|
Packit |
b5b901 |
else
|
|
Packit |
b5b901 |
return p - str;
|
|
Packit |
b5b901 |
}
|
|
Packit Service |
e08953 |
|
|
Packit Service |
e08953 |
|
|
Packit Service |
e08953 |
int sem_init(UV_PLATFORM_SEM_T* semid, int pshared, unsigned int value) {
|
|
Packit Service |
e08953 |
UNREACHABLE();
|
|
Packit Service |
e08953 |
}
|
|
Packit Service |
e08953 |
|
|
Packit Service |
e08953 |
|
|
Packit Service |
e08953 |
int sem_destroy(UV_PLATFORM_SEM_T* semid) {
|
|
Packit Service |
e08953 |
UNREACHABLE();
|
|
Packit Service |
e08953 |
}
|
|
Packit Service |
e08953 |
|
|
Packit Service |
e08953 |
|
|
Packit Service |
e08953 |
int sem_post(UV_PLATFORM_SEM_T* semid) {
|
|
Packit Service |
e08953 |
UNREACHABLE();
|
|
Packit Service |
e08953 |
}
|
|
Packit Service |
e08953 |
|
|
Packit Service |
e08953 |
|
|
Packit Service |
e08953 |
int sem_trywait(UV_PLATFORM_SEM_T* semid) {
|
|
Packit Service |
e08953 |
UNREACHABLE();
|
|
Packit Service |
e08953 |
}
|
|
Packit Service |
e08953 |
|
|
Packit Service |
e08953 |
|
|
Packit Service |
e08953 |
int sem_wait(UV_PLATFORM_SEM_T* semid) {
|
|
Packit Service |
e08953 |
UNREACHABLE();
|
|
Packit Service |
e08953 |
}
|