1
0
mirror of https://github.com/cookiengineer/audacity synced 2025-07-05 06:59:07 +02:00
audacity/lib-src/portmixer/src/px_unix_oss.c

535 lines
12 KiB
C

/*
* PortMixer
* Unix OSS Implementation
*
* Copyright (c) 2002, 2006
*
* Written by Dominic Mazzoni
* and Leland Lucius
*
* PortMixer is intended to work side-by-side with PortAudio,
* the Portable Real-Time Audio Library by Ross Bencina and
* Phil Burk.
*
* Permission is hereby granted, free of charge, to any person obtaining
* a copy of this software and associated documentation files
* (the "Software"), to deal in the Software without restriction,
* including without limitation the rights to use, copy, modify, merge,
* publish, distribute, sublicense, and/or sell copies of the Software,
* and to permit persons to whom the Software is furnished to do so,
* subject to the following conditions:
*
* The above copyright notice and this permission notice shall be
* included in all copies or substantial portions of the Software.
*
* Any person wishing to distribute modifications to the Software is
* requested to send the modifications to the original developer so that
* they can be incorporated into the canonical version.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR
* ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF
* CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
* WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*
*/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <fcntl.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/ioctl.h>
#include <ctype.h>
#if defined(HAVE_SYS_SOUNDCARD_H)
# include <sys/soundcard.h>
#elif defined(HAVE_LINUX_SOUNDCARD_H)
# include <linux/soundcard.h>
#elif defined(HAVE_MACHINE_SOUNDCARD_H)
# include <machine/soundcard.h> /* JH20010905 */
#else
# error No sound card header file
#endif
#include "portaudio.h"
#include "pa_unix_oss.h"
#include "portmixer.h"
#include "px_mixer.h"
#if !defined(FALSE)
#define FALSE 0
#endif
#if !defined(TRUE)
#define TRUE 1
#endif
#define MIXER_NAME_BASE "/dev/mixer"
#define MIXER_COUNT_MAX 10
typedef struct PxDev
{
const char *name;
int fd;
int num;
int chans[SOUND_MIXER_NRDEVICES];
} PxDev;
typedef struct PxInfo
{
int numMixers;
int mixerIndexes[MIXER_COUNT_MAX];
char mixers[MIXER_COUNT_MAX][sizeof(MIXER_NAME_BASE) + 1];
PxDev capture;
PxDev playback;
} PxInfo;
static const char *labels[] = SOUND_DEVICE_LABELS;
static int open_mixer(PxDev *dev, int cmd)
{
char name[sizeof(MIXER_NAME_BASE) + 1];
int mask;
int i;
int id = 0;
for (i = strlen(dev->name) - 1; i >= 0; i++) {
if (!isdigit(dev->name[i])) {
break;
}
}
id = atoi(&dev->name[i + 1]);
if (id < 0 || id >= MIXER_COUNT_MAX) {
return -1;
}
strcpy(name, MIXER_NAME_BASE);
if (id == 0)
name[strlen(MIXER_NAME_BASE)] = 0;
else
name[strlen(MIXER_NAME_BASE)] = '0' + (id - 1);
do {
dev->fd = open(name, O_RDWR);
if (dev->fd < 0) {
break;
}
if (ioctl(dev->fd, cmd, &mask) == -1) {
break;
}
dev->num = 0;
for (i = 0; i < SOUND_MIXER_NRDEVICES; i++) {
if (mask & (1 << i)) {
dev->chans[dev->num++] = i;
}
}
return TRUE;
} while (FALSE);
if (dev->fd >= 0) {
close(dev->fd);
dev->fd = -1;
}
return FALSE;
}
int OpenMixer_Unix_OSS(px_mixer *Px, int index)
{
PxInfo *info;
if (!initialize(Px)) {
return FALSE;
}
get_num_mixers(Px);
info = (PxInfo *) Px->info;
info->capture.fd = -1;
info->capture.num = 0;
info->playback.fd = -1;
info->playback.num = 0;
do {
info->capture.name = PaOSS_GetStreamInputDevice(Px->pa_stream);
if (info->capture.name) {
if (!open_mixer(&info->capture, SOUND_MIXER_READ_RECMASK)) {
break;
}
}
info->playback.name = PaOSS_GetStreamOutputDevice(Px->pa_stream);
if (info->playback.name) {
if (!open_mixer(&info->playback, SOUND_MIXER_READ_DEVMASK)) {
break;
}
}
return TRUE;
} while (FALSE);
return cleanup(Px);
}
static PxVolume get_volume(int fd, int channel)
{
int vol;
int stereo;
if (ioctl(fd, SOUND_MIXER_READ_STEREODEVS, &stereo) == 0)
stereo = ((stereo & (1 << channel)) != 0);
else
stereo = 0;
if (ioctl(fd, MIXER_READ(channel), &vol) == -1)
return 0.0;
if (stereo)
return ((vol & 0xFF)/200.0) + (((vol>>8) & 0xFF)/200.0);
else
return (vol & 0xFF)/100.0;
}
static int initialize(px_mixer *Px)
{
Px->info = calloc(1, sizeof(PxInfo));
if (Px->info == NULL) {
return FALSE;
}
Px->CloseMixer = close_mixer;
Px->GetNumMixers = get_num_mixers;
Px->GetMixerName = get_mixer_name;
Px->GetMasterVolume = get_master_volume;
Px->SetMasterVolume = set_master_volume;
Px->SupportsPCMOutputVolume = supports_pcm_output_volume;
Px->GetPCMOutputVolume = get_pcm_output_volume;
Px->SetPCMOutputVolume = set_pcm_output_volume;
Px->GetNumOutputVolumes = get_num_output_volumes;
Px->GetOutputVolumeName = get_output_volume_name;
Px->GetOutputVolume = get_output_volume;
Px->SetOutputVolume = set_output_volume;
Px->GetNumInputSources = get_num_input_sources;
Px->GetInputSourceName = get_input_source_name;
Px->GetCurrentInputSource = get_current_input_source;
Px->SetCurrentInputSource = set_current_input_source;
Px->GetInputVolume = get_input_volume;
Px->SetInputVolume = set_input_volume;
// Px->SupportsOutputBalance = supports_output_balance;
// Px->GetOutputBalance = get_output_balance;
// Px->SetOutputBalance = set_output_balance;
// Px->SupportsPlaythrough = supports_play_through;
// Px->GetPlaythrough = get_play_through;
// Px->SetPlaythrough = set_play_through;
return TRUE;
}
static int cleanup(px_mixer *Px)
{
PxInfo *info = (PxInfo *)Px->info;
if (info->capture.fd >= 0) {
close(info->capture.fd);
}
if (info->playback.fd >= 0) {
close(info->playback.fd);
}
if (info) {
free(info);
Px->info = NULL;
}
return FALSE;
}
static void close_mixer(px_mixer *Px)
{
cleanup(Px);
return;
}
static int get_num_mixers(px_mixer *Px)
{
PxInfo *info = (PxInfo *)Px->info;
int i;
int fd;
info->numMixers = 0;
for (i = 0; i < MIXER_COUNT_MAX; i++) {
strcpy(info->mixers[i], MIXER_NAME_BASE);
if (i == 0)
info->mixers[i][strlen(MIXER_NAME_BASE)] = 0;
else
info->mixers[i][strlen(MIXER_NAME_BASE)] = '0' + (i - 1);
fd = open(info->mixers[i], O_RDWR);
if (fd >= 0) {
info->mixerIndexes[info->numMixers] = i;
info->numMixers++;
close(fd);
}
}
return info->numMixers;
}
static const char *get_mixer_name(px_mixer *Px, int i)
{
PxInfo *info = (PxInfo *)Px->info;
if (info->numMixers <= 0)
get_num_mixers(Px);
if (i >= 0 && i < info->numMixers) {
return info->mixers[info->mixerIndexes[i]];
}
return NULL;
}
/*
|| Master volume
*/
static PxVolume get_master_volume(px_mixer *Px)
{
PxInfo *info = (PxInfo *)Px->info;
if (info->playback.fd >= 0) {
return get_volume(info->playback.fd, SOUND_MIXER_READ_VOLUME);
}
return 0.0;
}
static void set_master_volume(px_mixer *Px, PxVolume volume)
{
PxInfo *info = (PxInfo *)Px->info;
int vol = (int) ((volume * 100.0) + 0.5);
if (info->playback.fd >= 0) {
vol = (vol | (vol << 8));
ioctl(info->playback.fd, SOUND_MIXER_WRITE_VOLUME, &vol);
}
return;
}
/*
|| Main output volume
*/
static int supports_pcm_output_volume(px_mixer *Px)
{
PxInfo *info = (PxInfo *)Px->info;
int i;
if (info->playback.fd >= 0) {
for (i = 0; i < info->playback.num; i++) {
if (info->playback.chans[i] == SOUND_MIXER_PCM) {
return TRUE;
}
}
}
return FALSE;
}
static PxVolume get_pcm_output_volume(px_mixer *Px)
{
PxInfo *info = (PxInfo *)Px->info;
if (supports_pcm_output_volume(Px)) {
return get_volume(info->playback.fd, SOUND_MIXER_READ_PCM);
}
return 0.0;
}
static void set_pcm_output_volume(px_mixer *Px, PxVolume volume)
{
PxInfo *info = (PxInfo *)Px->info;
int vol = (int) ((volume * 100.0) + 0.5);
if (supports_pcm_output_volume(Px)) {
vol = (vol | (vol << 8));
ioctl(info->playback.fd, SOUND_MIXER_WRITE_PCM, &vol);
}
return;
}
/*
|| All output volumes
*/
static int get_num_output_volumes(px_mixer *Px)
{
PxInfo *info = (PxInfo *)Px->info;
return info->playback.num;
}
static const char *get_output_volume_name(px_mixer *Px, int i)
{
PxInfo *info = (PxInfo *)Px->info;
if (info->playback.fd >= 0) {
if (i >= 0 && i < info->playback.num) {
return labels[info->playback.chans[i]];
}
}
return NULL;
}
static PxVolume get_output_volume(px_mixer *Px, int i)
{
PxInfo *info = (PxInfo *)Px->info;
if (info->playback.fd >= 0) {
if (i >= 0 && i < info->playback.num) {
return get_volume(info->playback.fd, info->playback.chans[i]);
}
}
return 0.0;
}
static void set_output_volume(px_mixer *Px, int i, PxVolume volume)
{
PxInfo *info = (PxInfo *)Px->info;
int vol = (int) ((volume * 100.0) + 0.5);
if (info->playback.fd >= 0) {
if (i >= 0 && i < info->playback.num) {
vol = (vol | (vol << 8));
ioctl(info->playback.fd, MIXER_WRITE(info->playback.chans[i]), &vol);
}
}
return;
}
/*
|| Input source
*/
static int get_num_input_sources(px_mixer *Px)
{
PxInfo *info = (PxInfo *)Px->info;
return info->capture.num;
}
static const char *get_input_source_name(px_mixer *Px, int i)
{
PxInfo *info = (PxInfo *)Px->info;
if (info->capture.fd >= 0) {
if (i >= 0 && i < info->capture.num) {
return labels[info->capture.chans[i]];
}
}
return NULL;
}
static int get_current_input_source(px_mixer *Px)
{
PxInfo *info = (PxInfo *)Px->info;
int mask;
int i;
/* Note that there may be more than one in OSS; we pick
the first one */
if (info->capture.fd >= 0) {
if (ioctl(info->capture.fd, SOUND_MIXER_READ_RECSRC, &mask) == -1) {
return -1; /* none / error */
}
for (i = 0; i < info->capture.num; i++) {
if (mask & (1 << info->capture.chans[i])) {
return i;
}
}
}
return -1; /* none */
}
static void set_current_input_source(px_mixer *Px, int i)
{
PxInfo *info = (PxInfo *)Px->info;
int mask;
if (info->capture.fd >= 0) {
if (i >= 0 && i < info->capture.num) {
mask = (1 << info->capture.chans[i]);
ioctl(info->capture.fd, SOUND_MIXER_WRITE_RECSRC, &mask);
}
}
return;
}
/*
|| Input volume
*/
static PxVolume get_input_volume(px_mixer *Px)
{
PxInfo *info = (PxInfo *)Px->info;
int i;
if (info->capture.fd >= 0) {
i = get_current_input_source(Px);
if (i >= 0) {
return get_volume(info->capture.fd, info->capture.chans[i]);
}
}
return 0.0;
}
static void set_input_volume(px_mixer *Px, PxVolume volume)
{
PxInfo *info = (PxInfo *)Px->info;
int vol;
int i;
if (info->capture.fd >= 0) {
i = get_current_input_source(Px);
if (i >= 0) {
vol = (int) ((volume * 100.0) + 0.5);
vol = (vol | (vol << 8));
ioctl(info->capture.fd, MIXER_WRITE(info->capture.chans[i]), &vol);
}
}
return;
}