sem_open.c 4.1 KB
Newer Older
R
Rich Felker 已提交
1 2 3 4 5 6 7 8 9 10 11
#include <semaphore.h>
#include <sys/mman.h>
#include <limits.h>
#include <fcntl.h>
#include <unistd.h>
#include <string.h>
#include <stdarg.h>
#include <errno.h>
#include <time.h>
#include <stdio.h>
#include <sys/stat.h>
12 13
#include <stdlib.h>
#include <pthread.h>
R
Rich Felker 已提交
14 15 16
#include "libc.h"

char *__shm_mapname(const char *, char *);
R
Rich Felker 已提交
17

18 19 20 21 22
static struct {
	ino_t ino;
	sem_t *sem;
	int refcnt;
} *semtab;
23
static volatile int lock[2];
24

R
Rich Felker 已提交
25
#define FLAGS (O_RDWR|O_NOFOLLOW|O_CLOEXEC|O_NONBLOCK)
R
Rich Felker 已提交
26 27 28 29 30 31

sem_t *sem_open(const char *name, int flags, ...)
{
	va_list ap;
	mode_t mode;
	unsigned value;
32
	int fd, i, e, slot, first=1, cnt, cs;
R
Rich Felker 已提交
33 34 35 36
	sem_t newsem;
	void *map;
	char tmp[64];
	struct timespec ts;
37
	struct stat st;
R
Rich Felker 已提交
38
	char buf[NAME_MAX+10];
R
Rich Felker 已提交
39

R
Rich Felker 已提交
40
	if (!(name = __shm_mapname(name, buf)))
R
Rich Felker 已提交
41 42
		return SEM_FAILED;

R
Rich Felker 已提交
43 44 45 46
	LOCK(lock);
	/* Allocate table if we don't have one yet */
	if (!semtab && !(semtab = calloc(sizeof *semtab, SEM_NSEMS_MAX))) {
		UNLOCK(lock);
47 48 49
		return SEM_FAILED;
	}

R
Rich Felker 已提交
50 51 52 53 54 55 56
	/* Reserve a slot in case this semaphore is not mapped yet;
	 * this is necessary because there is no way to handle
	 * failures after creation of the file. */
	slot = -1;
	for (cnt=i=0; i<SEM_NSEMS_MAX; i++) {
		cnt += semtab[i].refcnt;
		if (!semtab[i].sem && slot < 0) slot = i;
R
Rich Felker 已提交
57
	}
R
Rich Felker 已提交
58 59 60 61 62 63 64 65 66
	/* Avoid possibility of overflow later */
	if (cnt == INT_MAX || slot < 0) {
		errno = EMFILE;
		UNLOCK(lock);
		return SEM_FAILED;
	}
	/* Dummy pointer to make a reservation */
	semtab[slot].sem = (sem_t *)-1;
	UNLOCK(lock);
R
Rich Felker 已提交
67

R
Rich Felker 已提交
68
	flags &= (O_CREAT|O_EXCL);
R
Rich Felker 已提交
69

70 71
	pthread_setcancelstate(PTHREAD_CANCEL_DISABLE, &cs);

R
Rich Felker 已提交
72 73 74 75
	/* Early failure check for exclusive open; otherwise the case
	 * where the semaphore already exists is expensive. */
	if (flags == (O_CREAT|O_EXCL) && access(name, F_OK) == 0) {
		errno = EEXIST;
76
		goto fail;
R
Rich Felker 已提交
77
	}
78

R
Rich Felker 已提交
79
	for (;;) {
R
Rich Felker 已提交
80 81 82 83 84
		/* If exclusive mode is not requested, try opening an
		 * existing file first and fall back to creation. */
		if (flags != (O_CREAT|O_EXCL)) {
			fd = open(name, FLAGS);
			if (fd >= 0) {
85 86
				if (fstat(fd, &st) < 0 ||
				    (map = mmap(0, sizeof(sem_t), PROT_READ|PROT_WRITE, MAP_SHARED, fd, 0)) == MAP_FAILED) {
87
					close(fd);
88
					goto fail;
89
				}
R
Rich Felker 已提交
90
				close(fd);
R
Rich Felker 已提交
91 92
				break;
			}
R
Rich Felker 已提交
93
			if (errno != ENOENT)
94
				goto fail;
R
Rich Felker 已提交
95
		}
R
Rich Felker 已提交
96
		if (!(flags & O_CREAT))
97
			goto fail;
R
Rich Felker 已提交
98 99 100 101 102 103 104 105
		if (first) {
			first = 0;
			va_start(ap, flags);
			mode = va_arg(ap, mode_t) & 0666;
			value = va_arg(ap, unsigned);
			va_end(ap);
			if (value > SEM_VALUE_MAX) {
				errno = EINVAL;
106
				goto fail;
R
Rich Felker 已提交
107 108
			}
			sem_init(&newsem, 1, value);
109
		}
R
Rich Felker 已提交
110 111 112 113 114 115 116
		/* Create a temp file with the new semaphore contents
		 * and attempt to atomically link it as the new name */
		clock_gettime(CLOCK_REALTIME, &ts);
		snprintf(tmp, sizeof(tmp), "/dev/shm/tmp-%d", (int)ts.tv_nsec);
		fd = open(tmp, O_CREAT|O_EXCL|FLAGS, mode);
		if (fd < 0) {
			if (errno == EEXIST) continue;
117
			goto fail;
R
Rich Felker 已提交
118
		}
R
Rich Felker 已提交
119 120 121
		if (write(fd, &newsem, sizeof newsem) != sizeof newsem || fstat(fd, &st) < 0 ||
		    (map = mmap(0, sizeof(sem_t), PROT_READ|PROT_WRITE, MAP_SHARED, fd, 0)) == MAP_FAILED) {
			close(fd);
R
Rich Felker 已提交
122
			unlink(tmp);
123
			goto fail;
R
Rich Felker 已提交
124
		}
125
		close(fd);
126
		e = link(tmp, name) ? errno : 0;
R
Rich Felker 已提交
127
		unlink(tmp);
128
		if (!e) break;
129
		munmap(map, sizeof(sem_t));
R
Rich Felker 已提交
130 131 132 133
		/* Failure is only fatal when doing an exclusive open;
		 * otherwise, next iteration will try to open the
		 * existing file. */
		if (e != EEXIST || flags == (O_CREAT|O_EXCL))
134
			goto fail;
135
	}
R
Rich Felker 已提交
136 137 138 139 140 141 142 143

	/* See if the newly mapped semaphore is already mapped. If
	 * so, unmap the new mapping and use the existing one. Otherwise,
	 * add it to the table of mapped semaphores. */
	LOCK(lock);
	for (i=0; i<SEM_NSEMS_MAX && semtab[i].ino != st.st_ino; i++);
	if (i<SEM_NSEMS_MAX) {
		munmap(map, sizeof(sem_t));
144 145 146
		semtab[slot].sem = 0;
		slot = i;
		map = semtab[i].sem;
147
	}
148
	semtab[slot].refcnt++;
R
Rich Felker 已提交
149 150 151
	semtab[slot].sem = map;
	semtab[slot].ino = st.st_ino;
	UNLOCK(lock);
152
	pthread_setcancelstate(cs, 0);
R
Rich Felker 已提交
153
	return map;
154 155 156

fail:
	pthread_setcancelstate(cs, 0);
157 158 159
	LOCK(lock);
	semtab[slot].sem = 0;
	UNLOCK(lock);
160
	return SEM_FAILED;
R
Rich Felker 已提交
161
}
162 163 164 165

int sem_close(sem_t *sem)
{
	int i;
R
Rich Felker 已提交
166
	LOCK(lock);
167 168 169 170 171
	for (i=0; i<SEM_NSEMS_MAX && semtab[i].sem != sem; i++);
	if (!--semtab[i].refcnt) {
		semtab[i].sem = 0;
		semtab[i].ino = 0;
	}
R
Rich Felker 已提交
172 173 174
	UNLOCK(lock);
	munmap(sem, sizeof *sem);
	return 0;
175
}