/* $OpenBSD: rthread_sem.c,v 1.31 2019/11/01 16:04:19 mpi Exp $ */ /* * Copyright (c) 2004,2005,2013 Ted Unangst * Copyright (c) 2018 Paul Irofti * All Rights Reserved. * * 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 #include #include #include #include #include #include #include #include #include #include #include #include #include #include "rthread.h" #include "cancel.h" /* in libc/include */ #include "synch.h" /* SHA256_DIGEST_STRING_LENGTH includes nul */ /* "/tmp/" + sha256 + ".sem" */ #define SEM_PATH_SIZE (5 + SHA256_DIGEST_STRING_LENGTH + 4) /* long enough to be hard to guess */ #define SEM_RANDOM_NAME_LEN 10 /* * Size of memory to be mmap()'ed by named semaphores. * Should be >= SEM_PATH_SIZE and page-aligned. */ #define SEM_MMAP_SIZE _thread_pagesize /* * Internal implementation of semaphores */ int _sem_wait(sem_t sem, int can_eintr, const struct timespec *abstime, int *delayed_cancel) { unsigned int val; int error = 0; atomic_inc_int(&sem->waitcount); for (;;) { while ((val = sem->value) > 0) { if (atomic_cas_uint(&sem->value, val, val - 1) == val) { membar_enter_after_atomic(); atomic_dec_int(&sem->waitcount); return (0); } } if (error) break; error = _twait(&sem->value, 0, CLOCK_REALTIME, abstime); /* ignore interruptions other than cancelation */ if ((error == ECANCELED && *delayed_cancel == 0) || (error == EINTR && !can_eintr) || error == EAGAIN) error = 0; } atomic_dec_int(&sem->waitcount); return (error); } /* always increment count */ int _sem_post(sem_t sem) { membar_exit_before_atomic(); atomic_inc_int(&sem->value); _wake(&sem->value, 1); return 0; } /* * exported semaphores */ int sem_init(sem_t *semp, int pshared, unsigned int value) { sem_t sem; if (value > SEM_VALUE_MAX) { errno = EINVAL; return (-1); } if (pshared) { errno = EPERM; return (-1); #ifdef notyet char name[SEM_RANDOM_NAME_LEN]; sem_t *sempshared; int i; for (;;) { for (i = 0; i < SEM_RANDOM_NAME_LEN - 1; i++) name[i] = arc4random_uniform(255) + 1; name[SEM_RANDOM_NAME_LEN - 1] = '\0'; sempshared = sem_open(name, O_CREAT | O_EXCL, 0, value); if (sempshared != SEM_FAILED) break; if (errno == EEXIST) continue; if (errno != EPERM) errno = ENOSPC; return (-1); } /* unnamed semaphore should not be opened twice */ if (sem_unlink(name) == -1) { sem_close(sempshared); errno = ENOSPC; return (-1); } *semp = *sempshared; free(sempshared); return (0); #endif } sem = calloc(1, sizeof(*sem)); if (!sem) { errno = ENOSPC; return (-1); } sem->value = value; *semp = sem; return (0); } int sem_destroy(sem_t *semp) { sem_t sem; if (!_threads_ready) /* for SEM_MMAP_SIZE */ _rthread_init(); if (!semp || !(sem = *semp)) { errno = EINVAL; return (-1); } if (sem->waitcount) { #define MSG "sem_destroy on semaphore with waiters!\n" write(2, MSG, sizeof(MSG) - 1); #undef MSG errno = EBUSY; return (-1); } *semp = NULL; if (sem->shared) munmap(sem, SEM_MMAP_SIZE); else free(sem); return (0); } int sem_getvalue(sem_t *semp, int *sval) { sem_t sem; if (!semp || !(sem = *semp)) { errno = EINVAL; return (-1); } *sval = sem->value; return (0); } int sem_post(sem_t *semp) { sem_t sem; if (!semp || !(sem = *semp)) { errno = EINVAL; return (-1); } _sem_post(sem); return (0); } int sem_wait(sem_t *semp) { struct tib *tib = TIB_GET(); pthread_t self; sem_t sem; int error; PREP_CANCEL_POINT(tib); if (!_threads_ready) _rthread_init(); self = tib->tib_thread; if (!semp || !(sem = *semp)) { errno = EINVAL; return (-1); } ENTER_DELAYED_CANCEL_POINT(tib, self); error = _sem_wait(sem, 1, NULL, &self->delayed_cancel); LEAVE_CANCEL_POINT_INNER(tib, error); if (error) { errno = error; _rthread_debug(1, "%s: v=%d errno=%d\n", __func__, sem->value, errno); return (-1); } return (0); } int sem_timedwait(sem_t *semp, const struct timespec *abstime) { struct tib *tib = TIB_GET(); pthread_t self; sem_t sem; int error; PREP_CANCEL_POINT(tib); if (!semp || !(sem = *semp) || abstime == NULL || abstime->tv_nsec < 0 || abstime->tv_nsec >= 1000000000) { errno = EINVAL; return (-1); } if (!_threads_ready) _rthread_init(); self = tib->tib_thread; ENTER_DELAYED_CANCEL_POINT(tib, self); error = _sem_wait(sem, 1, abstime, &self->delayed_cancel); LEAVE_CANCEL_POINT_INNER(tib, error); if (error) { errno = (error == EWOULDBLOCK) ? ETIMEDOUT : error; _rthread_debug(1, "%s: v=%d errno=%d\n", __func__, sem->value, errno); return (-1); } return (0); } int sem_trywait(sem_t *semp) { sem_t sem; unsigned int val; if (!semp || !(sem = *semp)) { errno = EINVAL; return (-1); } while ((val = sem->value) > 0) { if (atomic_cas_uint(&sem->value, val, val - 1) == val) { membar_enter_after_atomic(); return (0); } } errno = EAGAIN; _rthread_debug(1, "%s: v=%d errno=%d\n", __func__, sem->value, errno); return (-1); } static void makesempath(const char *origpath, char *sempath, size_t len) { char buf[SHA256_DIGEST_STRING_LENGTH]; SHA256Data(origpath, strlen(origpath), buf); snprintf(sempath, len, "/tmp/%s.sem", buf); } sem_t * sem_open(const char *name, int oflag, ...) { char sempath[SEM_PATH_SIZE]; struct stat sb; sem_t sem, *semp; unsigned int value = 0; int created = 0, fd; if (!_threads_ready) _rthread_init(); if (oflag & ~(O_CREAT | O_EXCL)) { errno = EINVAL; return (SEM_FAILED); } if (oflag & O_CREAT) { va_list ap; va_start(ap, oflag); /* 3rd parameter mode is not used */ va_arg(ap, mode_t); value = va_arg(ap, unsigned); va_end(ap); if (value > SEM_VALUE_MAX) { errno = EINVAL; return (SEM_FAILED); } } makesempath(name, sempath, sizeof(sempath)); fd = open(sempath, O_RDWR | O_NOFOLLOW | oflag, 0600); if (fd == -1) return (SEM_FAILED); if (fstat(fd, &sb) == -1 || !S_ISREG(sb.st_mode)) { close(fd); errno = EINVAL; return (SEM_FAILED); } if (sb.st_uid != geteuid()) { close(fd); errno = EPERM; return (SEM_FAILED); } if (sb.st_size != (off_t)SEM_MMAP_SIZE) { if (!(oflag & O_CREAT)) { close(fd); errno = EINVAL; return (SEM_FAILED); } if (sb.st_size != 0) { close(fd); errno = EINVAL; return (SEM_FAILED); } if (ftruncate(fd, SEM_MMAP_SIZE) == -1) { close(fd); errno = EINVAL; return (SEM_FAILED); } created = 1; } sem = mmap(NULL, SEM_MMAP_SIZE, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0); close(fd); if (sem == MAP_FAILED) { errno = EINVAL; return (SEM_FAILED); } semp = malloc(sizeof(*semp)); if (!semp) { munmap(sem, SEM_MMAP_SIZE); errno = ENOSPC; return (SEM_FAILED); } if (created) { sem->value = value; sem->shared = 1; } *semp = sem; return (semp); } int sem_close(sem_t *semp) { sem_t sem; if (!semp || !(sem = *semp) || !sem->shared) { errno = EINVAL; return (-1); } *semp = NULL; munmap(sem, SEM_MMAP_SIZE); free(semp); return (0); } int sem_unlink(const char *name) { char sempath[SEM_PATH_SIZE]; makesempath(name, sempath, sizeof(sempath)); return (unlink(sempath)); }