871 lines
25 KiB
C++
Raw Normal View History

2007-11-13 18:02:18 -08:00
/**
* OpenAL cross platform audio library
* Copyright (C) 1999-2007 by authors.
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library 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
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public
* License along with this library; if not, write to the
* Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
2007-11-13 18:02:18 -08:00
* Or go to http://www.gnu.org/copyleft/lgpl.html
*/
2008-01-16 14:09:04 -08:00
#include "config.h"
2007-11-13 18:02:18 -08:00
#include <sys/ioctl.h>
#include <sys/types.h>
#include <sys/time.h>
2007-11-13 18:02:18 -08:00
#include <sys/stat.h>
#include <fcntl.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
2007-11-13 18:02:18 -08:00
#include <memory.h>
#include <unistd.h>
#include <errno.h>
#include <math.h>
2012-09-14 02:14:29 -07:00
#include <vector>
#include <string>
#include <algorithm>
2007-11-13 18:02:18 -08:00
#include "alMain.h"
2012-09-14 02:14:29 -07:00
#include "alu.h"
#include "alconfig.h"
#include "ringbuffer.h"
2013-10-27 08:14:13 -07:00
#include "threads.h"
2013-10-28 12:05:33 -07:00
#include "compat.h"
2007-11-13 18:02:18 -08:00
#include "backends/base.h"
2007-11-13 18:02:18 -08:00
#include <sys/soundcard.h>
/*
* The OSS documentation talks about SOUND_MIXER_READ, but the header
* only contains MIXER_READ. Play safe. Same for WRITE.
*/
#ifndef SOUND_MIXER_READ
#define SOUND_MIXER_READ MIXER_READ
#endif
#ifndef SOUND_MIXER_WRITE
#define SOUND_MIXER_WRITE MIXER_WRITE
#endif
2016-01-21 01:05:29 -08:00
#if defined(SOUND_VERSION) && (SOUND_VERSION < 0x040000)
#define ALC_OSS_COMPAT
#endif
#ifndef SNDCTL_AUDIOINFO
#define ALC_OSS_COMPAT
#endif
/*
* FreeBSD strongly discourages the use of specific devices,
* such as those returned in oss_audioinfo.devnode
*/
#ifdef __FreeBSD__
#define ALC_OSS_DEVNODE_TRUC
#endif
namespace {
constexpr char DefaultName[] = "OSS Default";
const char *DefaultPlayback{"/dev/dsp"};
const char *DefaultCapture{"/dev/dsp"};
struct DevMap {
std::string name;
std::string device_name;
template<typename StrT0, typename StrT1>
DevMap(StrT0&& name_, StrT1&& devname_)
: name{std::forward<StrT0>(name_)}, device_name{std::forward<StrT1>(devname_)}
{ }
};
2007-11-13 18:02:18 -08:00
bool checkName(const std::vector<DevMap> &list, const std::string &name)
{
return std::find_if(list.cbegin(), list.cend(),
[&name](const DevMap &entry) -> bool
{ return entry.name == name; }
) != list.cend();
}
std::vector<DevMap> PlaybackDevices;
std::vector<DevMap> CaptureDevices;
2016-01-21 01:05:29 -08:00
#ifdef ALC_OSS_COMPAT
#define DSP_CAP_OUTPUT 0x00020000
#define DSP_CAP_INPUT 0x00010000
void ALCossListPopulate(std::vector<DevMap> *devlist, int type)
{
devlist->emplace_back(DefaultName, (type==DSP_CAP_INPUT) ? DefaultCapture : DefaultPlayback);
}
#else
#ifndef HAVE_STRNLEN
size_t my_strnlen(const char *str, size_t maxlen)
{
2018-11-12 23:49:11 -08:00
const char *end = static_cast<const char*>(memchr(str, 0, maxlen));
if(!end) return maxlen;
return end - str;
}
2018-11-12 23:49:11 -08:00
#define strnlen my_strnlen
#endif
void ALCossListAppend(std::vector<DevMap> *list, const char *handle, size_t hlen, const char *path, size_t plen)
2016-01-21 01:05:29 -08:00
{
#ifdef ALC_OSS_DEVNODE_TRUC
for(size_t i{0};i < plen;i++)
2016-01-21 01:05:29 -08:00
{
if(path[i] == '.')
{
2016-01-21 01:05:29 -08:00
if(strncmp(path + i, handle + hlen + i - plen, plen - i) == 0)
hlen = hlen + i - plen;
plen = i;
}
2016-01-21 01:05:29 -08:00
}
#endif
2016-01-21 01:05:29 -08:00
if(handle[0] == '\0')
{
handle = path;
hlen = plen;
}
2016-01-21 01:05:29 -08:00
std::string basename{handle, hlen};
basename.erase(std::find(basename.begin(), basename.end(), '\0'), basename.end());
std::string devname{path, plen};
devname.erase(std::find(devname.begin(), devname.end(), '\0'), devname.end());
auto iter = std::find_if(list->cbegin(), list->cend(),
[&devname](const DevMap &entry) -> bool
{ return entry.device_name == devname; }
);
if(iter != list->cend())
return;
int count{1};
std::string newname{basename};
while(checkName(PlaybackDevices, newname))
2016-01-21 01:05:29 -08:00
{
newname = basename;
newname += " #";
newname += std::to_string(++count);
}
2016-01-21 01:05:29 -08:00
list->emplace_back(std::move(newname), std::move(devname));
const DevMap &entry = list->back();
2016-01-21 01:05:29 -08:00
TRACE("Got device \"%s\", \"%s\"\n", entry.name.c_str(), entry.device_name.c_str());
}
void ALCossListPopulate(std::vector<DevMap> *devlist, int type_flag)
{
int fd{open("/dev/mixer", O_RDONLY)};
if(fd < 0)
{
TRACE("Could not open /dev/mixer: %s\n", strerror(errno));
goto done;
}
struct oss_sysinfo si;
2016-01-21 01:05:29 -08:00
if(ioctl(fd, SNDCTL_SYSINFO, &si) == -1)
{
TRACE("SNDCTL_SYSINFO failed: %s\n", strerror(errno));
2016-01-21 01:05:29 -08:00
goto done;
}
2016-01-21 01:05:29 -08:00
for(int i{0};i < si.numaudios;i++)
{
struct oss_audioinfo ai;
ai.dev = i;
2016-01-21 01:05:29 -08:00
if(ioctl(fd, SNDCTL_AUDIOINFO, &ai) == -1)
{
2016-01-21 01:05:29 -08:00
ERR("SNDCTL_AUDIOINFO (%d) failed: %s\n", i, strerror(errno));
continue;
}
if(!(ai.caps&type_flag) || ai.devnode[0] == '\0')
continue;
const char *handle;
size_t len;
if(ai.handle[0] != '\0')
{
2016-01-21 01:05:29 -08:00
len = strnlen(ai.handle, sizeof(ai.handle));
handle = ai.handle;
}
else
{
2016-01-21 01:05:29 -08:00
len = strnlen(ai.name, sizeof(ai.name));
handle = ai.name;
}
ALCossListAppend(devlist, handle, len, ai.devnode,
strnlen(ai.devnode, sizeof(ai.devnode)));
}
2016-01-21 01:05:29 -08:00
done:
if(fd >= 0)
close(fd);
fd = -1;
const char *defdev{(type_flag==DSP_CAP_INPUT) ? DefaultCapture : DefaultPlayback};
auto iter = std::find_if(devlist->cbegin(), devlist->cend(),
[defdev](const DevMap &entry) -> bool
{ return entry.device_name == defdev; }
);
if(iter == devlist->cend())
devlist->insert(devlist->begin(), DevMap{DefaultName, defdev});
else
{
DevMap entry{std::move(*iter)};
devlist->erase(iter);
devlist->insert(devlist->begin(), std::move(entry));
}
devlist->shrink_to_fit();
}
#endif
int log2i(ALCuint x)
2007-11-13 18:02:18 -08:00
{
int y = 0;
while (x > 1)
{
x >>= 1;
y++;
}
return y;
}
} // namespace
2018-11-12 23:49:11 -08:00
struct ALCplaybackOSS final : public ALCbackend {
int fd;
ALubyte *mix_data;
int data_size;
2017-01-10 03:17:23 -08:00
ATOMIC(ALenum) killNow;
althrd_t thread;
2018-11-12 23:49:11 -08:00
};
static int ALCplaybackOSS_mixerProc(void *ptr);
static void ALCplaybackOSS_Construct(ALCplaybackOSS *self, ALCdevice *device);
static void ALCplaybackOSS_Destruct(ALCplaybackOSS *self);
static ALCenum ALCplaybackOSS_open(ALCplaybackOSS *self, const ALCchar *name);
static ALCboolean ALCplaybackOSS_reset(ALCplaybackOSS *self);
static ALCboolean ALCplaybackOSS_start(ALCplaybackOSS *self);
static void ALCplaybackOSS_stop(ALCplaybackOSS *self);
static DECLARE_FORWARD2(ALCplaybackOSS, ALCbackend, ALCenum, captureSamples, ALCvoid*, ALCuint)
static DECLARE_FORWARD(ALCplaybackOSS, ALCbackend, ALCuint, availableSamples)
static DECLARE_FORWARD(ALCplaybackOSS, ALCbackend, ClockLatency, getClockLatency)
static DECLARE_FORWARD(ALCplaybackOSS, ALCbackend, void, lock)
static DECLARE_FORWARD(ALCplaybackOSS, ALCbackend, void, unlock)
DECLARE_DEFAULT_ALLOCATORS(ALCplaybackOSS)
DEFINE_ALCBACKEND_VTABLE(ALCplaybackOSS);
2018-11-12 23:49:11 -08:00
static void ALCplaybackOSS_Construct(ALCplaybackOSS *self, ALCdevice *device)
{
new (self) ALCplaybackOSS{};
ALCbackend_Construct(STATIC_CAST(ALCbackend, self), device);
SET_VTABLE2(ALCplaybackOSS, ALCbackend, self);
self->fd = -1;
ATOMIC_INIT(&self->killNow, AL_FALSE);
}
static void ALCplaybackOSS_Destruct(ALCplaybackOSS *self)
{
if(self->fd != -1)
close(self->fd);
self->fd = -1;
ALCbackend_Destruct(STATIC_CAST(ALCbackend, self));
self->~ALCplaybackOSS();
}
static int ALCplaybackOSS_mixerProc(void *ptr)
2007-11-13 18:02:18 -08:00
{
2018-11-12 23:49:11 -08:00
ALCplaybackOSS *self = static_cast<ALCplaybackOSS*>(ptr);
ALCdevice *device = STATIC_CAST(ALCbackend, self)->mDevice;
struct timeval timeout;
ALubyte *write_ptr;
ALint frame_size;
ALint to_write;
2009-10-19 00:33:00 -07:00
ssize_t wrote;
fd_set wfds;
int sret;
2007-11-13 18:02:18 -08:00
SetRTPriority();
2014-04-17 20:41:32 -07:00
althrd_setname(althrd_current(), MIXER_THREAD_NAME);
frame_size = FrameSizeFromDevFmt(device->FmtChans, device->FmtType, device->AmbiOrder);
2009-09-15 18:19:00 -07:00
ALCplaybackOSS_lock(self);
while(!ATOMIC_LOAD(&self->killNow, almemory_order_acquire) &&
ATOMIC_LOAD(&device->Connected, almemory_order_acquire))
2007-11-13 18:02:18 -08:00
{
FD_ZERO(&wfds);
FD_SET(self->fd, &wfds);
timeout.tv_sec = 1;
timeout.tv_usec = 0;
ALCplaybackOSS_unlock(self);
2018-11-12 23:49:11 -08:00
sret = select(self->fd+1, nullptr, &wfds, nullptr, &timeout);
ALCplaybackOSS_lock(self);
if(sret < 0)
{
if(errno == EINTR)
continue;
ERR("select failed: %s\n", strerror(errno));
aluHandleDisconnect(device, "Failed waiting for playback buffer: %s", strerror(errno));
break;
}
else if(sret == 0)
{
WARN("select timeout\n");
continue;
}
2007-11-13 18:02:18 -08:00
write_ptr = self->mix_data;
to_write = self->data_size;
aluMixData(device, write_ptr, to_write/frame_size);
while(to_write > 0 && !ATOMIC_LOAD_SEQ(&self->killNow))
2007-11-13 18:02:18 -08:00
{
wrote = write(self->fd, write_ptr, to_write);
2009-05-25 12:12:37 -07:00
if(wrote < 0)
{
if(errno == EAGAIN || errno == EWOULDBLOCK || errno == EINTR)
continue;
ERR("write failed: %s\n", strerror(errno));
aluHandleDisconnect(device, "Failed writing playback samples: %s",
strerror(errno));
break;
2009-05-25 12:12:37 -07:00
}
to_write -= wrote;
write_ptr += wrote;
2007-11-13 18:02:18 -08:00
}
}
ALCplaybackOSS_unlock(self);
2007-11-13 18:02:18 -08:00
return 0;
}
2007-12-16 19:34:52 -08:00
static ALCenum ALCplaybackOSS_open(ALCplaybackOSS *self, const ALCchar *name)
2007-11-13 18:02:18 -08:00
{
ALCdevice *device = STATIC_CAST(ALCbackend, self)->mDevice;
2007-11-13 18:02:18 -08:00
const char *devname{DefaultPlayback};
if(!name)
name = DefaultName;
else
{
if(PlaybackDevices.empty())
ALCossListPopulate(&PlaybackDevices, DSP_CAP_OUTPUT);
auto iter = std::find_if(PlaybackDevices.cbegin(), PlaybackDevices.cend(),
[&name](const DevMap &entry) -> bool
{ return entry.name == name; }
);
if(iter == PlaybackDevices.cend())
return ALC_INVALID_VALUE;
devname = iter->device_name.c_str();
}
2007-11-13 18:02:18 -08:00
self->fd = open(devname, O_WRONLY);
if(self->fd == -1)
2007-11-13 18:02:18 -08:00
{
ERR("Could not open %s: %s\n", devname, strerror(errno));
return ALC_INVALID_VALUE;
2007-11-13 18:02:18 -08:00
}
alstr_copy_cstr(&device->DeviceName, name);
return ALC_NO_ERROR;
}
static ALCboolean ALCplaybackOSS_reset(ALCplaybackOSS *self)
{
ALCdevice *device = STATIC_CAST(ALCbackend, self)->mDevice;
int numFragmentsLogSize;
int log2FragmentSize;
unsigned int periods;
audio_buf_info info;
ALuint frameSize;
int numChannels;
int ossFormat;
int ossSpeed;
2018-11-12 23:49:11 -08:00
const char *err;
switch(device->FmtType)
2007-11-13 18:02:18 -08:00
{
case DevFmtByte:
ossFormat = AFMT_S8;
break;
case DevFmtUByte:
2007-11-13 18:02:18 -08:00
ossFormat = AFMT_U8;
break;
case DevFmtUShort:
case DevFmtInt:
case DevFmtUInt:
case DevFmtFloat:
device->FmtType = DevFmtShort;
2009-08-15 13:20:35 -07:00
/* fall-through */
case DevFmtShort:
2007-11-13 18:02:18 -08:00
ossFormat = AFMT_S16_NE;
break;
}
periods = device->NumUpdates;
numChannels = ChannelsFromDevFmt(device->FmtChans, device->AmbiOrder);
2007-11-13 18:02:18 -08:00
ossSpeed = device->Frequency;
frameSize = numChannels * BytesFromDevFmt(device->FmtType);
/* According to the OSS spec, 16 bytes (log2(16)) is the minimum. */
log2FragmentSize = maxi(log2i(device->UpdateSize*frameSize), 4);
2007-11-13 18:02:18 -08:00
numFragmentsLogSize = (periods << 16) | log2FragmentSize;
2010-12-09 23:32:47 -08:00
#define CHECKERR(func) if((func) < 0) { \
err = #func; \
goto err; \
}
/* Don't fail if SETFRAGMENT fails. We can handle just about anything
* that's reported back via GETOSPACE */
ioctl(self->fd, SNDCTL_DSP_SETFRAGMENT, &numFragmentsLogSize);
CHECKERR(ioctl(self->fd, SNDCTL_DSP_SETFMT, &ossFormat));
CHECKERR(ioctl(self->fd, SNDCTL_DSP_CHANNELS, &numChannels));
CHECKERR(ioctl(self->fd, SNDCTL_DSP_SPEED, &ossSpeed));
CHECKERR(ioctl(self->fd, SNDCTL_DSP_GETOSPACE, &info));
2010-12-09 23:32:47 -08:00
if(0)
2007-11-13 18:02:18 -08:00
{
2010-12-09 23:32:47 -08:00
err:
2011-07-13 01:43:00 -07:00
ERR("%s failed: %s\n", err, strerror(errno));
2007-11-13 18:02:18 -08:00
return ALC_FALSE;
}
2010-12-09 23:32:47 -08:00
#undef CHECKERR
2007-11-13 18:02:18 -08:00
if((int)ChannelsFromDevFmt(device->FmtChans, device->AmbiOrder) != numChannels)
2007-11-13 18:02:18 -08:00
{
2011-07-13 01:43:00 -07:00
ERR("Failed to set %s, got %d channels instead\n", DevFmtChannelsString(device->FmtChans), numChannels);
return ALC_FALSE;
2007-11-13 18:02:18 -08:00
}
if(!((ossFormat == AFMT_S8 && device->FmtType == DevFmtByte) ||
(ossFormat == AFMT_U8 && device->FmtType == DevFmtUByte) ||
(ossFormat == AFMT_S16_NE && device->FmtType == DevFmtShort)))
2007-11-13 18:02:18 -08:00
{
2011-07-13 01:43:00 -07:00
ERR("Failed to set %s samples, got OSS format %#x\n", DevFmtTypeString(device->FmtType), ossFormat);
2007-11-13 18:02:18 -08:00
return ALC_FALSE;
}
device->Frequency = ossSpeed;
device->UpdateSize = info.fragsize / frameSize;
device->NumUpdates = info.fragments;
SetDefaultChannelOrder(device);
return ALC_TRUE;
}
static ALCboolean ALCplaybackOSS_start(ALCplaybackOSS *self)
{
ALCdevice *device = STATIC_CAST(ALCbackend, self)->mDevice;
self->data_size = device->UpdateSize * FrameSizeFromDevFmt(
device->FmtChans, device->FmtType, device->AmbiOrder
);
2018-11-12 23:49:11 -08:00
self->mix_data = static_cast<ALubyte*>(calloc(1, self->data_size));
2017-01-10 03:17:23 -08:00
ATOMIC_STORE_SEQ(&self->killNow, AL_FALSE);
if(althrd_create(&self->thread, ALCplaybackOSS_mixerProc, self) != althrd_success)
2007-11-13 18:02:18 -08:00
{
free(self->mix_data);
2018-11-12 23:49:11 -08:00
self->mix_data = nullptr;
2007-11-13 18:02:18 -08:00
return ALC_FALSE;
}
return ALC_TRUE;
}
static void ALCplaybackOSS_stop(ALCplaybackOSS *self)
2007-11-13 18:02:18 -08:00
{
int res;
if(ATOMIC_EXCHANGE_SEQ(&self->killNow, AL_TRUE))
return;
althrd_join(self->thread, &res);
2007-11-13 18:02:18 -08:00
if(ioctl(self->fd, SNDCTL_DSP_RESET) != 0)
2011-07-13 01:43:00 -07:00
ERR("Error resetting device: %s\n", strerror(errno));
free(self->mix_data);
2018-11-12 23:49:11 -08:00
self->mix_data = nullptr;
2007-11-13 18:02:18 -08:00
}
2018-11-12 23:49:11 -08:00
struct ALCcaptureOSS final : public ALCbackend {
int fd;
ll_ringbuffer_t *ring;
2017-01-10 03:17:23 -08:00
ATOMIC(ALenum) killNow;
althrd_t thread;
2018-11-12 23:49:11 -08:00
};
static int ALCcaptureOSS_recordProc(void *ptr);
static void ALCcaptureOSS_Construct(ALCcaptureOSS *self, ALCdevice *device);
static void ALCcaptureOSS_Destruct(ALCcaptureOSS *self);
static ALCenum ALCcaptureOSS_open(ALCcaptureOSS *self, const ALCchar *name);
static DECLARE_FORWARD(ALCcaptureOSS, ALCbackend, ALCboolean, reset)
static ALCboolean ALCcaptureOSS_start(ALCcaptureOSS *self);
static void ALCcaptureOSS_stop(ALCcaptureOSS *self);
static ALCenum ALCcaptureOSS_captureSamples(ALCcaptureOSS *self, ALCvoid *buffer, ALCuint samples);
static ALCuint ALCcaptureOSS_availableSamples(ALCcaptureOSS *self);
static DECLARE_FORWARD(ALCcaptureOSS, ALCbackend, ClockLatency, getClockLatency)
static DECLARE_FORWARD(ALCcaptureOSS, ALCbackend, void, lock)
static DECLARE_FORWARD(ALCcaptureOSS, ALCbackend, void, unlock)
DECLARE_DEFAULT_ALLOCATORS(ALCcaptureOSS)
DEFINE_ALCBACKEND_VTABLE(ALCcaptureOSS);
2018-11-12 23:49:11 -08:00
static void ALCcaptureOSS_Construct(ALCcaptureOSS *self, ALCdevice *device)
{
new (self) ALCcaptureOSS{};
ALCbackend_Construct(STATIC_CAST(ALCbackend, self), device);
SET_VTABLE2(ALCcaptureOSS, ALCbackend, self);
self->fd = -1;
self->ring = nullptr;
ATOMIC_INIT(&self->killNow, AL_FALSE);
}
static void ALCcaptureOSS_Destruct(ALCcaptureOSS *self)
{
if(self->fd != -1)
close(self->fd);
self->fd = -1;
ll_ringbuffer_free(self->ring);
self->ring = nullptr;
ALCbackend_Destruct(STATIC_CAST(ALCbackend, self));
self->~ALCcaptureOSS();
}
static int ALCcaptureOSS_recordProc(void *ptr)
{
2018-11-12 23:49:11 -08:00
ALCcaptureOSS *self = static_cast<ALCcaptureOSS*>(ptr);
ALCdevice *device = STATIC_CAST(ALCbackend, self)->mDevice;
struct timeval timeout;
int frame_size;
fd_set rfds;
ssize_t amt;
int sret;
SetRTPriority();
2014-12-21 10:38:40 -08:00
althrd_setname(althrd_current(), RECORD_THREAD_NAME);
frame_size = FrameSizeFromDevFmt(device->FmtChans, device->FmtType, device->AmbiOrder);
2017-01-10 03:17:23 -08:00
while(!ATOMIC_LOAD_SEQ(&self->killNow))
{
ll_ringbuffer_data_t vec[2];
FD_ZERO(&rfds);
FD_SET(self->fd, &rfds);
timeout.tv_sec = 1;
timeout.tv_usec = 0;
2018-11-12 23:49:11 -08:00
sret = select(self->fd+1, &rfds, nullptr, nullptr, &timeout);
if(sret < 0)
{
if(errno == EINTR)
continue;
ERR("select failed: %s\n", strerror(errno));
aluHandleDisconnect(device, "Failed to check capture samples: %s", strerror(errno));
break;
}
else if(sret == 0)
{
WARN("select timeout\n");
continue;
}
ll_ringbuffer_get_write_vector(self->ring, vec);
if(vec[0].len > 0)
{
amt = read(self->fd, vec[0].buf, vec[0].len*frame_size);
if(amt < 0)
{
ERR("read failed: %s\n", strerror(errno));
ALCcaptureOSS_lock(self);
aluHandleDisconnect(device, "Failed reading capture samples: %s", strerror(errno));
ALCcaptureOSS_unlock(self);
break;
}
ll_ringbuffer_write_advance(self->ring, amt/frame_size);
}
}
return 0;
}
static ALCenum ALCcaptureOSS_open(ALCcaptureOSS *self, const ALCchar *name)
2007-11-13 18:02:18 -08:00
{
ALCdevice *device = STATIC_CAST(ALCbackend, self)->mDevice;
2007-12-16 19:34:52 -08:00
const char *devname{DefaultCapture};
if(!name)
name = DefaultName;
else
{
if(CaptureDevices.empty())
ALCossListPopulate(&CaptureDevices, DSP_CAP_INPUT);
auto iter = std::find_if(CaptureDevices.cbegin(), CaptureDevices.cend(),
[&name](const DevMap &entry) -> bool
{ return entry.name == name; }
);
if(iter == CaptureDevices.cend())
return ALC_INVALID_VALUE;
devname = iter->device_name.c_str();
}
2007-12-16 19:34:52 -08:00
self->fd = open(devname, O_RDONLY);
if(self->fd == -1)
2007-12-16 19:34:52 -08:00
{
ERR("Could not open %s: %s\n", devname, strerror(errno));
return ALC_INVALID_VALUE;
2007-12-16 19:34:52 -08:00
}
int ossFormat{};
switch(device->FmtType)
2007-12-16 19:34:52 -08:00
{
case DevFmtByte:
ossFormat = AFMT_S8;
break;
case DevFmtUByte:
2007-12-16 19:34:52 -08:00
ossFormat = AFMT_U8;
break;
case DevFmtShort:
2007-12-16 19:34:52 -08:00
ossFormat = AFMT_S16_NE;
break;
case DevFmtUShort:
case DevFmtInt:
case DevFmtUInt:
case DevFmtFloat:
2012-02-29 21:45:44 -08:00
ERR("%s capture samples not supported\n", DevFmtTypeString(device->FmtType));
return ALC_INVALID_VALUE;
2007-12-16 19:34:52 -08:00
}
int periods{4};
int numChannels{ChannelsFromDevFmt(device->FmtChans, device->AmbiOrder)};
int frameSize{numChannels * BytesFromDevFmt(device->FmtType)};
int ossSpeed{static_cast<int>(device->Frequency)};
int log2FragmentSize{log2i(device->UpdateSize * device->NumUpdates *
frameSize / periods)};
2007-12-16 19:34:52 -08:00
/* according to the OSS spec, 16 bytes are the minimum */
log2FragmentSize = std::max(log2FragmentSize, 4);
int numFragmentsLogSize{(periods << 16) | log2FragmentSize};
2007-12-16 19:34:52 -08:00
audio_buf_info info;
const char *err;
2010-12-09 23:32:47 -08:00
#define CHECKERR(func) if((func) < 0) { \
err = #func; \
goto err; \
}
CHECKERR(ioctl(self->fd, SNDCTL_DSP_SETFRAGMENT, &numFragmentsLogSize));
CHECKERR(ioctl(self->fd, SNDCTL_DSP_SETFMT, &ossFormat));
CHECKERR(ioctl(self->fd, SNDCTL_DSP_CHANNELS, &numChannels));
CHECKERR(ioctl(self->fd, SNDCTL_DSP_SPEED, &ossSpeed));
CHECKERR(ioctl(self->fd, SNDCTL_DSP_GETISPACE, &info));
2010-12-09 23:32:47 -08:00
if(0)
2007-12-16 19:34:52 -08:00
{
2010-12-09 23:32:47 -08:00
err:
2011-07-13 01:43:00 -07:00
ERR("%s failed: %s\n", err, strerror(errno));
close(self->fd);
self->fd = -1;
return ALC_INVALID_VALUE;
2007-12-16 19:34:52 -08:00
}
2010-12-09 23:32:47 -08:00
#undef CHECKERR
2007-12-16 19:34:52 -08:00
if((int)ChannelsFromDevFmt(device->FmtChans, device->AmbiOrder) != numChannels)
2007-12-16 19:34:52 -08:00
{
2011-07-13 01:43:00 -07:00
ERR("Failed to set %s, got %d channels instead\n", DevFmtChannelsString(device->FmtChans), numChannels);
close(self->fd);
self->fd = -1;
return ALC_INVALID_VALUE;
2007-12-16 19:34:52 -08:00
}
if(!((ossFormat == AFMT_S8 && device->FmtType == DevFmtByte) ||
(ossFormat == AFMT_U8 && device->FmtType == DevFmtUByte) ||
(ossFormat == AFMT_S16_NE && device->FmtType == DevFmtShort)))
2007-12-16 19:34:52 -08:00
{
2011-07-13 01:43:00 -07:00
ERR("Failed to set %s samples, got OSS format %#x\n", DevFmtTypeString(device->FmtType), ossFormat);
close(self->fd);
self->fd = -1;
return ALC_INVALID_VALUE;
2007-12-16 19:34:52 -08:00
}
self->ring = ll_ringbuffer_create(device->UpdateSize*device->NumUpdates, frameSize, false);
if(!self->ring)
2007-12-16 19:34:52 -08:00
{
2011-07-13 01:43:00 -07:00
ERR("Ring buffer create failed\n");
close(self->fd);
self->fd = -1;
return ALC_OUT_OF_MEMORY;
2007-12-16 19:34:52 -08:00
}
alstr_copy_cstr(&device->DeviceName, name);
return ALC_NO_ERROR;
2007-11-13 18:02:18 -08:00
}
static ALCboolean ALCcaptureOSS_start(ALCcaptureOSS *self)
2007-11-13 18:02:18 -08:00
{
ATOMIC_STORE_SEQ(&self->killNow, AL_FALSE);
if(althrd_create(&self->thread, ALCcaptureOSS_recordProc, self) != althrd_success)
return ALC_FALSE;
return ALC_TRUE;
2007-11-13 18:02:18 -08:00
}
static void ALCcaptureOSS_stop(ALCcaptureOSS *self)
2007-11-13 18:02:18 -08:00
{
int res;
if(ATOMIC_EXCHANGE_SEQ(&self->killNow, AL_TRUE))
return;
althrd_join(self->thread, &res);
if(ioctl(self->fd, SNDCTL_DSP_RESET) != 0)
ERR("Error resetting device: %s\n", strerror(errno));
2007-11-13 18:02:18 -08:00
}
static ALCenum ALCcaptureOSS_captureSamples(ALCcaptureOSS *self, ALCvoid *buffer, ALCuint samples)
2007-11-13 18:02:18 -08:00
{
2018-11-12 23:49:11 -08:00
ll_ringbuffer_read(self->ring, static_cast<char*>(buffer), samples);
return ALC_NO_ERROR;
2007-11-13 18:02:18 -08:00
}
static ALCuint ALCcaptureOSS_availableSamples(ALCcaptureOSS *self)
{
return ll_ringbuffer_read_space(self->ring);
}
2018-11-12 23:49:11 -08:00
struct ALCossBackendFactory final : public ALCbackendFactory {
ALCossBackendFactory() noexcept;
};
ALCbackendFactory *ALCossBackendFactory_getFactory(void);
static ALCboolean ALCossBackendFactory_init(ALCossBackendFactory *self);
static void ALCossBackendFactory_deinit(ALCossBackendFactory *self);
static ALCboolean ALCossBackendFactory_querySupport(ALCossBackendFactory *self, ALCbackend_Type type);
static void ALCossBackendFactory_probe(ALCossBackendFactory *self, enum DevProbe type, al_string *outnames);
static ALCbackend* ALCossBackendFactory_createBackend(ALCossBackendFactory *self, ALCdevice *device, ALCbackend_Type type);
DEFINE_ALCBACKENDFACTORY_VTABLE(ALCossBackendFactory);
2018-11-12 23:49:11 -08:00
ALCossBackendFactory::ALCossBackendFactory() noexcept
: ALCbackendFactory{GET_VTABLE2(ALCossBackendFactory, ALCbackendFactory)}
{ }
ALCbackendFactory *ALCossBackendFactory_getFactory(void)
2007-11-13 18:02:18 -08:00
{
2018-11-12 23:49:11 -08:00
static ALCossBackendFactory factory{};
return STATIC_CAST(ALCbackendFactory, &factory);
2007-11-13 18:02:18 -08:00
}
ALCboolean ALCossBackendFactory_init(ALCossBackendFactory* UNUSED(self))
2007-11-13 18:02:18 -08:00
{
ConfigValueStr(nullptr, "oss", "device", &DefaultPlayback);
ConfigValueStr(nullptr, "oss", "capture", &DefaultCapture);
2012-02-29 21:45:44 -08:00
return ALC_TRUE;
2007-11-13 18:02:18 -08:00
}
2009-08-26 23:45:00 -07:00
void ALCossBackendFactory_deinit(ALCossBackendFactory* UNUSED(self))
{
PlaybackDevices.clear();
CaptureDevices.clear();
}
ALCboolean ALCossBackendFactory_querySupport(ALCossBackendFactory* UNUSED(self), ALCbackend_Type type)
2009-08-26 23:45:00 -07:00
{
if(type == ALCbackend_Playback || type == ALCbackend_Capture)
return ALC_TRUE;
return ALC_FALSE;
2009-08-26 23:45:00 -07:00
}
void ALCossBackendFactory_probe(ALCossBackendFactory* UNUSED(self), enum DevProbe type, al_string *outnames)
{
auto add_device = [outnames](const DevMap &entry) -> void
{
#ifdef HAVE_STAT
struct stat buf;
if(stat(entry.device_name.c_str(), &buf) == 0)
#endif
{
const char *name{entry.name.c_str()};
alstr_append_range(outnames, name, name+entry.name.length()+1);
}
};
2011-06-14 04:02:58 -07:00
switch(type)
{
2011-06-14 04:02:58 -07:00
case ALL_DEVICE_PROBE:
PlaybackDevices.clear();
ALCossListPopulate(&PlaybackDevices, DSP_CAP_OUTPUT);
std::for_each(PlaybackDevices.cbegin(), PlaybackDevices.cend(), add_device);
2016-12-01 18:26:18 -08:00
break;
2011-06-14 04:02:58 -07:00
case CAPTURE_DEVICE_PROBE:
CaptureDevices.clear();
ALCossListPopulate(&CaptureDevices, DSP_CAP_INPUT);
std::for_each(CaptureDevices.cbegin(), CaptureDevices.cend(), add_device);
break;
}
}
ALCbackend* ALCossBackendFactory_createBackend(ALCossBackendFactory* UNUSED(self), ALCdevice *device, ALCbackend_Type type)
{
if(type == ALCbackend_Playback)
{
ALCplaybackOSS *backend;
NEW_OBJ(backend, ALCplaybackOSS)(device);
2018-11-12 23:49:11 -08:00
if(!backend) return nullptr;
return STATIC_CAST(ALCbackend, backend);
}
if(type == ALCbackend_Capture)
{
ALCcaptureOSS *backend;
NEW_OBJ(backend, ALCcaptureOSS)(device);
2018-11-12 23:49:11 -08:00
if(!backend) return nullptr;
return STATIC_CAST(ALCbackend, backend);
}
2018-11-12 23:49:11 -08:00
return nullptr;
}