|
Packit Service |
db8eaa |
/**
|
|
Packit Service |
db8eaa |
* \file control/cards.c
|
|
Packit Service |
db8eaa |
* \brief Basic Soundcard Operations
|
|
Packit Service |
db8eaa |
* \author Jaroslav Kysela <perex@perex.cz>
|
|
Packit Service |
db8eaa |
* \date 1998-2001
|
|
Packit Service |
db8eaa |
*/
|
|
Packit Service |
db8eaa |
/*
|
|
Packit Service |
db8eaa |
* Soundcard Operations - main file
|
|
Packit Service |
db8eaa |
* Copyright (c) 1998 by Jaroslav Kysela <perex@perex.cz>
|
|
Packit Service |
db8eaa |
*
|
|
Packit Service |
db8eaa |
*
|
|
Packit Service |
db8eaa |
* This library is free software; you can redistribute it and/or modify
|
|
Packit Service |
db8eaa |
* it under the terms of the GNU Lesser General Public License as
|
|
Packit Service |
db8eaa |
* published by the Free Software Foundation; either version 2.1 of
|
|
Packit Service |
db8eaa |
* the License, or (at your option) any later version.
|
|
Packit Service |
db8eaa |
*
|
|
Packit Service |
db8eaa |
* This program is distributed in the hope that it will be useful,
|
|
Packit Service |
db8eaa |
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
Packit Service |
db8eaa |
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
Packit Service |
db8eaa |
* GNU Lesser General Public License for more details.
|
|
Packit Service |
db8eaa |
*
|
|
Packit Service |
db8eaa |
* You should have received a copy of the GNU Lesser General Public
|
|
Packit Service |
db8eaa |
* License along with this library; if not, write to the Free Software
|
|
Packit Service |
db8eaa |
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
Packit Service |
db8eaa |
*
|
|
Packit Service |
db8eaa |
*/
|
|
Packit Service |
db8eaa |
|
|
Packit Service |
db8eaa |
#include <stdio.h>
|
|
Packit Service |
db8eaa |
#include <stdlib.h>
|
|
Packit Service |
db8eaa |
#include <unistd.h>
|
|
Packit Service |
db8eaa |
#include <string.h>
|
|
Packit Service |
db8eaa |
#include <ctype.h>
|
|
Packit Service |
db8eaa |
#include <fcntl.h>
|
|
Packit Service |
db8eaa |
#include <sys/ioctl.h>
|
|
Packit Service |
db8eaa |
#include "control_local.h"
|
|
Packit Service |
db8eaa |
|
|
Packit Service |
db8eaa |
#ifndef DOC_HIDDEN
|
|
Packit Service |
db8eaa |
#define SND_FILE_CONTROL ALSA_DEVICE_DIRECTORY "controlC%i"
|
|
Packit Service |
db8eaa |
#define SND_FILE_LOAD ALOAD_DEVICE_DIRECTORY "aloadC%i"
|
|
Packit Service |
db8eaa |
#endif
|
|
Packit Service |
db8eaa |
|
|
Packit Service |
db8eaa |
static int snd_card_load2(const char *control)
|
|
Packit Service |
db8eaa |
{
|
|
Packit Service |
db8eaa |
int open_dev;
|
|
Packit Service |
db8eaa |
snd_ctl_card_info_t info;
|
|
Packit Service |
db8eaa |
|
|
Packit Service |
db8eaa |
open_dev = snd_open_device(control, O_RDONLY);
|
|
Packit Service |
db8eaa |
if (open_dev >= 0) {
|
|
Packit Service |
db8eaa |
if (ioctl(open_dev, SNDRV_CTL_IOCTL_CARD_INFO, &info) < 0) {
|
|
Packit Service |
db8eaa |
int err = -errno;
|
|
Packit Service |
db8eaa |
close(open_dev);
|
|
Packit Service |
db8eaa |
return err;
|
|
Packit Service |
db8eaa |
}
|
|
Packit Service |
db8eaa |
close(open_dev);
|
|
Packit Service |
db8eaa |
return info.card;
|
|
Packit Service |
db8eaa |
} else {
|
|
Packit Service |
db8eaa |
return -errno;
|
|
Packit Service |
db8eaa |
}
|
|
Packit Service |
db8eaa |
}
|
|
Packit Service |
db8eaa |
|
|
Packit Service |
db8eaa |
static int snd_card_load1(int card)
|
|
Packit Service |
db8eaa |
{
|
|
Packit Service |
db8eaa |
int res;
|
|
Packit Service |
db8eaa |
char control[sizeof(SND_FILE_CONTROL) + 10];
|
|
Packit Service |
db8eaa |
|
|
Packit Service |
db8eaa |
sprintf(control, SND_FILE_CONTROL, card);
|
|
Packit Service |
db8eaa |
res = snd_card_load2(control);
|
|
Packit Service |
db8eaa |
#ifdef SUPPORT_ALOAD
|
|
Packit Service |
db8eaa |
if (res < 0) {
|
|
Packit Service |
db8eaa |
char aload[sizeof(SND_FILE_LOAD) + 10];
|
|
Packit Service |
db8eaa |
sprintf(aload, SND_FILE_LOAD, card);
|
|
Packit Service |
db8eaa |
res = snd_card_load2(aload);
|
|
Packit Service |
db8eaa |
}
|
|
Packit Service |
db8eaa |
#endif
|
|
Packit Service |
db8eaa |
return res;
|
|
Packit Service |
db8eaa |
}
|
|
Packit Service |
db8eaa |
|
|
Packit Service |
db8eaa |
/**
|
|
Packit Service |
db8eaa |
* \brief Try to load the driver for a card.
|
|
Packit Service |
db8eaa |
* \param card Card number.
|
|
Packit Service |
db8eaa |
* \return 1 if driver is present, zero if driver is not present
|
|
Packit Service |
db8eaa |
*/
|
|
Packit Service |
db8eaa |
int snd_card_load(int card)
|
|
Packit Service |
db8eaa |
{
|
|
Packit Service |
db8eaa |
return !!(snd_card_load1(card) >= 0);
|
|
Packit Service |
db8eaa |
}
|
|
Packit Service |
db8eaa |
|
|
Packit Service |
db8eaa |
/**
|
|
Packit Service |
db8eaa |
* \brief Try to determine the next card.
|
|
Packit Service |
db8eaa |
* \param rcard pointer to card number
|
|
Packit Service |
db8eaa |
* \result zero if success, otherwise a negative error code
|
|
Packit Service |
db8eaa |
*
|
|
Packit Service |
db8eaa |
* Tries to determine the next card from given card number.
|
|
Packit Service |
db8eaa |
* If card number is -1, then the first available card is
|
|
Packit Service |
db8eaa |
* returned. If the result card number is -1, no more cards
|
|
Packit Service |
db8eaa |
* are available.
|
|
Packit Service |
db8eaa |
*/
|
|
Packit Service |
db8eaa |
int snd_card_next(int *rcard)
|
|
Packit Service |
db8eaa |
{
|
|
Packit Service |
db8eaa |
int card;
|
|
Packit Service |
db8eaa |
|
|
Packit Service |
db8eaa |
if (rcard == NULL)
|
|
Packit Service |
db8eaa |
return -EINVAL;
|
|
Packit Service |
db8eaa |
card = *rcard;
|
|
Packit Service |
db8eaa |
card = card < 0 ? 0 : card + 1;
|
|
Packit Service |
db8eaa |
for (; card < SND_MAX_CARDS; card++) {
|
|
Packit Service |
db8eaa |
if (snd_card_load(card)) {
|
|
Packit Service |
db8eaa |
*rcard = card;
|
|
Packit Service |
db8eaa |
return 0;
|
|
Packit Service |
db8eaa |
}
|
|
Packit Service |
db8eaa |
}
|
|
Packit Service |
db8eaa |
*rcard = -1;
|
|
Packit Service |
db8eaa |
return 0;
|
|
Packit Service |
db8eaa |
}
|
|
Packit Service |
db8eaa |
|
|
Packit Service |
db8eaa |
/**
|
|
Packit Service |
db8eaa |
* \brief Convert card string to an integer value.
|
|
Packit Service |
db8eaa |
* \param string String containing card identifier
|
|
Packit Service |
db8eaa |
* \return zero if success, otherwise a negative error code
|
|
Packit Service |
db8eaa |
*
|
|
Packit Service |
db8eaa |
* The accepted format is an integer value in ASCII representation
|
|
Packit Service |
db8eaa |
* or the card identifier (the id parameter for sound-card drivers).
|
|
Packit Service |
db8eaa |
* The control device name like /dev/snd/controlC0 is accepted, too.
|
|
Packit Service |
db8eaa |
*/
|
|
Packit Service |
db8eaa |
int snd_card_get_index(const char *string)
|
|
Packit Service |
db8eaa |
{
|
|
Packit Service |
db8eaa |
int card, err;
|
|
Packit Service |
db8eaa |
snd_ctl_t *handle;
|
|
Packit Service |
db8eaa |
snd_ctl_card_info_t info;
|
|
Packit Service |
db8eaa |
|
|
Packit Service |
db8eaa |
if (!string || *string == '\0')
|
|
Packit Service |
db8eaa |
return -EINVAL;
|
|
Packit Service |
db8eaa |
if ((isdigit(*string) && *(string + 1) == 0) ||
|
|
Packit Service |
db8eaa |
(isdigit(*string) && isdigit(*(string + 1)) && *(string + 2) == 0)) {
|
|
Packit Service |
db8eaa |
if (sscanf(string, "%i", &card) != 1)
|
|
Packit Service |
db8eaa |
return -EINVAL;
|
|
Packit Service |
db8eaa |
if (card < 0 || card >= SND_MAX_CARDS)
|
|
Packit Service |
db8eaa |
return -EINVAL;
|
|
Packit Service |
db8eaa |
err = snd_card_load1(card);
|
|
Packit Service |
db8eaa |
if (err >= 0)
|
|
Packit Service |
db8eaa |
return card;
|
|
Packit Service |
db8eaa |
return err;
|
|
Packit Service |
db8eaa |
}
|
|
Packit Service |
db8eaa |
if (string[0] == '/') /* device name */
|
|
Packit Service |
db8eaa |
return snd_card_load2(string);
|
|
Packit Service |
db8eaa |
for (card = 0; card < SND_MAX_CARDS; card++) {
|
|
Packit Service |
db8eaa |
#ifdef SUPPORT_ALOAD
|
|
Packit Service |
db8eaa |
if (! snd_card_load(card))
|
|
Packit Service |
db8eaa |
continue;
|
|
Packit Service |
db8eaa |
#endif
|
|
Packit Service |
db8eaa |
if (snd_ctl_hw_open(&handle, NULL, card, 0) < 0)
|
|
Packit Service |
db8eaa |
continue;
|
|
Packit Service |
db8eaa |
if (snd_ctl_card_info(handle, &info) < 0) {
|
|
Packit Service |
db8eaa |
snd_ctl_close(handle);
|
|
Packit Service |
db8eaa |
continue;
|
|
Packit Service |
db8eaa |
}
|
|
Packit Service |
db8eaa |
snd_ctl_close(handle);
|
|
Packit Service |
db8eaa |
if (!strcmp((const char *)info.id, string))
|
|
Packit Service |
db8eaa |
return card;
|
|
Packit Service |
db8eaa |
}
|
|
Packit Service |
db8eaa |
return -ENODEV;
|
|
Packit Service |
db8eaa |
}
|
|
Packit Service |
db8eaa |
|
|
Packit Service |
db8eaa |
/**
|
|
Packit Service |
db8eaa |
* \brief Obtain the card name.
|
|
Packit Service |
db8eaa |
* \param card Card number
|
|
Packit Service |
db8eaa |
* \param name Result - card name corresponding to card number
|
|
Packit Service |
db8eaa |
* \result zero if success, otherwise a negative error code
|
|
Packit Service |
db8eaa |
*
|
|
Packit Service |
db8eaa |
* The value returned in name is allocated with strdup and should be
|
|
Packit Service |
db8eaa |
* freed when no longer used.
|
|
Packit Service |
db8eaa |
*/
|
|
Packit Service |
db8eaa |
int snd_card_get_name(int card, char **name)
|
|
Packit Service |
db8eaa |
{
|
|
Packit Service |
db8eaa |
snd_ctl_t *handle;
|
|
Packit Service |
db8eaa |
snd_ctl_card_info_t info;
|
|
Packit Service |
db8eaa |
int err;
|
|
Packit Service |
db8eaa |
|
|
Packit Service |
db8eaa |
if (name == NULL)
|
|
Packit Service |
db8eaa |
return -EINVAL;
|
|
Packit Service |
db8eaa |
if ((err = snd_ctl_hw_open(&handle, NULL, card, 0)) < 0)
|
|
Packit Service |
db8eaa |
return err;
|
|
Packit Service |
db8eaa |
if ((err = snd_ctl_card_info(handle, &info)) < 0) {
|
|
Packit Service |
db8eaa |
snd_ctl_close(handle);
|
|
Packit Service |
db8eaa |
return err;
|
|
Packit Service |
db8eaa |
}
|
|
Packit Service |
db8eaa |
snd_ctl_close(handle);
|
|
Packit Service |
db8eaa |
*name = strdup((const char *)info.name);
|
|
Packit Service |
db8eaa |
if (*name == NULL)
|
|
Packit Service |
db8eaa |
return -ENOMEM;
|
|
Packit Service |
db8eaa |
return 0;
|
|
Packit Service |
db8eaa |
}
|
|
Packit Service |
db8eaa |
|
|
Packit Service |
db8eaa |
/**
|
|
Packit Service |
db8eaa |
* \brief Obtain the card long name.
|
|
Packit Service |
db8eaa |
* \param card Card number
|
|
Packit Service |
db8eaa |
* \param name Result - card long name corresponding to card number
|
|
Packit Service |
db8eaa |
* \result zero if success, otherwise a negative error code
|
|
Packit Service |
db8eaa |
*
|
|
Packit Service |
db8eaa |
* The value returned in name is allocated with strdup and should be
|
|
Packit Service |
db8eaa |
* freed when no longer used.
|
|
Packit Service |
db8eaa |
*/
|
|
Packit Service |
db8eaa |
int snd_card_get_longname(int card, char **name)
|
|
Packit Service |
db8eaa |
{
|
|
Packit Service |
db8eaa |
snd_ctl_t *handle;
|
|
Packit Service |
db8eaa |
snd_ctl_card_info_t info;
|
|
Packit Service |
db8eaa |
int err;
|
|
Packit Service |
db8eaa |
|
|
Packit Service |
db8eaa |
if (name == NULL)
|
|
Packit Service |
db8eaa |
return -EINVAL;
|
|
Packit Service |
db8eaa |
if ((err = snd_ctl_hw_open(&handle, NULL, card, 0)) < 0)
|
|
Packit Service |
db8eaa |
return err;
|
|
Packit Service |
db8eaa |
if ((err = snd_ctl_card_info(handle, &info)) < 0) {
|
|
Packit Service |
db8eaa |
snd_ctl_close(handle);
|
|
Packit Service |
db8eaa |
return err;
|
|
Packit Service |
db8eaa |
}
|
|
Packit Service |
db8eaa |
snd_ctl_close(handle);
|
|
Packit Service |
db8eaa |
*name = strdup((const char *)info.longname);
|
|
Packit Service |
db8eaa |
if (*name == NULL)
|
|
Packit Service |
db8eaa |
return -ENOMEM;
|
|
Packit Service |
db8eaa |
return 0;
|
|
Packit Service |
db8eaa |
}
|