2006-08-18 19:46:20 +00:00
|
|
|
/* $Id$ */
|
|
|
|
|
|
|
|
|
|
/***
|
|
|
|
|
This file is part of PulseAudio.
|
2007-01-04 13:43:45 +00:00
|
|
|
|
2006-08-18 19:46:20 +00:00
|
|
|
PulseAudio is free software; you can redistribute it and/or modify
|
|
|
|
|
it under the terms of the GNU Lesser General Public License as
|
|
|
|
|
published by the Free Software Foundation; either version 2.1 of the
|
|
|
|
|
License, or (at your option) any later version.
|
2007-01-04 13:43:45 +00:00
|
|
|
|
2006-08-18 19:46:20 +00:00
|
|
|
PulseAudio is distributed in the hope that it will be useful, but
|
|
|
|
|
WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
|
Lesser General Public License for more details.
|
2007-01-04 13:43:45 +00:00
|
|
|
|
2006-08-18 19:46:20 +00:00
|
|
|
You should have received a copy of the GNU Lesser General Public
|
|
|
|
|
License along with PulseAudio; if not, write to the Free Software
|
|
|
|
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
|
|
|
|
|
USA.
|
|
|
|
|
***/
|
|
|
|
|
|
2006-08-19 17:27:27 +00:00
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
|
#include <config.h>
|
|
|
|
|
#endif
|
|
|
|
|
|
2006-08-18 19:46:20 +00:00
|
|
|
#include <assert.h>
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <unistd.h>
|
|
|
|
|
#include <fcntl.h>
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <errno.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
#include <sys/stat.h>
|
|
|
|
|
|
2006-08-22 11:41:14 +00:00
|
|
|
#ifdef HAVE_SYS_MMAN_H
|
|
|
|
|
#include <sys/mman.h>
|
|
|
|
|
#endif
|
|
|
|
|
|
2006-08-18 19:46:20 +00:00
|
|
|
#include <pulsecore/core-error.h>
|
|
|
|
|
#include <pulsecore/log.h>
|
|
|
|
|
#include <pulsecore/random.h>
|
2006-08-19 17:27:27 +00:00
|
|
|
#include <pulse/xmalloc.h>
|
2006-08-18 19:46:20 +00:00
|
|
|
|
|
|
|
|
#include "shm.h"
|
|
|
|
|
|
|
|
|
|
#if defined(__linux__) && !defined(MADV_REMOVE)
|
|
|
|
|
#define MADV_REMOVE 9
|
2007-01-04 13:43:45 +00:00
|
|
|
#endif
|
2006-08-18 19:46:20 +00:00
|
|
|
|
|
|
|
|
#define MAX_SHM_SIZE (1024*1024*20)
|
|
|
|
|
|
|
|
|
|
static char *segment_name(char *fn, size_t l, unsigned id) {
|
|
|
|
|
snprintf(fn, l, "/pulse-shm-%u", id);
|
|
|
|
|
return fn;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int pa_shm_create_rw(pa_shm *m, size_t size, int shared, mode_t mode) {
|
|
|
|
|
char fn[32];
|
|
|
|
|
int fd = -1;
|
2007-01-04 13:43:45 +00:00
|
|
|
|
2006-08-18 19:46:20 +00:00
|
|
|
assert(m);
|
|
|
|
|
assert(size > 0);
|
|
|
|
|
assert(size < MAX_SHM_SIZE);
|
|
|
|
|
assert(mode >= 0600);
|
|
|
|
|
|
|
|
|
|
if (!shared) {
|
|
|
|
|
m->id = 0;
|
|
|
|
|
m->size = size;
|
|
|
|
|
|
|
|
|
|
#ifdef MAP_ANONYMOUS
|
2006-08-19 01:15:22 +00:00
|
|
|
if ((m->ptr = mmap(NULL, m->size, PROT_READ|PROT_WRITE, MAP_ANONYMOUS|MAP_PRIVATE, -1, 0)) == MAP_FAILED) {
|
2006-08-18 21:38:40 +00:00
|
|
|
pa_log("mmap() failed: %s", pa_cstrerror(errno));
|
2006-08-18 19:46:20 +00:00
|
|
|
goto fail;
|
|
|
|
|
}
|
2006-08-19 17:27:27 +00:00
|
|
|
#elif defined(HAVE_POSIX_MEMALIGN)
|
|
|
|
|
{
|
|
|
|
|
int r;
|
2007-01-04 13:43:45 +00:00
|
|
|
|
2006-08-19 17:27:27 +00:00
|
|
|
if ((r = posix_memalign(&m->ptr, sysconf(_SC_PAGESIZE), size)) < 0) {
|
|
|
|
|
pa_log("posix_memalign() failed: %s", pa_cstrerror(r));
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
}
|
2006-08-18 19:46:20 +00:00
|
|
|
#else
|
|
|
|
|
m->ptr = pa_xmalloc(m->size);
|
|
|
|
|
#endif
|
2007-01-04 13:43:45 +00:00
|
|
|
|
2006-08-18 19:46:20 +00:00
|
|
|
m->do_unlink = 0;
|
2007-01-04 13:43:45 +00:00
|
|
|
|
2006-08-18 19:46:20 +00:00
|
|
|
} else {
|
2006-08-22 12:45:43 +00:00
|
|
|
#ifdef HAVE_SHM_OPEN
|
2006-08-18 19:46:20 +00:00
|
|
|
pa_random(&m->id, sizeof(m->id));
|
|
|
|
|
segment_name(fn, sizeof(fn), m->id);
|
2006-08-22 12:45:43 +00:00
|
|
|
|
2006-08-18 19:46:20 +00:00
|
|
|
if ((fd = shm_open(fn, O_RDWR|O_CREAT|O_EXCL, mode & 0444)) < 0) {
|
2006-08-18 21:38:40 +00:00
|
|
|
pa_log("shm_open() failed: %s", pa_cstrerror(errno));
|
2006-08-18 19:46:20 +00:00
|
|
|
goto fail;
|
|
|
|
|
}
|
2007-01-04 13:43:45 +00:00
|
|
|
|
2006-08-18 19:46:20 +00:00
|
|
|
if (ftruncate(fd, m->size = size) < 0) {
|
2006-08-18 21:38:40 +00:00
|
|
|
pa_log("ftruncate() failed: %s", pa_cstrerror(errno));
|
2006-08-18 19:46:20 +00:00
|
|
|
goto fail;
|
|
|
|
|
}
|
2007-01-04 13:43:45 +00:00
|
|
|
|
2006-08-18 19:46:20 +00:00
|
|
|
if ((m->ptr = mmap(NULL, m->size, PROT_READ|PROT_WRITE, MAP_SHARED, fd, 0)) == MAP_FAILED) {
|
2006-08-18 21:38:40 +00:00
|
|
|
pa_log("mmap() failed: %s", pa_cstrerror(errno));
|
2006-08-18 19:46:20 +00:00
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
close(fd);
|
|
|
|
|
m->do_unlink = 1;
|
2006-08-22 12:45:43 +00:00
|
|
|
#else
|
|
|
|
|
return -1;
|
|
|
|
|
#endif
|
2006-08-18 19:46:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
m->shared = shared;
|
2007-01-04 13:43:45 +00:00
|
|
|
|
2006-08-18 19:46:20 +00:00
|
|
|
return 0;
|
2007-01-04 13:43:45 +00:00
|
|
|
|
2006-08-18 19:46:20 +00:00
|
|
|
fail:
|
|
|
|
|
|
2006-08-22 12:45:43 +00:00
|
|
|
#ifdef HAVE_SHM_OPEN
|
2006-08-18 19:46:20 +00:00
|
|
|
if (fd >= 0) {
|
|
|
|
|
shm_unlink(fn);
|
|
|
|
|
close(fd);
|
|
|
|
|
}
|
2006-08-22 12:45:43 +00:00
|
|
|
#endif
|
2006-08-18 19:46:20 +00:00
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void pa_shm_free(pa_shm *m) {
|
|
|
|
|
assert(m);
|
2006-08-22 12:45:43 +00:00
|
|
|
assert(m->ptr);
|
2006-08-18 19:46:20 +00:00
|
|
|
assert(m->size > 0);
|
|
|
|
|
|
2006-08-22 12:45:43 +00:00
|
|
|
#ifdef MAP_FAILED
|
|
|
|
|
assert(m->ptr != MAP_FAILED);
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
if (!m->shared) {
|
|
|
|
|
#ifdef MAP_ANONYMOUS
|
|
|
|
|
if (munmap(m->ptr, m->size) < 0)
|
|
|
|
|
pa_log("munmap() failed: %s", pa_cstrerror(errno));
|
|
|
|
|
#elif defined(HAVE_POSIX_MEMALIGN)
|
2006-08-19 17:27:27 +00:00
|
|
|
free(m->ptr);
|
2006-08-22 12:45:43 +00:00
|
|
|
#else
|
2006-08-18 19:46:20 +00:00
|
|
|
pa_xfree(m->ptr);
|
2006-08-22 12:45:43 +00:00
|
|
|
#endif
|
|
|
|
|
} else {
|
|
|
|
|
#ifdef HAVE_SHM_OPEN
|
|
|
|
|
if (munmap(m->ptr, m->size) < 0)
|
|
|
|
|
pa_log("munmap() failed: %s", pa_cstrerror(errno));
|
2006-08-18 19:46:20 +00:00
|
|
|
|
2006-08-22 12:45:43 +00:00
|
|
|
if (m->do_unlink) {
|
|
|
|
|
char fn[32];
|
|
|
|
|
|
2006-08-28 19:16:00 +00:00
|
|
|
segment_name(fn, sizeof(fn), m->id);
|
2007-01-04 13:43:45 +00:00
|
|
|
|
2006-08-28 19:16:00 +00:00
|
|
|
if (shm_unlink(fn) < 0)
|
|
|
|
|
pa_log(" shm_unlink(%s) failed: %s", fn, pa_cstrerror(errno));
|
2006-08-22 12:45:43 +00:00
|
|
|
}
|
|
|
|
|
#else
|
|
|
|
|
/* We shouldn't be here without shm support */
|
|
|
|
|
assert(0);
|
|
|
|
|
#endif
|
|
|
|
|
}
|
2006-08-18 19:46:20 +00:00
|
|
|
|
|
|
|
|
memset(m, 0, sizeof(*m));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void pa_shm_punch(pa_shm *m, size_t offset, size_t size) {
|
|
|
|
|
void *ptr;
|
2007-01-04 13:43:45 +00:00
|
|
|
|
2006-08-18 19:46:20 +00:00
|
|
|
assert(m);
|
2006-08-22 12:45:43 +00:00
|
|
|
assert(m->ptr);
|
2006-08-18 19:46:20 +00:00
|
|
|
assert(m->size > 0);
|
2006-08-28 19:16:00 +00:00
|
|
|
assert(offset+size <= m->size);
|
2006-08-18 19:46:20 +00:00
|
|
|
|
2006-08-22 12:45:43 +00:00
|
|
|
#ifdef MAP_FAILED
|
|
|
|
|
assert(m->ptr != MAP_FAILED);
|
|
|
|
|
#endif
|
|
|
|
|
|
2006-08-18 19:46:20 +00:00
|
|
|
/* You're welcome to implement this as NOOP on systems that don't
|
|
|
|
|
* support it */
|
|
|
|
|
|
|
|
|
|
ptr = (uint8_t*) m->ptr + offset;
|
2007-01-04 13:43:45 +00:00
|
|
|
|
2006-08-18 19:46:20 +00:00
|
|
|
#ifdef __linux__
|
|
|
|
|
{
|
|
|
|
|
/* On Linux ptr must be page aligned */
|
|
|
|
|
long psz = sysconf(_SC_PAGESIZE);
|
|
|
|
|
unsigned o;
|
|
|
|
|
|
|
|
|
|
o = ((unsigned long) ptr) - ((((unsigned long) ptr)/psz) * psz);
|
2007-01-04 13:43:45 +00:00
|
|
|
|
2006-08-18 19:46:20 +00:00
|
|
|
if (o > 0) {
|
|
|
|
|
ptr = (uint8_t*) ptr + (psz - o);
|
|
|
|
|
size -= psz - o;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#endif
|
2007-01-04 13:43:45 +00:00
|
|
|
|
2006-08-18 19:46:20 +00:00
|
|
|
#ifdef MADV_REMOVE
|
|
|
|
|
if (madvise(ptr, size, MADV_REMOVE) >= 0)
|
|
|
|
|
return;
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#ifdef MADV_FREE
|
|
|
|
|
if (madvise(ptr, size, MADV_FREE) >= 0)
|
|
|
|
|
return;
|
2007-01-04 13:43:45 +00:00
|
|
|
#endif
|
|
|
|
|
|
2006-08-18 19:46:20 +00:00
|
|
|
#ifdef MADV_DONTNEED
|
|
|
|
|
madvise(ptr, size, MADV_DONTNEED);
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2006-08-22 12:45:43 +00:00
|
|
|
#ifdef HAVE_SHM_OPEN
|
|
|
|
|
|
2006-08-18 19:46:20 +00:00
|
|
|
int pa_shm_attach_ro(pa_shm *m, unsigned id) {
|
|
|
|
|
char fn[32];
|
|
|
|
|
int fd = -1;
|
|
|
|
|
struct stat st;
|
|
|
|
|
|
|
|
|
|
assert(m);
|
|
|
|
|
|
|
|
|
|
segment_name(fn, sizeof(fn), m->id = id);
|
|
|
|
|
|
|
|
|
|
if ((fd = shm_open(fn, O_RDONLY, 0)) < 0) {
|
2006-08-18 21:38:40 +00:00
|
|
|
pa_log("shm_open() failed: %s", pa_cstrerror(errno));
|
2006-08-18 19:46:20 +00:00
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (fstat(fd, &st) < 0) {
|
2006-08-18 21:38:40 +00:00
|
|
|
pa_log("fstat() failed: %s", pa_cstrerror(errno));
|
2006-08-18 19:46:20 +00:00
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (st.st_size <= 0 || st.st_size > MAX_SHM_SIZE) {
|
2006-08-18 21:38:40 +00:00
|
|
|
pa_log("Invalid shared memory segment size");
|
2006-08-18 19:46:20 +00:00
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
m->size = st.st_size;
|
2007-01-04 13:43:45 +00:00
|
|
|
|
2006-08-18 19:46:20 +00:00
|
|
|
if ((m->ptr = mmap(NULL, m->size, PROT_READ, MAP_SHARED, fd, 0)) == MAP_FAILED) {
|
2006-08-18 21:38:40 +00:00
|
|
|
pa_log("mmap() failed: %s", pa_cstrerror(errno));
|
2006-08-18 19:46:20 +00:00
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
m->do_unlink = 0;
|
|
|
|
|
m->shared = 1;
|
2007-01-04 13:43:45 +00:00
|
|
|
|
2006-08-18 19:46:20 +00:00
|
|
|
close(fd);
|
2007-01-04 13:43:45 +00:00
|
|
|
|
2006-08-18 19:46:20 +00:00
|
|
|
return 0;
|
2007-01-04 13:43:45 +00:00
|
|
|
|
2006-08-18 19:46:20 +00:00
|
|
|
fail:
|
|
|
|
|
if (fd >= 0)
|
|
|
|
|
close(fd);
|
|
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
2006-08-22 11:41:14 +00:00
|
|
|
|
|
|
|
|
#else /* HAVE_SHM_OPEN */
|
|
|
|
|
|
|
|
|
|
int pa_shm_attach_ro(pa_shm *m, unsigned id) {
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endif /* HAVE_SHM_OPEN */
|