#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> #include <stdlib.h> #include <pthread.h> static struct { ino_t ino; sem_t *sem; int refcnt; } *semtab; static int semcnt; static pthread_spinlock_t lock; static pthread_once_t once; static void init() { semtab = calloc(sizeof *semtab, SEM_NSEMS_MAX); } static sem_t *find_map(ino_t ino) { int i; for (i=0; i<SEM_NSEMS_MAX && semtab[i].ino != ino; i++); if (i==SEM_NSEMS_MAX) return 0; if (semtab[i].refcnt == INT_MAX) return (sem_t *)-1; semtab[i].refcnt++; return semtab[i].sem; } sem_t *sem_open(const char *name, int flags, ...) { va_list ap; mode_t mode; unsigned value; int fd, tfd, dir; sem_t newsem; void *map; char tmp[64]; struct timespec ts; struct stat st; int i; while (*name=='/') name++; if (strchr(name, '/')) { errno = EINVAL; return SEM_FAILED; } pthread_once(&once, init); if (!semtab) { errno = ENOMEM; return SEM_FAILED; } if (flags & O_CREAT) { 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; return SEM_FAILED; } sem_init(&newsem, 0, value); clock_gettime(CLOCK_REALTIME, &ts); snprintf(tmp, sizeof(tmp), "/dev/shm/%p-%p-%d-%d", &name, name, (int)getpid(), (int)ts.tv_nsec); tfd = open(tmp, O_CREAT|O_EXCL|O_RDWR, mode); if (tfd<0) return SEM_FAILED; dir = open("/dev/shm", O_DIRECTORY|O_RDONLY); if (dir<0 || write(tfd,&newsem,sizeof newsem)!=sizeof newsem) { if (dir >= 0) close(dir); close(tfd); unlink(tmp); return SEM_FAILED; } } flags &= ~O_ACCMODE; flags |= O_RDWR; pthread_spin_lock(&lock); for (;;) { if (!(flags & O_EXCL)) { fd = shm_open(name, flags&~O_CREAT, 0); if (fd >= 0 || errno != ENOENT) { if (flags & O_CREAT) { close(dir); close(tfd); unlink(tmp); } if (fd >= 0 && fstat(fd, &st) < 0) { close(fd); fd = -1; } if (fd < 0) { pthread_spin_unlock(&lock); return SEM_FAILED; } if ((map = find_map(st.st_ino))) { pthread_spin_unlock(&lock); close(fd); if (map == (sem_t *)-1) return SEM_FAILED; return map; } break; } } if (!(flags & O_CREAT)) { pthread_spin_unlock(&lock); return SEM_FAILED; } if (!linkat(AT_FDCWD, tmp, dir, name, 0)) { fd = tfd; close(dir); unlink(tmp); break; } if ((flags & O_EXCL) || errno != EEXIST) { close(dir); close(tfd); unlink(tmp); return SEM_FAILED; } } if (fstat(fd, &st) < 0) { pthread_spin_unlock(&lock); close(fd); return SEM_FAILED; } if (semcnt == SEM_NSEMS_MAX) { pthread_spin_unlock(&lock); close(fd); errno = EMFILE; return SEM_FAILED; } for (i=0; i<SEM_NSEMS_MAX && semtab[i].sem; i++); map = mmap(0, sizeof(sem_t), PROT_READ|PROT_WRITE, MAP_SHARED, fd, 0); close(fd); if (map == MAP_FAILED) { pthread_spin_unlock(&lock); return SEM_FAILED; } semtab[i].ino = st.st_ino; semtab[i].sem = map; semtab[i].refcnt = 1; pthread_spin_unlock(&lock); return map; } int sem_close(sem_t *sem) { int i; pthread_spin_lock(&lock); for (i=0; i<SEM_NSEMS_MAX && semtab[i].sem != sem; i++); if (!--semtab[i].refcnt) { semtab[i].sem = 0; semtab[i].ino = 0; } pthread_spin_unlock(&lock); return munmap(sem, sizeof *sem); }