/* Copyright (c) 2012 The Chromium OS Authors. All rights reserved.
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#ifndef _GNU_SOURCE
#define _GNU_SOURCE /* For asprintf */
#endif
#include <alsa/asoundlib.h>
#include <syslog.h>
#include "cras_alsa_card.h"
#include "cras_alsa_io.h"
#include "cras_alsa_mixer.h"
#include "cras_alsa_ucm.h"
#include "cras_device_blacklist.h"
#include "cras_card_config.h"
#include "cras_config.h"
#include "cras_iodev.h"
#include "cras_iodev_list.h"
#include "cras_system_state.h"
#include "cras_types.h"
#include "cras_util.h"
#include "utlist.h"
#define MAX_ALSA_CARDS 32 /* Alsa limit on number of cards. */
#define MAX_ALSA_PCM_NAME_LENGTH 6 /* Alsa names "hw:XX" + 1 for null. */
#define MAX_INI_NAME_LENGTH 63 /* 63 chars + 1 for null where declared. */
#define MAX_COUPLED_OUTPUT_SIZE 4
struct iodev_list_node {
struct cras_iodev *iodev;
enum CRAS_STREAM_DIRECTION direction;
struct iodev_list_node *prev, *next;
};
/* Keeps an fd that is registered with system state. A list of fds must be
* kept so that they can be removed when the card is destroyed. */
struct hctl_poll_fd {
int fd;
struct hctl_poll_fd *prev, *next;
};
/* Holds information about each sound card on the system.
* name - of the form hw:XX,YY.
* card_index - 0 based index, value of "XX" in the name.
* iodevs - Input and output devices for this card.
* mixer - Controls the mixer controls for this card.
* ucm - CRAS use case manager if available.
* hctl - ALSA high-level control interface.
* hctl_poll_fds - List of fds registered with cras_system_state.
* config - Config info for this card, can be NULL if none found.
*/
struct cras_alsa_card {
char name[MAX_ALSA_PCM_NAME_LENGTH];
size_t card_index;
struct iodev_list_node *iodevs;
struct cras_alsa_mixer *mixer;
struct cras_use_case_mgr *ucm;
snd_hctl_t *hctl;
struct hctl_poll_fd *hctl_poll_fds;
struct cras_card_config *config;
};
/* Creates an iodev for the given device.
* Args:
* alsa_card - the alsa_card the device will be added to.
* info - Information about the card type and priority.
* card_name - The name of the card.
* dev_name - The name of the device.
* dev_id - The id string of the device.
* device_index - 0 based index, value of "YY" in "hw:XX,YY".
* direction - Input or output.
* Returns:
* Pointer to the created iodev, or NULL on error.
* other negative error code otherwise.
*/
struct cras_iodev *create_iodev_for_device(
struct cras_alsa_card *alsa_card,
struct cras_alsa_card_info *info,
const char *card_name,
const char *dev_name,
const char *dev_id,
unsigned device_index,
enum CRAS_STREAM_DIRECTION direction)
{
struct iodev_list_node *new_dev;
struct iodev_list_node *node;
int first = 1;
/* Find whether this is the first device in this direction, and
* avoid duplicate device indexes. */
DL_FOREACH(alsa_card->iodevs, node) {
if (node->direction != direction)
continue;
first = 0;
if (alsa_iodev_index(node->iodev) == device_index) {
syslog(LOG_DEBUG,
"Skipping duplicate device for %s:%s:%s [%u]",
card_name, dev_name, dev_id, device_index);
return node->iodev;
}
}
new_dev = calloc(1, sizeof(*new_dev));
if (new_dev == NULL)
return NULL;
new_dev->direction = direction;
new_dev->iodev = alsa_iodev_create(info->card_index,
card_name,
device_index,
dev_name,
dev_id,
info->card_type,
first,
alsa_card->mixer,
alsa_card->config,
alsa_card->ucm,
alsa_card->hctl,
direction,
info->usb_vendor_id,
info->usb_product_id,
info->usb_serial_number);
if (new_dev->iodev == NULL) {
syslog(LOG_ERR, "Couldn't create alsa_iodev for %u:%u\n",
info->card_index, device_index);
free(new_dev);
return NULL;
}
syslog(LOG_DEBUG, "New %s device %u:%d",
direction == CRAS_STREAM_OUTPUT ? "playback" : "capture",
info->card_index,
device_index);
DL_APPEND(alsa_card->iodevs, new_dev);
return new_dev->iodev;
}
/* Returns non-zero if this card has hctl jacks.
*/
static int card_has_hctl_jack(struct cras_alsa_card *alsa_card)
{
struct iodev_list_node *node;
/* Find the first device that has an hctl jack. */
DL_FOREACH(alsa_card->iodevs, node) {
if (alsa_iodev_has_hctl_jacks(node->iodev))
return 1;
}
return 0;
}
/* Check if a device should be ignored for this card. Returns non-zero if the
* device is in the blacklist and should be ignored.
*/
static int should_ignore_dev(struct cras_alsa_card_info *info,
struct cras_device_blacklist *blacklist,
size_t device_index)
{
if (info->card_type == ALSA_CARD_TYPE_USB)
return cras_device_blacklist_check(blacklist,
info->usb_vendor_id,
info->usb_product_id,
info->usb_desc_checksum,
device_index);
return 0;
}
/* Filters an array of mixer control names. Keep a name if it is
* specified in the ucm config. */
static struct mixer_name *filter_controls(struct cras_use_case_mgr *ucm,
struct mixer_name *controls)
{
struct mixer_name *control;
DL_FOREACH(controls, control) {
char *dev = ucm_get_dev_for_mixer(ucm, control->name,
CRAS_STREAM_OUTPUT);
if (!dev)
DL_DELETE(controls, control);
else
free(dev);
}
return controls;
}
/* Handles notifications from alsa controls. Called by main thread when a poll
* fd provided by alsa signals there is an event available. */
static void alsa_control_event_pending(void *arg)
{
struct cras_alsa_card *card;
card = (struct cras_alsa_card *)arg;
if (card == NULL) {
syslog(LOG_ERR, "Invalid card from control event.");
return;
}
/* handle_events will trigger the callback registered with each control
* that has changed. */
snd_hctl_handle_events(card->hctl);
}
static int add_controls_and_iodevs_by_matching(
struct cras_alsa_card_info *info,
struct cras_device_blacklist *blacklist,
struct cras_alsa_card *alsa_card,
const char *card_name,
snd_ctl_t *handle)
{
struct mixer_name *coupled_controls = NULL;
int dev_idx;
snd_pcm_info_t *dev_info;
struct mixer_name *extra_controls = NULL;
int rc = 0;
snd_pcm_info_alloca(&dev_info);
if (alsa_card->ucm) {
char *extra_main_volume;
/* Filter the extra output mixer names */
extra_controls =
filter_controls(alsa_card->ucm,
mixer_name_add(extra_controls, "IEC958",
CRAS_STREAM_OUTPUT,
MIXER_NAME_VOLUME));
/* Get the extra main volume control. */
extra_main_volume = ucm_get_flag(alsa_card->ucm,
"ExtraMainVolume");
if (extra_main_volume) {
extra_controls =
mixer_name_add(extra_controls,
extra_main_volume,
CRAS_STREAM_OUTPUT,
MIXER_NAME_MAIN_VOLUME);
free(extra_main_volume);
}
mixer_name_dump(extra_controls, "extra controls");
/* Check if coupled controls has been specified for speaker. */
coupled_controls = ucm_get_coupled_mixer_names(
alsa_card->ucm, "Speaker");
mixer_name_dump(coupled_controls, "coupled controls");
}
/* Add controls to mixer by name matching. */
rc = cras_alsa_mixer_add_controls_by_name_matching(
alsa_card->mixer,
extra_controls,
coupled_controls);
if (rc) {
syslog(LOG_ERR, "Fail adding controls to mixer for %s.",
alsa_card->name);
goto error;
}
/* Go through every device. */
dev_idx = -1;
while (1) {
rc = snd_ctl_pcm_next_device(handle, &dev_idx);
if (rc < 0)
goto error;
if (dev_idx < 0)
break;
snd_pcm_info_set_device(dev_info, dev_idx);
snd_pcm_info_set_subdevice(dev_info, 0);
/* Check for playback devices. */
snd_pcm_info_set_stream(
dev_info, SND_PCM_STREAM_PLAYBACK);
if (snd_ctl_pcm_info(handle, dev_info) == 0 &&
!should_ignore_dev(info, blacklist, dev_idx)) {
struct cras_iodev *iodev =
create_iodev_for_device(
alsa_card,
info,
card_name,
snd_pcm_info_get_name(dev_info),
snd_pcm_info_get_id(dev_info),
dev_idx,
CRAS_STREAM_OUTPUT);
if (iodev) {
rc = alsa_iodev_legacy_complete_init(
iodev);
if (rc < 0)
goto error;
}
}
/* Check for capture devices. */
snd_pcm_info_set_stream(
dev_info, SND_PCM_STREAM_CAPTURE);
if (snd_ctl_pcm_info(handle, dev_info) == 0) {
struct cras_iodev *iodev =
create_iodev_for_device(
alsa_card,
info,
card_name,
snd_pcm_info_get_name(dev_info),
snd_pcm_info_get_id(dev_info),
dev_idx,
CRAS_STREAM_INPUT);
if (iodev) {
rc = alsa_iodev_legacy_complete_init(
iodev);
if (rc < 0)
goto error;
}
}
}
error:
mixer_name_free(coupled_controls);
mixer_name_free(extra_controls);
return rc;
}
static int add_controls_and_iodevs_with_ucm(
struct cras_alsa_card_info *info,
struct cras_alsa_card *alsa_card,
const char *card_name,
snd_ctl_t *handle)
{
snd_pcm_info_t *dev_info;
struct iodev_list_node *node;
int rc = 0;
struct ucm_section *section;
struct ucm_section *ucm_sections;
snd_pcm_info_alloca(&dev_info);
/* Get info on the devices specified in the UCM config. */
ucm_sections = ucm_get_sections(alsa_card->ucm);
if (!ucm_sections) {
syslog(LOG_ERR,
"Could not retrieve any UCM SectionDevice"
" info for '%s'.", card_name);
rc = -ENOENT;
goto error;
}
/* Create all of the controls first. */
DL_FOREACH(ucm_sections, section) {
rc = cras_alsa_mixer_add_controls_in_section(
alsa_card->mixer, section);
if (rc) {
syslog(LOG_ERR, "Failed adding controls to"
" mixer for '%s:%s'",
card_name,
section->name);
goto error;
}
}
/* Create all of the devices. */
DL_FOREACH(ucm_sections, section) {
snd_pcm_info_set_device(dev_info, section->dev_idx);
snd_pcm_info_set_subdevice(dev_info, 0);
if (section->dir == CRAS_STREAM_OUTPUT)
snd_pcm_info_set_stream(
dev_info, SND_PCM_STREAM_PLAYBACK);
else if (section->dir == CRAS_STREAM_INPUT)
snd_pcm_info_set_stream(
dev_info, SND_PCM_STREAM_CAPTURE);
else {
syslog(LOG_ERR, "Unexpected direction: %d",
section->dir);
rc = -EINVAL;
goto error;
}
if (snd_ctl_pcm_info(handle, dev_info)) {
syslog(LOG_ERR,
"Could not get info for device: %s",
section->name);
continue;
}
create_iodev_for_device(
alsa_card, info, card_name,
snd_pcm_info_get_name(dev_info),
snd_pcm_info_get_id(dev_info),
section->dev_idx, section->dir);
}
/* Setup jacks and controls for the devices. */
DL_FOREACH(ucm_sections, section) {
DL_FOREACH(alsa_card->iodevs, node) {
if (node->direction == section->dir &&
alsa_iodev_index(node->iodev) ==
section->dev_idx)
break;
}
if (node) {
rc = alsa_iodev_ucm_add_nodes_and_jacks(
node->iodev, section);
if (rc < 0)
goto error;
}
}
DL_FOREACH(alsa_card->iodevs, node) {
alsa_iodev_ucm_complete_init(node->iodev);
}
error:
ucm_section_free_list(ucm_sections);
return rc;
}
static void configure_echo_reference_dev(struct cras_alsa_card *alsa_card)
{
struct iodev_list_node *dev_node, *echo_ref_node;
const char *echo_ref_name;
if (!alsa_card->ucm)
return;
DL_FOREACH(alsa_card->iodevs, dev_node) {
if (!dev_node->iodev->nodes)
continue;
echo_ref_name = ucm_get_echo_reference_dev_name_for_dev(
alsa_card->ucm,
dev_node->iodev->nodes->name);
if (!echo_ref_name)
continue;
DL_FOREACH(alsa_card->iodevs, echo_ref_node) {
if (echo_ref_node->iodev->nodes == NULL)
continue;
if (!strcmp(echo_ref_name,
echo_ref_node->iodev->nodes->name))
break;
}
if (echo_ref_node)
dev_node->iodev->echo_reference_dev =
echo_ref_node->iodev;
else
syslog(LOG_ERR,
"Echo ref dev %s doesn't exist on card %s",
echo_ref_name, alsa_card->name);
free((void *)echo_ref_name);
}
}
/*
* Exported Interface.
*/
struct cras_alsa_card *cras_alsa_card_create(
struct cras_alsa_card_info *info,
const char *device_config_dir,
struct cras_device_blacklist *blacklist,
const char *ucm_suffix)
{
snd_ctl_t *handle = NULL;
int rc, n;
snd_ctl_card_info_t *card_info;
const char *card_name;
struct cras_alsa_card *alsa_card;
if (info->card_index >= MAX_ALSA_CARDS) {
syslog(LOG_ERR,
"Invalid alsa card index %u",
info->card_index);
return NULL;
}
snd_ctl_card_info_alloca(&card_info);
alsa_card = calloc(1, sizeof(*alsa_card));
if (alsa_card == NULL)
return NULL;
alsa_card->card_index = info->card_index;
snprintf(alsa_card->name,
MAX_ALSA_PCM_NAME_LENGTH,
"hw:%u",
info->card_index);
rc = snd_ctl_open(&handle, alsa_card->name, 0);
if (rc < 0) {
syslog(LOG_ERR, "Fail opening control %s.", alsa_card->name);
goto error_bail;
}
rc = snd_ctl_card_info(handle, card_info);
if (rc < 0) {
syslog(LOG_ERR, "Error getting card info.");
goto error_bail;
}
card_name = snd_ctl_card_info_get_name(card_info);
if (card_name == NULL) {
syslog(LOG_ERR, "Error getting card name.");
goto error_bail;
}
/* Read config file for this card if it exists. */
alsa_card->config = cras_card_config_create(device_config_dir,
card_name);
if (alsa_card->config == NULL)
syslog(LOG_DEBUG, "No config file for %s", alsa_card->name);
/* Create a use case manager if a configuration is available. */
if (ucm_suffix) {
char *ucm_name;
if (asprintf(&ucm_name, "%s.%s", card_name, ucm_suffix) == -1) {
syslog(LOG_ERR, "Error creating ucm name");
goto error_bail;
}
alsa_card->ucm = ucm_create(ucm_name);
syslog(LOG_INFO, "Card %s (%s) has UCM: %s",
alsa_card->name, ucm_name,
alsa_card->ucm ? "yes" : "no");
free(ucm_name);
} else {
alsa_card->ucm = ucm_create(card_name);
syslog(LOG_INFO, "Card %s (%s) has UCM: %s",
alsa_card->name, card_name,
alsa_card->ucm ? "yes" : "no");
}
rc = snd_hctl_open(&alsa_card->hctl,
alsa_card->name,
SND_CTL_NONBLOCK);
if (rc < 0) {
syslog(LOG_DEBUG,
"failed to get hctl for %s", alsa_card->name);
alsa_card->hctl = NULL;
} else {
rc = snd_hctl_nonblock(alsa_card->hctl, 1);
if (rc < 0) {
syslog(LOG_ERR,
"failed to nonblock hctl for %s", alsa_card->name);
goto error_bail;
}
rc = snd_hctl_load(alsa_card->hctl);
if (rc < 0) {
syslog(LOG_ERR,
"failed to load hctl for %s", alsa_card->name);
goto error_bail;
}
}
/* Create one mixer per card. */
alsa_card->mixer = cras_alsa_mixer_create(alsa_card->name);
if (alsa_card->mixer == NULL) {
syslog(LOG_ERR, "Fail opening mixer for %s.", alsa_card->name);
goto error_bail;
}
if (alsa_card->ucm && ucm_has_fully_specified_ucm_flag(alsa_card->ucm))
rc = add_controls_and_iodevs_with_ucm(
info, alsa_card, card_name, handle);
else
rc = add_controls_and_iodevs_by_matching(
info, blacklist, alsa_card, card_name, handle);
if (rc)
goto error_bail;
configure_echo_reference_dev(alsa_card);
n = alsa_card->hctl ?
snd_hctl_poll_descriptors_count(alsa_card->hctl) : 0;
if (n != 0 && card_has_hctl_jack(alsa_card)) {
struct hctl_poll_fd *registered_fd;
struct pollfd *pollfds;
int i;
pollfds = malloc(n * sizeof(*pollfds));
if (pollfds == NULL) {
rc = -ENOMEM;
goto error_bail;
}
n = snd_hctl_poll_descriptors(alsa_card->hctl, pollfds, n);
for (i = 0; i < n; i++) {
registered_fd = calloc(1, sizeof(*registered_fd));
if (registered_fd == NULL) {
free(pollfds);
rc = -ENOMEM;
goto error_bail;
}
registered_fd->fd = pollfds[i].fd;
DL_APPEND(alsa_card->hctl_poll_fds, registered_fd);
rc = cras_system_add_select_fd(
registered_fd->fd,
alsa_control_event_pending,
alsa_card);
if (rc < 0) {
DL_DELETE(alsa_card->hctl_poll_fds,
registered_fd);
free(pollfds);
goto error_bail;
}
}
free(pollfds);
}
snd_ctl_close(handle);
return alsa_card;
error_bail:
if (handle != NULL)
snd_ctl_close(handle);
cras_alsa_card_destroy(alsa_card);
return NULL;
}
void cras_alsa_card_destroy(struct cras_alsa_card *alsa_card)
{
struct iodev_list_node *curr;
struct hctl_poll_fd *poll_fd;
if (alsa_card == NULL)
return;
DL_FOREACH(alsa_card->iodevs, curr) {
alsa_iodev_destroy(curr->iodev);
DL_DELETE(alsa_card->iodevs, curr);
free(curr);
}
DL_FOREACH(alsa_card->hctl_poll_fds, poll_fd) {
cras_system_rm_select_fd(poll_fd->fd);
DL_DELETE(alsa_card->hctl_poll_fds, poll_fd);
free(poll_fd);
}
if (alsa_card->hctl)
snd_hctl_close(alsa_card->hctl);
if (alsa_card->ucm)
ucm_destroy(alsa_card->ucm);
if (alsa_card->mixer)
cras_alsa_mixer_destroy(alsa_card->mixer);
if (alsa_card->config)
cras_card_config_destroy(alsa_card->config);
free(alsa_card);
}
size_t cras_alsa_card_get_index(const struct cras_alsa_card *alsa_card)
{
assert(alsa_card);
return alsa_card->card_index;
}