/* * Copyright (c) 2021 Omar Polo * * Permission to use, copy, modify, and distribute this software for any * purpose with or without fee is hereby granted, provided that the above * copyright notice and this permission notice appear in all copies. * * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. */ #include "compat.h" #include #include #include #include #include #include #include "telescope.h" #include "utils.h" int mark_nonblock_cloexec(int fd) { int flags; if ((flags = fcntl(fd, F_GETFL)) == -1) return 0; if (fcntl(fd, F_SETFL, flags | O_NONBLOCK) == -1) return 0; if (fcntl(fd, F_SETFD, FD_CLOEXEC) == -1) return 0; return 1; } int has_suffix(const char *str, const char *sufx) { size_t l, s; l = strlen(str); s = strlen(sufx); if (l < s) return 0; return !strcmp(str + (l - s), sufx); } int unicode_isspace(uint32_t cp) { if (cp < INT8_MAX) return isspace(cp); return 0; } int unicode_isgraph(uint32_t cp) { if (cp < INT8_MAX) return isgraph(cp); return 1; } void imsg_event_add(struct imsgev *iev) { iev->events = EV_READ; if (iev->ibuf.w.queued) iev->events |= EV_WRITE; event_del(&iev->ev); event_set(&iev->ev, iev->ibuf.fd, iev->events, iev->handler, iev); event_add(&iev->ev, NULL); } int dispatch_imsg(struct imsgev *iev, short event, imsg_handlerfn **handlers, size_t size) { struct imsgbuf *ibuf; struct imsg imsg; size_t datalen, i; ssize_t n; ibuf = &iev->ibuf; if (event & EV_READ) { if ((n = imsg_read(ibuf)) == -1 && errno != EAGAIN) err(1, "imsg_read error"); if (n == 0) return -1; } if (event & EV_WRITE) { if ((n = msgbuf_write(&ibuf->w)) == -1 && errno != EAGAIN) err(1, "msgbuf_write"); if (n == 0) return -1; } for (;;) { if ((n = imsg_get(ibuf, &imsg)) == -1) _exit(1); if (n == 0) break; datalen = IMSG_DATA_SIZE(imsg); i = imsg.hdr.type; if (i > (size / sizeof(imsg_handlerfn*)) || handlers[i] == NULL) abort(); handlers[i](&imsg, datalen); imsg_free(&imsg); } imsg_event_add(iev); return 0; } int imsg_compose_event(struct imsgev *iev, uint16_t type, uint32_t peerid, pid_t pid, int fd, const void *data, uint16_t datalen) { int ret; if ((ret = imsg_compose(&iev->ibuf, type, peerid, pid, fd, data, datalen) != -1)) imsg_event_add(iev); return ret; } void * hash_alloc(size_t len, void *d) { if ((d = malloc(len)) == NULL) abort(); return d; } void * hash_calloc(size_t nmemb, size_t size, void *d) { if ((d = calloc(nmemb, size)) == NULL) abort(); return d; } void hash_free(void *ptr, void *d) { free(ptr); }