summaryrefslogtreecommitdiff
path: root/lib/librthread/rthread_sem.c
diff options
context:
space:
mode:
Diffstat (limited to 'lib/librthread/rthread_sem.c')
-rw-r--r--lib/librthread/rthread_sem.c190
1 files changed, 168 insertions, 22 deletions
diff --git a/lib/librthread/rthread_sem.c b/lib/librthread/rthread_sem.c
index b4a08140817..822fb5a96f8 100644
--- a/lib/librthread/rthread_sem.c
+++ b/lib/librthread/rthread_sem.c
@@ -1,6 +1,6 @@
-/* $OpenBSD: rthread_sem.c,v 1.9 2013/06/01 23:06:26 tedu Exp $ */
+/* $OpenBSD: rthread_sem.c,v 1.10 2013/11/18 23:10:48 tedu Exp $ */
/*
- * Copyright (c) 2004,2005 Ted Unangst <tedu@openbsd.org>
+ * Copyright (c) 2004,2005,2013 Ted Unangst <tedu@openbsd.org>
* All Rights Reserved.
*
* Permission to use, copy, modify, and distribute this software for any
@@ -15,15 +15,39 @@
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
*/
+#include <sys/param.h>
+#include <sys/types.h>
+#include <sys/mman.h>
+#include <sys/stat.h>
+#include <err.h>
+#include <errno.h>
+#include <fcntl.h>
+#include <sha2.h>
#include <stdlib.h>
+#include <stdio.h>
+#include <string.h>
#include <unistd.h>
-#include <errno.h>
#include <pthread.h>
#include "rthread.h"
+#define SHARED_IDENT ((void *)-1)
+
+/* 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 160
+
+/*
+ * Size of memory to be mmap()'ed by named semaphores.
+ * Should be >= SEM_PATH_SIZE and page-aligned.
+ */
+#define SEM_MMAP_SIZE PAGE_SIZE
+
/*
* Internal implementation of semaphores
*/
@@ -31,8 +55,14 @@ int
_sem_wait(sem_t sem, int tryonly, const struct timespec *abstime,
int *delayed_cancel)
{
+ void *ident = (void *)&sem->waitcount;
int r;
+ if (sem->shared) {
+ sem = sem->shared;
+ ident = SHARED_IDENT;
+ }
+
_spinlock(&sem->lock);
if (sem->value) {
sem->value--;
@@ -42,7 +72,7 @@ _sem_wait(sem_t sem, int tryonly, const struct timespec *abstime,
} else {
sem->waitcount++;
do {
- r = __thrsleep(&sem->waitcount, CLOCK_REALTIME |
+ r = __thrsleep(ident, CLOCK_REALTIME |
_USING_TICKETS, abstime, &sem->lock.ticket,
delayed_cancel);
_spinlock(&sem->lock);
@@ -63,12 +93,18 @@ _sem_wait(sem_t sem, int tryonly, const struct timespec *abstime,
int
_sem_post(sem_t sem)
{
+ void *ident = (void *)&sem->waitcount;
int rv = 0;
+ if (sem->shared) {
+ sem = sem->shared;
+ ident = SHARED_IDENT;
+ }
+
_spinlock(&sem->lock);
sem->value++;
if (sem->waitcount) {
- __thrwakeup(&sem->waitcount, 1);
+ __thrwakeup(ident, 1);
rv = 1;
}
_spinunlock(&sem->lock);
@@ -81,11 +117,39 @@ _sem_post(sem_t sem)
int
sem_init(sem_t *semp, int pshared, unsigned int value)
{
- sem_t sem;
+ sem_t sem, *sempshared;
+ int i, oerrno;
+ char name[SEM_RANDOM_NAME_LEN];
if (pshared) {
- errno = EPERM;
- return (-1);
+ while (1) {
+ 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);
+ if (sempshared) {
+ break;
+ }
+ if (errno == EEXIST)
+ continue;
+ if (errno != EINVAL && errno != EPERM)
+ errno = ENOSPC;
+ return (-1);
+ }
+
+ /* unnamed semaphore should not be opened twice */
+ if (sem_unlink(name) == -1) {
+ oerrno = errno;
+ sem_close(sempshared);
+ errno = oerrno;
+ return (-1);
+ }
+
+ sem = *sempshared;
+ free(sempshared);
+ sem->value = value;
+ *semp = sem;
+ return (0);
}
if (value > SEM_VALUE_MAX) {
@@ -108,20 +172,30 @@ sem_init(sem_t *semp, int pshared, unsigned int value)
int
sem_destroy(sem_t *semp)
{
+ sem_t sem;
+
if (!semp || !*semp) {
errno = EINVAL;
return (-1);
}
+ sem = *semp;
- if ((*semp)->waitcount) {
+ if (sem->waitcount) {
#define MSG "sem_destroy on semaphore with waiters!\n"
write(2, MSG, sizeof(MSG) - 1);
#undef MSG
errno = EBUSY;
return (-1);
}
- free(*semp);
+
+ if (sem->shared)
+ if (munmap(sem->shared, SEM_MMAP_SIZE) == -1) {
+ warn("sem_destroy: someone borked shared memory");
+ abort();
+ }
+
*semp = NULL;
+ free(sem);
return (0);
}
@@ -136,6 +210,9 @@ sem_getvalue(sem_t *semp, int *sval)
return (-1);
}
+ if (sem->shared)
+ sem = sem->shared;
+
_spinlock(&sem->lock);
*sval = sem->value;
_spinunlock(&sem->lock);
@@ -227,27 +304,96 @@ sem_trywait(sem_t *semp)
return (0);
}
-/* ARGSUSED */
+
+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 __unused, int oflag __unused, ...)
+sem_open(const char *name, int oflag, ...)
{
- errno = ENOSYS;
- return (SEM_FAILED);
+ char sempath[SEM_PATH_SIZE];
+ struct stat sb;
+ int fd, oerrno;
+ sem_t sem, shared;
+ sem_t *semp = SEM_FAILED;
+
+ if (oflag & ~(O_CREAT | O_EXCL)) {
+ errno = EINVAL;
+ return semp;
+ }
+
+ makesempath(name, sempath, sizeof(sempath));
+ fd = open(sempath, O_RDWR | O_NOFOLLOW | oflag, 0600);
+ if (fd == -1)
+ return semp;
+ if (fstat(fd, &sb) == -1 || !S_ISREG(sb.st_mode)) {
+ close(fd);
+ errno = EINVAL;
+ return semp;
+ }
+ if (sb.st_uid != getuid()) {
+ close(fd);
+ errno = EPERM;
+ return semp;
+ }
+ if (sb.st_size < SEM_MMAP_SIZE)
+ if (ftruncate(fd, SEM_MMAP_SIZE) == -1) {
+ oerrno = errno;
+ close(fd);
+ errno = oerrno;
+ /* XXX can set errno to EIO, ENOTDIR... */
+ return semp;
+ }
+ shared = mmap(NULL, SEM_MMAP_SIZE, PROT_READ | PROT_WRITE,
+ MAP_FILE | MAP_SHARED | MAP_HASSEMAPHORE, fd, 0);
+ oerrno = errno;
+ close(fd);
+ if (shared == MAP_FAILED) {
+ errno = oerrno;
+ return semp;
+ }
+ semp = malloc(sizeof(*semp));
+ sem = calloc(1, sizeof(*sem));
+ if (!semp || !sem) {
+ free(sem);
+ free(semp);
+ munmap(shared, SEM_MMAP_SIZE);
+ errno = ENOSPC;
+ return SEM_FAILED;
+ }
+ *semp = sem;
+ sem->shared = shared;
+
+ return semp;
}
-/* ARGSUSED */
int
-sem_close(sem_t *sem __unused)
+sem_close(sem_t *semp)
{
- errno = ENOSYS;
- return (-1);
+ sem_t sem = *semp;
+
+ if (munmap(sem->shared, SEM_MMAP_SIZE) == -1) {
+ warn("sem_close: someone borked shared memory");
+ abort();
+ }
+ free(sem);
+ free(semp);
+
+ return (0);
}
-/* ARGSUSED */
int
-sem_unlink(const char *name __unused)
+sem_unlink(const char *name)
{
- errno = ENOSYS;
- return (-1);
+ char sempath[SEM_PATH_SIZE];
+
+ makesempath(name, sempath, sizeof(sempath));
+ return unlink(sempath);
}