mpv/audio/out/ao_oss.c

401 lines
11 KiB
C

/*
* OSS audio output driver
*
* Original author: A'rpi
* Support for >2 output channels added 2001-11-25
* - Steve Davies <steve@daviesfam.org>
* Rozhuk Ivan <rozhuk.im@gmail.com> 2020-2023
*
* This file is part of mpv.
*
* mpv 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 Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* mpv 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 mpv. If not, see <http://www.gnu.org/licenses/>.
*/
#include <errno.h>
#include <fcntl.h>
#include <stdio.h>
#include <unistd.h>
#include <sys/ioctl.h>
#include <sys/soundcard.h>
#include <sys/stat.h>
#if defined(__DragonFly__) || defined(__FreeBSD__)
#include <sys/sysctl.h>
#endif
#include <sys/types.h>
#include "audio/format.h"
#include "common/msg.h"
#include "options/options.h"
#include "osdep/endian.h"
#include "osdep/io.h"
#include "ao.h"
#include "internal.h"
#ifndef AFMT_AC3
#define AFMT_AC3 -1
#endif
#define PATH_DEV_DSP "/dev/dsp"
#define PATH_DEV_MIXER "/dev/mixer"
struct priv {
int dsp_fd;
double bps; /* Bytes per second. */
};
/* like alsa except for 6.1 and 7.1, from pcm/matrix_map.h */
static const struct mp_chmap oss_layouts[MP_NUM_CHANNELS + 1] = {
{0}, /* empty */
MP_CHMAP_INIT_MONO, /* mono */
MP_CHMAP2(FL, FR), /* stereo */
MP_CHMAP3(FL, FR, LFE), /* 2.1 */
MP_CHMAP4(FL, FR, BL, BR), /* 4.0 */
MP_CHMAP5(FL, FR, BL, BR, FC), /* 5.0 */
MP_CHMAP6(FL, FR, BL, BR, FC, LFE), /* 5.1 */
MP_CHMAP7(FL, FR, BL, BR, FC, LFE, BC), /* 6.1 */
MP_CHMAP8(FL, FR, BL, BR, FC, LFE, SL, SR), /* 7.1 */
};
#if !defined(AFMT_S32_NE) && defined(AFMT_S32_LE) && defined(AFMT_S32_BE)
#define AFMT_S32_NE AFMT_S32MP_SELECT_LE_BE(AFMT_S32_LE, AFMT_S32_BE)
#endif
static const int format_table[][2] = {
{AFMT_U8, AF_FORMAT_U8},
{AFMT_S16_NE, AF_FORMAT_S16},
#ifdef AFMT_S32_NE
{AFMT_S32_NE, AF_FORMAT_S32},
#endif
#ifdef AFMT_FLOAT
{AFMT_FLOAT, AF_FORMAT_FLOAT},
#endif
#ifdef AFMT_MPEG
{AFMT_MPEG, AF_FORMAT_S_MP3},
#endif
{-1, -1}
};
#define MP_WARN_IOCTL_ERR(__ao) \
MP_WARN((__ao), "%s: ioctl() fail, err = %i: %s\n", \
__FUNCTION__, errno, strerror(errno))
static void uninit(struct ao *ao);
static void device_descr_get(size_t dev_idx, char *buf, size_t buf_size)
{
#if defined(__DragonFly__) || defined(__FreeBSD__)
char dev_path[32];
size_t tmp = (buf_size - 1);
snprintf(dev_path, sizeof(dev_path), "dev.pcm.%zu.%%desc", dev_idx);
if (sysctlbyname(dev_path, buf, &tmp, NULL, 0) != 0) {
tmp = 0;
}
buf[tmp] = 0x00;
#elif defined(SOUND_MIXER_INFO)
size_t tmp = 0;
char dev_path[32];
mixer_info mi;
snprintf(dev_path, sizeof(dev_path), PATH_DEV_MIXER"%zu", dev_idx);
int fd = open(dev_path, O_RDONLY);
if (ioctl(fd, SOUND_MIXER_INFO, &mi) == 0) {
strncpy(buf, mi.name, buf_size - 1);
tmp = (buf_size - 1);
}
close(fd);
buf[tmp] = 0x00;
#else
buf[0] = 0x00;
#endif
}
static int format2oss(int format)
{
for (size_t i = 0; format_table[i][0] != -1; i++) {
if (format_table[i][1] == format)
return format_table[i][0];
}
return -1;
}
static bool try_format(struct ao *ao, int *format)
{
struct priv *p = ao->priv;
int oss_format = format2oss(*format);
if (oss_format == -1 && af_fmt_is_spdif(*format))
oss_format = AFMT_AC3;
if (oss_format == -1) {
MP_VERBOSE(ao, "Unknown/not supported internal format: %s\n",
af_fmt_to_str(*format));
*format = 0;
return false;
}
return (ioctl(p->dsp_fd, SNDCTL_DSP_SETFMT, &oss_format) != -1);
}
static int init(struct ao *ao)
{
struct priv *p = ao->priv;
struct mp_chmap channels = ao->channels;
audio_buf_info info;
size_t i;
int format, samplerate, nchannels, reqchannels, trig = 0;
int best_sample_formats[AF_FORMAT_COUNT + 1];
const char *device = ((ao->device) ? ao->device : PATH_DEV_DSP);
/* Opening device. */
MP_VERBOSE(ao, "Using '%s' audio device.\n", device);
p->dsp_fd = open(device, (O_WRONLY | O_CLOEXEC));
if (p->dsp_fd < 0) {
MP_ERR(ao, "Can't open audio device %s: %s.\n",
device, mp_strerror(errno));
goto err_out;
}
/* Selecting sound format. */
format = af_fmt_from_planar(ao->format);
af_get_best_sample_formats(format, best_sample_formats);
for (i = 0; best_sample_formats[i]; i++) {
format = best_sample_formats[i];
if (try_format(ao, &format))
break;
}
if (!format) {
MP_ERR(ao, "Can't set sample format.\n");
goto err_out;
}
MP_VERBOSE(ao, "Sample format: %s\n", af_fmt_to_str(format));
/* Channels count. */
if (af_fmt_is_spdif(format)) {
/* Probably could be fixed by setting number of channels;
* needs testing. */
if (channels.num != 2) {
MP_ERR(ao, "Format %s not implemented.\n", af_fmt_to_str(format));
goto err_out;
}
} else {
struct mp_chmap_sel sel = {0};
for (i = 0; i < MP_ARRAY_SIZE(oss_layouts); i++) {
mp_chmap_sel_add_map(&sel, &oss_layouts[i]);
}
if (!ao_chmap_sel_adjust(ao, &sel, &channels))
goto err_out;
nchannels = reqchannels = channels.num;
if (ioctl(p->dsp_fd, SNDCTL_DSP_CHANNELS, &nchannels) == -1) {
MP_ERR(ao, "Failed to set audio device to %d channels.\n",
reqchannels);
goto err_out_ioctl;
}
if (nchannels != reqchannels) {
/* Update number of channels to OSS suggested value. */
if (!ao_chmap_sel_get_def(ao, &sel, &channels, nchannels))
goto err_out;
}
MP_VERBOSE(ao, "Using %d channels (requested: %d).\n",
channels.num, reqchannels);
}
/* Sample rate. */
samplerate = ao->samplerate;
if (ioctl(p->dsp_fd, SNDCTL_DSP_SPEED, &samplerate) == -1)
goto err_out_ioctl;
MP_VERBOSE(ao, "Using %d Hz samplerate.\n", samplerate);
/* Get buffer size. */
if (ioctl(p->dsp_fd, SNDCTL_DSP_GETOSPACE, &info) == -1)
goto err_out_ioctl;
/* See ao.c ao->sstride initializations and get_state(). */
ao->device_buffer = ((info.fragstotal * info.fragsize) /
af_fmt_to_bytes(format));
if (!af_fmt_is_planar(format)) {
ao->device_buffer /= channels.num;
}
/* Do not start playback after data written. */
if (ioctl(p->dsp_fd, SNDCTL_DSP_SETTRIGGER, &trig) == -1)
goto err_out_ioctl;
/* Update sound params. */
ao->format = format;
ao->samplerate = samplerate;
ao->channels = channels;
p->bps = (channels.num * samplerate * af_fmt_to_bytes(format));
return 0;
err_out_ioctl:
MP_WARN_IOCTL_ERR(ao);
err_out:
uninit(ao);
return -1;
}
static void uninit(struct ao *ao)
{
struct priv *p = ao->priv;
if (p->dsp_fd == -1)
return;
ioctl(p->dsp_fd, SNDCTL_DSP_HALT, NULL);
close(p->dsp_fd);
p->dsp_fd = -1;
}
static int control(struct ao *ao, enum aocontrol cmd, void *arg)
{
struct priv *p = ao->priv;
float *vol = arg;
int v;
if (p->dsp_fd < 0)
return CONTROL_ERROR;
switch (cmd) {
case AOCONTROL_GET_VOLUME:
if (ioctl(p->dsp_fd, SNDCTL_DSP_GETPLAYVOL, &v) == -1) {
MP_WARN_IOCTL_ERR(ao);
return CONTROL_ERROR;
}
*vol = ((v & 0x00ff) + ((v & 0xff00) >> 8)) / 2.0;
return CONTROL_OK;
case AOCONTROL_SET_VOLUME:
v = ((int)*vol << 8) | (int)*vol;
if (ioctl(p->dsp_fd, SNDCTL_DSP_SETPLAYVOL, &v) == -1) {
MP_WARN_IOCTL_ERR(ao);
return CONTROL_ERROR;
}
return CONTROL_OK;
}
return CONTROL_UNKNOWN;
}
static void reset(struct ao *ao)
{
struct priv *p = ao->priv;
int trig = 0;
/* Clear buf and do not start playback after data written. */
if (ioctl(p->dsp_fd, SNDCTL_DSP_HALT, NULL) == -1 ||
ioctl(p->dsp_fd, SNDCTL_DSP_SETTRIGGER, &trig) == -1)
{
MP_WARN_IOCTL_ERR(ao);
MP_WARN(ao, "Force reinitialize audio device.\n");
uninit(ao);
init(ao);
}
}
static void start(struct ao *ao)
{
struct priv *p = ao->priv;
int trig = PCM_ENABLE_OUTPUT;
if (ioctl(p->dsp_fd, SNDCTL_DSP_SETTRIGGER, &trig) == -1) {
MP_WARN_IOCTL_ERR(ao);
return;
}
}
static bool audio_write(struct ao *ao, void **data, int samples)
{
struct priv *p = ao->priv;
ssize_t rc;
const size_t size = (samples * ao->sstride);
if (size == 0)
return true;
while ((rc = write(p->dsp_fd, data[0], size)) == -1) {
if (errno == EINTR)
continue;
MP_WARN(ao, "audio_write: write() fail, err = %i: %s.\n",
errno, strerror(errno));
return false;
}
if ((size_t)rc != size) {
MP_WARN(ao, "audio_write: unexpected partial write: required: %zu, written: %zu.\n",
size, (size_t)rc);
return false;
}
return true;
}
static void get_state(struct ao *ao, struct mp_pcm_state *state)
{
struct priv *p = ao->priv;
audio_buf_info info;
int odelay;
if (ioctl(p->dsp_fd, SNDCTL_DSP_GETOSPACE, &info) == -1 ||
ioctl(p->dsp_fd, SNDCTL_DSP_GETODELAY, &odelay) == -1)
{
MP_WARN_IOCTL_ERR(ao);
memset(state, 0x00, sizeof(struct mp_pcm_state));
state->delay = 0.0;
return;
}
state->free_samples = (info.bytes / ao->sstride);
state->queued_samples = (ao->device_buffer - state->free_samples);
state->delay = (odelay / p->bps);
state->playing = (state->queued_samples != 0);
}
static void list_devs(struct ao *ao, struct ao_device_list *list)
{
struct stat st;
char dev_path[32] = PATH_DEV_DSP, dev_descr[256] = "Default";
struct ao_device_desc dev = {.name = dev_path, .desc = dev_descr};
if (stat(PATH_DEV_DSP, &st) == 0) {
ao_device_list_add(list, ao, &dev);
}
/* Auto detect. */
for (size_t i = 0, fail_cnt = 0; fail_cnt < 8; i ++, fail_cnt ++) {
snprintf(dev_path, sizeof(dev_path), PATH_DEV_DSP"%zu", i);
if (stat(dev_path, &st) != 0)
continue;
device_descr_get(i, dev_descr, sizeof(dev_descr));
ao_device_list_add(list, ao, &dev);
fail_cnt = 0; /* Reset fail counter. */
}
}
const struct ao_driver audio_out_oss = {
.name = "oss",
.description = "OSS/ioctl audio output",
.init = init,
.uninit = uninit,
.control = control,
.reset = reset,
.start = start,
.write = audio_write,
.get_state = get_state,
.list_devs = list_devs,
.priv_size = sizeof(struct priv),
.priv_defaults = &(const struct priv) {
.dsp_fd = -1,
},
};