/* SSNES - A Super Nintendo Entertainment System (SNES) Emulator frontend for libsnes. * Copyright (C) 2010-2011 - Hans-Kristian Arntzen * * Some code herein may be based on code found in BSNES. * * SSNES is free software: you can redistribute it and/or modify it under the terms * of the GNU General Public License as published by the Free Software Found- * ation, either version 3 of the License, or (at your option) any later version. * * SSNES 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 General Public License for more details. * * You should have received a copy of the GNU General Public License along with SSNES. * If not, see . */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include "driver.h" #include "general.h" #include #ifdef HAVE_OSS_BSD #include #else #include #endif #include #include #include #include #include #ifdef HAVE_OSS_BSD #define DEFAULT_OSS_DEV "/dev/audio" #else #define DEFAULT_OSS_DEV "/dev/dsp" #endif static void *oss_init(const char *device, unsigned rate, unsigned latency) { int *fd = (int*)calloc(1, sizeof(int)); if (fd == NULL) return NULL; const char *oss_device = DEFAULT_OSS_DEV; if (device != NULL) oss_device = device; if ((*fd = open(oss_device, O_WRONLY)) < 0) { free(fd); return NULL; } int frags = (latency * rate * 4)/(1000 * (1 << 9)); int frag = (frags << 16) | 9; if (ioctl(*fd, SNDCTL_DSP_SETFRAGMENT, &frag) < 0) { close(*fd); free(fd); return NULL; } int channels = 2; int format = is_little_endian() ? AFMT_S16_LE : AFMT_S16_BE; if (ioctl(*fd, SNDCTL_DSP_CHANNELS, &channels) < 0) { close(*fd); free(fd); return NULL; } if (ioctl(*fd, SNDCTL_DSP_SETFMT, &format) < 0) { close(*fd); free(fd); return NULL; } if (ioctl(*fd, SNDCTL_DSP_SPEED, &rate) < 0) { close(*fd); free(fd); return NULL; } return fd; } static ssize_t oss_write(void *data, const void *buf, size_t size) { int *fd = (int*)data; if (size == 0) return 0; ssize_t ret; if ((ret = write(*fd, buf, size)) <= 0) { if ((fcntl(*fd, F_GETFL) & O_NONBLOCK) && errno == EAGAIN) return 0; return -1; } return ret; } static bool oss_stop(void *data) { int *fd = (int*)data; ioctl(*fd, SNDCTL_DSP_RESET, 0); return true; } static bool oss_start(void *data) { return true; } static void oss_set_nonblock_state(void *data, bool state) { int *fd = (int*)data; int rc; if (state) rc = fcntl(*fd, F_SETFL, fcntl(*fd, F_GETFL) | O_NONBLOCK); else rc = fcntl(*fd, F_SETFL, fcntl(*fd, F_GETFL) & (~O_NONBLOCK)); if (rc != 0) fprintf(stderr, "SSNES [ERROR]: Could not set nonblocking on OSS file descriptor. Will not be able to fast-forward.\n"); } static void oss_free(void *data) { int *fd = (int*)data; ioctl(*fd, SNDCTL_DSP_RESET, 0); close(*fd); free(fd); } const audio_driver_t audio_oss = { oss_init, oss_write, oss_stop, oss_start, oss_set_nonblock_state, oss_free, NULL, "oss" };