mirror of
https://github.com/cookiengineer/audacity
synced 2025-06-17 00:20:06 +02:00
Also, fix a broken(?) include in px_linux_alsa.c. I don't know if it was actually broken, but it seems like that was wrong.
775 lines
18 KiB
C
775 lines
18 KiB
C
/*
|
|
* PortMixer
|
|
* Linux ALSA 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 <alsa/asoundlib.h>
|
|
|
|
#include "portaudio.h"
|
|
#include "pa_linux_alsa.h"
|
|
|
|
#include "portmixer.h"
|
|
#include "px_mixer.h"
|
|
|
|
#if !defined(FALSE)
|
|
#define FALSE 0
|
|
#endif
|
|
|
|
#if !defined(TRUE)
|
|
#define TRUE 1
|
|
#endif
|
|
|
|
typedef struct PxSelem
|
|
{
|
|
snd_mixer_elem_t *elem;
|
|
snd_mixer_elem_t *vol;
|
|
unsigned int index;
|
|
unsigned int item;
|
|
char *name;
|
|
} PxSelem;
|
|
|
|
typedef struct PxDev
|
|
{
|
|
snd_mixer_t *handle;
|
|
int card;
|
|
int playback;
|
|
int source;
|
|
|
|
int numselems;
|
|
PxSelem *selems;
|
|
} PxDev;
|
|
|
|
typedef struct PxInfo
|
|
{
|
|
int numMixers;
|
|
char *mixers;
|
|
|
|
PxDev playback;
|
|
PxDev capture;
|
|
} PxInfo;
|
|
|
|
static int open_mixer(PxDev *dev, int card, int playback)
|
|
{
|
|
do {
|
|
snd_mixer_selem_id_t *sid;
|
|
snd_mixer_elem_t *elem;
|
|
char name[256];
|
|
int err;
|
|
int i;
|
|
|
|
snd_mixer_selem_id_alloca(&sid);
|
|
|
|
sprintf(name, "hw:%d", card);
|
|
|
|
dev->card = card;
|
|
dev->handle = NULL;
|
|
dev->playback = playback;
|
|
dev->source = -1;
|
|
|
|
err = snd_mixer_open(&dev->handle, 0);
|
|
if (err < 0) {
|
|
break;
|
|
}
|
|
|
|
err = snd_mixer_attach(dev->handle, name);
|
|
if (err < 0) {
|
|
break;
|
|
}
|
|
|
|
err = snd_mixer_selem_register(dev->handle, NULL, NULL);
|
|
if (err < 0) {
|
|
break;
|
|
}
|
|
|
|
err = snd_mixer_load(dev->handle);
|
|
if (err < 0) {
|
|
break;
|
|
}
|
|
|
|
for (elem = snd_mixer_first_elem(dev->handle);
|
|
elem != NULL;
|
|
elem = snd_mixer_elem_next(elem))
|
|
{
|
|
if (playback) {
|
|
if (snd_mixer_selem_has_common_volume(elem) ||
|
|
snd_mixer_selem_has_playback_volume(elem)) {
|
|
dev->numselems++;
|
|
}
|
|
}
|
|
else {
|
|
if (snd_mixer_selem_get_capture_group(elem) >= 0) {
|
|
dev->numselems++;
|
|
}
|
|
else if (snd_mixer_selem_is_enum_capture(elem)) {
|
|
int retval = snd_mixer_selem_get_enum_items(elem);
|
|
|
|
if (retval > 0)
|
|
dev->numselems += retval;
|
|
}
|
|
}
|
|
}
|
|
|
|
dev->selems = calloc(dev->numselems, sizeof(PxSelem));
|
|
if (dev->selems == NULL) {
|
|
break;
|
|
}
|
|
|
|
|
|
i = 0;
|
|
for (elem = snd_mixer_first_elem(dev->handle);
|
|
elem != NULL;
|
|
elem = snd_mixer_elem_next(elem))
|
|
{
|
|
snd_mixer_elem_t *vol;
|
|
int ndx;
|
|
|
|
if (playback) {
|
|
if (snd_mixer_selem_has_common_volume(elem) ||
|
|
snd_mixer_selem_has_playback_volume(elem)) {
|
|
snprintf(name,
|
|
sizeof(name),
|
|
"%s:%d",
|
|
snd_mixer_selem_get_name(elem),
|
|
snd_mixer_selem_get_index(elem));
|
|
|
|
dev->selems[i].elem = elem;
|
|
dev->selems[i].index = snd_mixer_selem_get_index(elem);
|
|
dev->selems[i].name = strdup(name);
|
|
if (!dev->selems[i].name) {
|
|
break;
|
|
}
|
|
i++;
|
|
}
|
|
continue;
|
|
}
|
|
|
|
snd_mixer_selem_id_set_name(sid, "Capture");
|
|
snd_mixer_selem_id_set_index(sid, snd_mixer_selem_get_index(elem));
|
|
vol = snd_mixer_find_selem(dev->handle, sid);
|
|
|
|
if (snd_mixer_selem_get_capture_group(elem) >= 0) {
|
|
snprintf(name,
|
|
sizeof(name),
|
|
"%s:%d",
|
|
snd_mixer_selem_get_name(elem),
|
|
snd_mixer_selem_get_index(elem));
|
|
|
|
dev->selems[i].vol = vol;
|
|
dev->selems[i].elem = elem;
|
|
dev->selems[i].index = snd_mixer_selem_get_index(elem);
|
|
dev->selems[i].name = strdup(name);
|
|
if (!dev->selems[i].name) {
|
|
break;
|
|
}
|
|
i++;
|
|
}
|
|
else if (snd_mixer_selem_is_enum_capture(elem)) {
|
|
int j;
|
|
int cnt = snd_mixer_selem_get_enum_items(elem);
|
|
|
|
if (cnt < 0)
|
|
continue;
|
|
|
|
for (j = 0; j < cnt; j++) {
|
|
char iname[256];
|
|
snd_mixer_selem_get_enum_item_name(elem, (unsigned int) j, sizeof(iname), iname);
|
|
snprintf(name,
|
|
sizeof(name),
|
|
"%s:%d",
|
|
iname,
|
|
snd_mixer_selem_get_index(elem));
|
|
|
|
dev->selems[i].vol = vol;
|
|
dev->selems[i].item = j;
|
|
dev->selems[i].elem = elem;
|
|
dev->selems[i].index = snd_mixer_selem_get_index(elem);
|
|
dev->selems[i].name = strdup(name);
|
|
if (!dev->selems[i].name) {
|
|
break;
|
|
}
|
|
i++;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (i != dev->numselems) {
|
|
break;
|
|
}
|
|
|
|
if (playback) {
|
|
return TRUE;
|
|
}
|
|
|
|
for (i = 0; i < dev->numselems; i++) {
|
|
elem = dev->selems[i].elem;
|
|
if (snd_mixer_selem_get_capture_group(elem) >= 0) {
|
|
int sw = 0;
|
|
if (snd_mixer_selem_get_capture_switch(elem, SND_MIXER_SCHN_FRONT_LEFT, &sw) < 0) {
|
|
continue;
|
|
}
|
|
if (!sw) {
|
|
continue;
|
|
}
|
|
dev->source = i;
|
|
break;
|
|
}
|
|
else if (snd_mixer_selem_is_enum_capture(elem)) {
|
|
unsigned int src;
|
|
if (snd_mixer_selem_get_enum_item(elem, SND_MIXER_SCHN_FRONT_LEFT, &src) < 0) {
|
|
continue;
|
|
}
|
|
if (src == dev->selems[i].item) {
|
|
dev->source = i;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (dev->source == -1) {
|
|
dev->source = 0;
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
} while (FALSE);
|
|
|
|
if (dev->selems) {
|
|
int i;
|
|
for (i = 0; i < dev->numselems; i++) {
|
|
if (dev->selems[i].name) {
|
|
free(dev->selems[i].name);
|
|
}
|
|
}
|
|
free(dev->selems);
|
|
dev->selems = NULL;
|
|
}
|
|
|
|
if (dev->handle) {
|
|
snd_mixer_close(dev->handle);
|
|
dev->handle = NULL;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
int OpenMixer_Linux_ALSA(px_mixer *Px, int index)
|
|
{
|
|
PxInfo *info;
|
|
int card;
|
|
|
|
if (!initialize(Px)) {
|
|
return FALSE;
|
|
}
|
|
|
|
info = (PxInfo *) Px->info;
|
|
|
|
if (PaAlsa_GetStreamInputCard(Px->pa_stream, &card) == paNoError) {
|
|
if (!open_mixer(&info->capture, card, FALSE)) {
|
|
return cleanup(Px);
|
|
}
|
|
}
|
|
|
|
if (PaAlsa_GetStreamOutputCard(Px->pa_stream, &card) == paNoError) {
|
|
if (!open_mixer(&info->playback, card, TRUE)) {
|
|
return cleanup(Px);
|
|
}
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
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;
|
|
int i;
|
|
|
|
if (info->capture.selems) {
|
|
for (i = 0; i < info->capture.numselems; i++) {
|
|
if (info->capture.selems[i].name) {
|
|
free(info->capture.selems[i].name);
|
|
}
|
|
}
|
|
free(info->capture.selems);
|
|
}
|
|
|
|
if (info->capture.handle) {
|
|
snd_mixer_close(info->capture.handle);
|
|
}
|
|
|
|
if (info->playback.selems) {
|
|
for (i = 0; i < info->playback.numselems; i++) {
|
|
if (info->playback.selems[i].name) {
|
|
free(info->playback.selems[i].name);
|
|
}
|
|
}
|
|
free(info->playback.selems);
|
|
}
|
|
|
|
if (info->playback.handle) {
|
|
snd_mixer_close(info->playback.handle);
|
|
}
|
|
|
|
if (info) {
|
|
free(info);
|
|
Px->info = NULL;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static int generic_lookup(PxDev *dev, const char *generic)
|
|
{
|
|
snd_mixer_selem_id_t *id;
|
|
snd_mixer_elem_t *elem;
|
|
int i;
|
|
|
|
if (dev == NULL) {
|
|
return -1;
|
|
}
|
|
|
|
for (i = 0; i < dev->numselems; i++) {
|
|
if (strncmp(dev->selems[i].name, generic, strlen(generic)) == 0) {
|
|
return i;
|
|
}
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
static PxVolume get_volume_indexed(PxDev *dev, int i)
|
|
{
|
|
snd_mixer_elem_t *elem;
|
|
long vol, min, max;
|
|
|
|
if (!dev->handle) {
|
|
return 0.0;
|
|
}
|
|
|
|
if (i < 0 || i > dev->numselems) {
|
|
return 0.0;
|
|
}
|
|
|
|
elem = dev->selems[i].elem;
|
|
if (dev->playback) {
|
|
snd_mixer_selem_get_playback_volume_range(elem, &min, &max);
|
|
if (snd_mixer_selem_has_playback_channel(elem, SND_MIXER_SCHN_FRONT_LEFT)) {
|
|
snd_mixer_selem_get_playback_volume(elem, SND_MIXER_SCHN_FRONT_LEFT, &vol);
|
|
return (PxVolume) vol / (max - min);
|
|
}
|
|
}
|
|
else {
|
|
snd_mixer_selem_get_capture_volume_range(elem, &min, &max);
|
|
if (snd_mixer_selem_has_capture_channel(elem, SND_MIXER_SCHN_FRONT_LEFT)) {
|
|
snd_mixer_selem_get_capture_volume(elem, SND_MIXER_SCHN_FRONT_LEFT, &vol);
|
|
return (PxVolume) vol / (max - min);
|
|
}
|
|
}
|
|
|
|
return 0.0;
|
|
}
|
|
|
|
static PxVolume get_volume(PxDev *dev, const char *name)
|
|
{
|
|
int i;
|
|
|
|
if (!dev->handle) {
|
|
return 0.0;
|
|
}
|
|
|
|
for (i = 0; i < dev->numselems; i++) {
|
|
if (strcmp(dev->selems[i].name, name) == 0) {
|
|
return get_volume_indexed(dev, i);
|
|
}
|
|
}
|
|
|
|
return 0.0;
|
|
}
|
|
|
|
static void set_volume_indexed(PxDev *dev, int i, PxVolume volume)
|
|
{
|
|
snd_mixer_elem_t *elem;
|
|
long vol, min, max;
|
|
int j;
|
|
|
|
if (!dev->handle) {
|
|
return;
|
|
}
|
|
|
|
if (i < 0 || i > dev->numselems) {
|
|
return;
|
|
}
|
|
|
|
elem = dev->selems[i].elem;
|
|
if (dev->playback) {
|
|
snd_mixer_selem_get_playback_volume_range(elem, &min, &max);
|
|
for (j = 0; j < SND_MIXER_SCHN_LAST; j++) {
|
|
if (snd_mixer_selem_has_playback_channel(elem, j)) {
|
|
vol = (long) (volume * (max - min) + 0.5);
|
|
snd_mixer_selem_set_playback_volume(elem, j, vol);
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
snd_mixer_selem_get_capture_volume_range(elem, &min, &max);
|
|
for (j = 0; j < SND_MIXER_SCHN_LAST; j++) {
|
|
if (snd_mixer_selem_has_capture_channel(elem, j)) {
|
|
vol = (long) (volume * (max - min) + 0.5);
|
|
snd_mixer_selem_set_capture_volume(elem, j, vol);
|
|
}
|
|
}
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
static void set_volume(PxDev *dev, const char *name, PxVolume volume)
|
|
{
|
|
int i;
|
|
|
|
if (!dev->handle) {
|
|
return;
|
|
}
|
|
|
|
for (i = 0; i < dev->numselems; i++) {
|
|
if (strcmp(dev->selems[i].name, name) == 0) {
|
|
set_volume_indexed(dev, i, volume);
|
|
break;
|
|
}
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
static void close_mixer(px_mixer *Px)
|
|
{
|
|
cleanup(Px);
|
|
|
|
return;
|
|
}
|
|
|
|
static int get_num_mixers(px_mixer *Px)
|
|
{
|
|
PxInfo *info = (PxInfo *)Px->info;
|
|
|
|
return 1;
|
|
}
|
|
|
|
static const char *get_mixer_name(px_mixer *Px, int i)
|
|
{
|
|
PxInfo *info = (PxInfo *)Px->info;
|
|
|
|
return "ALSA";
|
|
}
|
|
|
|
/*
|
|
|| Master volume
|
|
*/
|
|
|
|
static PxVolume get_master_volume(px_mixer *Px)
|
|
{
|
|
PxInfo *info = (PxInfo *)Px->info;
|
|
PxDev *dev = &info->playback;
|
|
|
|
return get_volume_indexed(dev, generic_lookup(dev, "Master"));
|
|
}
|
|
|
|
static void set_master_volume(px_mixer *Px, PxVolume volume)
|
|
{
|
|
PxInfo *info = (PxInfo *)Px->info;
|
|
PxDev *dev = &info->playback;
|
|
|
|
set_volume_indexed(dev, generic_lookup(dev, "Master"), volume);
|
|
|
|
/* Ensure pending events are handled...otherwise, they build up */
|
|
if (dev->handle)
|
|
snd_mixer_handle_events(dev->handle);
|
|
|
|
return;
|
|
}
|
|
|
|
/*
|
|
|| Main output volume
|
|
*/
|
|
|
|
static int supports_pcm_output_volume(px_mixer *Px)
|
|
{
|
|
PxInfo *info = (PxInfo *)Px->info;
|
|
PxDev *dev = &info->playback;
|
|
|
|
return generic_lookup(dev, "PCM") != -1;
|
|
}
|
|
|
|
static PxVolume get_pcm_output_volume(px_mixer *Px)
|
|
{
|
|
PxInfo *info = (PxInfo *)Px->info;
|
|
PxDev *dev = &info->playback;
|
|
|
|
return get_volume_indexed(dev, generic_lookup(dev, "PCM"));
|
|
}
|
|
|
|
static void set_pcm_output_volume(px_mixer *Px, PxVolume volume)
|
|
{
|
|
PxInfo *info = (PxInfo *)Px->info;
|
|
PxDev *dev = &info->playback;
|
|
|
|
set_volume_indexed(dev, generic_lookup(dev, "PCM"), volume);
|
|
|
|
/* Ensure pending events are handled...otherwise, they build up */
|
|
if (dev->handle)
|
|
snd_mixer_handle_events(dev->handle);
|
|
|
|
return;
|
|
}
|
|
|
|
/*
|
|
|| All output volumes
|
|
*/
|
|
|
|
static int get_num_output_volumes(px_mixer *Px)
|
|
{
|
|
PxInfo *info = (PxInfo *)Px->info;
|
|
|
|
if (info->playback.handle) {
|
|
return info->playback.numselems;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const char *get_output_volume_name(px_mixer *Px, int i)
|
|
{
|
|
PxInfo *info = (PxInfo *)Px->info;
|
|
|
|
if (info->playback.handle) {
|
|
if (i >= 0 && i < info->playback.numselems) {
|
|
return info->playback.selems[i].name;
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static PxVolume get_output_volume(px_mixer *Px, int i)
|
|
{
|
|
PxInfo *info = (PxInfo *)Px->info;
|
|
|
|
return get_volume_indexed(&info->playback, i);
|
|
}
|
|
|
|
static void set_output_volume(px_mixer *Px, int i, PxVolume volume)
|
|
{
|
|
PxInfo *info = (PxInfo *)Px->info;
|
|
|
|
set_volume_indexed(&info->playback, i, volume);
|
|
|
|
return;
|
|
}
|
|
|
|
/*
|
|
|| Input source
|
|
*/
|
|
|
|
static int get_num_input_sources(px_mixer *Px)
|
|
{
|
|
PxInfo *info = (PxInfo *)Px->info;
|
|
|
|
if (info->capture.handle) {
|
|
return info->capture.numselems;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const char *get_input_source_name(px_mixer *Px, int i)
|
|
{
|
|
PxInfo *info = (PxInfo *)Px->info;
|
|
|
|
if (info->capture.handle) {
|
|
if (i >= 0 && i < info->capture.numselems) {
|
|
return info->capture.selems[i].name;
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static int get_current_input_source(px_mixer *Px)
|
|
{
|
|
PxInfo *info = (PxInfo *)Px->info;
|
|
snd_mixer_elem_t *elem;
|
|
int i;
|
|
|
|
if (!info->capture.handle) {
|
|
return -1;
|
|
}
|
|
|
|
return info->capture.source;
|
|
}
|
|
|
|
static void set_current_input_source(px_mixer *Px, int i)
|
|
{
|
|
PxInfo *info = (PxInfo *)Px->info;
|
|
snd_mixer_elem_t *elem;
|
|
|
|
if (!info->capture.handle) {
|
|
return;
|
|
}
|
|
|
|
if (i < 0 || i >= info->capture.numselems) {
|
|
return;
|
|
}
|
|
|
|
elem = info->capture.selems[i].elem;
|
|
if (snd_mixer_selem_get_capture_group(elem) >= 0) {
|
|
snd_mixer_selem_set_capture_switch_all(elem, TRUE);
|
|
}
|
|
else if (snd_mixer_selem_is_enum_capture(elem)) {
|
|
int j;
|
|
for (j = 0; j < SND_MIXER_SCHN_LAST; j++) {
|
|
snd_mixer_selem_set_enum_item(elem, j, info->capture.selems[i].item);
|
|
}
|
|
}
|
|
|
|
info->capture.source = i;
|
|
|
|
/* Ensure pending events are handled...otherwise, they build up */
|
|
snd_mixer_handle_events(info->capture.handle);
|
|
|
|
set_input_volume(Px, get_input_volume(Px));
|
|
|
|
return;
|
|
}
|
|
|
|
/*
|
|
|| Input volume
|
|
*/
|
|
|
|
static PxVolume get_input_volume(px_mixer *Px)
|
|
{
|
|
PxInfo *info = (PxInfo *)Px->info;
|
|
snd_mixer_elem_t *elem;
|
|
long min;
|
|
long max;
|
|
long vol;
|
|
PxVolume volume = 0.0;
|
|
|
|
if (info->capture.source < 0 || info->capture.numselems < 1) {
|
|
return volume;
|
|
}
|
|
|
|
elem = info->capture.selems[info->capture.source].vol;
|
|
if (!elem) {
|
|
return volume;
|
|
}
|
|
|
|
snd_mixer_selem_get_capture_volume_range(elem, &min, &max);
|
|
if (snd_mixer_selem_has_capture_channel(elem, SND_MIXER_SCHN_FRONT_LEFT)) {
|
|
snd_mixer_selem_get_capture_volume(elem, SND_MIXER_SCHN_FRONT_LEFT, &vol);
|
|
volume = ((PxVolume) vol) / (max - min);
|
|
}
|
|
|
|
return volume;
|
|
}
|
|
|
|
static void set_input_volume(px_mixer *Px, PxVolume volume)
|
|
{
|
|
PxInfo *info = (PxInfo *)Px->info;
|
|
snd_mixer_elem_t *elem;
|
|
long min;
|
|
long max;
|
|
long vol;
|
|
|
|
if (info->capture.source < 0 || info->capture.numselems < 1) {
|
|
return;
|
|
}
|
|
|
|
elem = info->capture.selems[info->capture.source].vol;
|
|
if (!elem) {
|
|
return;
|
|
}
|
|
|
|
snd_mixer_selem_get_capture_volume_range(elem, &min, &max);
|
|
vol = (long) (volume * (max - min) + 0.5);
|
|
snd_mixer_selem_set_capture_volume_all(elem, vol);
|
|
|
|
if (snd_mixer_selem_has_capture_switch(elem)) {
|
|
snd_mixer_selem_set_capture_switch_all(elem, vol > 0 ? TRUE : FALSE);
|
|
}
|
|
|
|
/* Ensure pending events are handled...otherwise, they build up */
|
|
snd_mixer_handle_events(info->capture.handle);
|
|
|
|
return;
|
|
}
|