Blame src/control/namehint.c

Packit Service db8eaa
/**
Packit Service db8eaa
 * \file control/namehint.c
Packit Service db8eaa
 * \brief Give device name hints
Packit Service db8eaa
 * \author Jaroslav Kysela <perex@perex.cz>
Packit Service db8eaa
 * \date 2006
Packit Service db8eaa
 */
Packit Service db8eaa
/*
Packit Service db8eaa
 *  Give device name hints  - main file
Packit Service db8eaa
 *  Copyright (c) 2006 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 "local.h"
Packit Service db8eaa
Packit Service db8eaa
#ifndef DOC_HIDDEN
Packit Service db8eaa
#define DEV_SKIP	9999 /* some non-existing device number */
Packit Service db8eaa
struct hint_list {
Packit Service db8eaa
	char **list;
Packit Service db8eaa
	unsigned int count;
Packit Service db8eaa
	unsigned int allocated;
Packit Service db8eaa
	const char *siface;
Packit Service db8eaa
	snd_ctl_elem_iface_t iface;
Packit Service db8eaa
	snd_ctl_t *ctl;
Packit Service db8eaa
	snd_ctl_card_info_t *info;	
Packit Service db8eaa
	int card;
Packit Service db8eaa
	int device;
Packit Service db8eaa
	long device_input;
Packit Service db8eaa
	long device_output;
Packit Service db8eaa
	int stream;
Packit Service db8eaa
	int show_all;
Packit Service db8eaa
	char *cardname;
Packit Service db8eaa
};
Packit Service db8eaa
#endif
Packit Service db8eaa
Packit Service db8eaa
static int hint_list_add(struct hint_list *list,
Packit Service db8eaa
			 const char *name,
Packit Service db8eaa
			 const char *description)
Packit Service db8eaa
{
Packit Service db8eaa
	char *x;
Packit Service db8eaa
Packit Service db8eaa
	if (list->count + 1 >= list->allocated) {
Packit Service db8eaa
		char **n = realloc(list->list, (list->allocated + 10) * sizeof(char *));
Packit Service db8eaa
		if (n == NULL)
Packit Service db8eaa
			return -ENOMEM;
Packit Service db8eaa
		memset(n + list->allocated, 0, 10 * sizeof(*n));
Packit Service db8eaa
		list->allocated += 10;
Packit Service db8eaa
		list->list = n;
Packit Service db8eaa
	}
Packit Service db8eaa
	if (name == NULL) {
Packit Service db8eaa
		x = NULL;
Packit Service db8eaa
	} else {
Packit Service db8eaa
		x = malloc(4 + strlen(name) + (description != NULL ? (4 + strlen(description) + 1) : 0) + 1);
Packit Service db8eaa
		if (x == NULL)
Packit Service db8eaa
			return -ENOMEM;
Packit Service db8eaa
		memcpy(x, "NAME", 4);
Packit Service db8eaa
		strcpy(x + 4, name);
Packit Service db8eaa
		if (description != NULL) {
Packit Service db8eaa
			strcat(x, "|DESC");
Packit Service db8eaa
			strcat(x, description);
Packit Service db8eaa
		}
Packit Service db8eaa
	}
Packit Service db8eaa
	list->list[list->count++] = x;
Packit Service db8eaa
	return 0;
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
/**
Packit Service db8eaa
 * Add a namehint from string given in a user configuration file
Packit Service db8eaa
 */
Packit Service db8eaa
static int hint_list_add_custom(struct hint_list *list,
Packit Service db8eaa
				const char *entry)
Packit Service db8eaa
{
Packit Service db8eaa
	int err;
Packit Service db8eaa
	const char *sep;
Packit Service db8eaa
	char *name;
Packit Service db8eaa
Packit Service db8eaa
	assert(entry);
Packit Service db8eaa
Packit Service db8eaa
	sep = strchr(entry, '|');
Packit Service db8eaa
	if (sep == NULL)
Packit Service db8eaa
		return hint_list_add(list, entry, NULL);
Packit Service db8eaa
Packit Service db8eaa
	name = strndup(entry, sep - entry);
Packit Service db8eaa
	if (name == NULL)
Packit Service db8eaa
		return -ENOMEM;
Packit Service db8eaa
Packit Service db8eaa
	err = hint_list_add(list, name, sep + 1);
Packit Service db8eaa
	free(name);
Packit Service db8eaa
	return err;
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
static void zero_handler(const char *file ATTRIBUTE_UNUSED,
Packit Service db8eaa
			 int line ATTRIBUTE_UNUSED,
Packit Service db8eaa
			 const char *function ATTRIBUTE_UNUSED,
Packit Service db8eaa
			 int err ATTRIBUTE_UNUSED,
Packit Service db8eaa
			 const char *fmt ATTRIBUTE_UNUSED,
Packit Service db8eaa
			 va_list arg ATTRIBUTE_UNUSED)
Packit Service db8eaa
{
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
static int get_dev_name1(struct hint_list *list, char **res, int device,
Packit Service db8eaa
			 int stream)
Packit Service db8eaa
{
Packit Service db8eaa
	*res = NULL;
Packit Service db8eaa
	if (device < 0 || device == DEV_SKIP)
Packit Service db8eaa
		return 0;
Packit Service db8eaa
	switch (list->iface) {
Packit Service db8eaa
#ifdef BUILD_HWDEP
Packit Service db8eaa
	case SND_CTL_ELEM_IFACE_HWDEP:
Packit Service db8eaa
		{
Packit Service db8eaa
			snd_hwdep_info_t info = {0};
Packit Service db8eaa
			snd_hwdep_info_set_device(&info, device);
Packit Service db8eaa
			if (snd_ctl_hwdep_info(list->ctl, &info) < 0)
Packit Service db8eaa
				return 0;
Packit Service db8eaa
			*res = strdup(snd_hwdep_info_get_name(&info));
Packit Service db8eaa
			return 0;
Packit Service db8eaa
		}
Packit Service db8eaa
#endif
Packit Service db8eaa
#ifdef BUILD_PCM
Packit Service db8eaa
	case SND_CTL_ELEM_IFACE_PCM:
Packit Service db8eaa
		{
Packit Service db8eaa
			snd_pcm_info_t info = {0};
Packit Service db8eaa
			snd_pcm_info_set_device(&info, device);
Packit Service db8eaa
			snd_pcm_info_set_stream(&info, stream ? SND_PCM_STREAM_CAPTURE : SND_PCM_STREAM_PLAYBACK);
Packit Service db8eaa
			if (snd_ctl_pcm_info(list->ctl, &info) < 0)
Packit Service db8eaa
				return 0;
Packit Service db8eaa
			switch (snd_pcm_info_get_class(&info)) {
Packit Service db8eaa
			case SND_PCM_CLASS_MODEM:
Packit Service db8eaa
			case SND_PCM_CLASS_DIGITIZER:
Packit Service db8eaa
				return -ENODEV;
Packit Service db8eaa
			default:
Packit Service db8eaa
				break;
Packit Service db8eaa
			}
Packit Service db8eaa
			*res = strdup(snd_pcm_info_get_name(&info));
Packit Service db8eaa
			return 0;
Packit Service db8eaa
		}
Packit Service db8eaa
#endif
Packit Service db8eaa
#ifdef BUILD_RAWMIDI
Packit Service db8eaa
	case SND_CTL_ELEM_IFACE_RAWMIDI:
Packit Service db8eaa
		{
Packit Service db8eaa
			snd_rawmidi_info_t info = {0};
Packit Service db8eaa
			snd_rawmidi_info_set_device(&info, device);
Packit Service db8eaa
			snd_rawmidi_info_set_stream(&info, stream ? SND_RAWMIDI_STREAM_INPUT : SND_RAWMIDI_STREAM_OUTPUT);
Packit Service db8eaa
			if (snd_ctl_rawmidi_info(list->ctl, &info) < 0)
Packit Service db8eaa
				return 0;
Packit Service db8eaa
			*res = strdup(snd_rawmidi_info_get_name(&info));
Packit Service db8eaa
			return 0;
Packit Service db8eaa
		}
Packit Service db8eaa
#endif
Packit Service db8eaa
	default:
Packit Service db8eaa
		return 0;
Packit Service db8eaa
	}
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
static char *get_dev_name(struct hint_list *list)
Packit Service db8eaa
{
Packit Service db8eaa
	char *str1, *str2, *res;
Packit Service db8eaa
	int device;
Packit Service db8eaa
	
Packit Service db8eaa
	device = list->device_input >= 0 ? list->device_input : list->device;
Packit Service db8eaa
	if (get_dev_name1(list, &str1, device, 1) < 0)
Packit Service db8eaa
		return NULL;
Packit Service db8eaa
	device = list->device_output >= 0 ? list->device_output : list->device;
Packit Service db8eaa
	if (get_dev_name1(list, &str2, device, 0) < 0) {
Packit Service db8eaa
		if (str1)
Packit Service db8eaa
			free(str1);
Packit Service db8eaa
		return NULL;
Packit Service db8eaa
	}
Packit Service db8eaa
	if (str1 != NULL || str2 != NULL) {
Packit Service db8eaa
		if (str1 != NULL && str2 != NULL) {
Packit Service db8eaa
			if (strcmp(str1, str2) == 0) {
Packit Service db8eaa
				res = malloc(strlen(list->cardname) + strlen(str2) + 3);
Packit Service db8eaa
				if (res != NULL) {
Packit Service db8eaa
					strcpy(res, list->cardname);
Packit Service db8eaa
					strcat(res, ", ");
Packit Service db8eaa
					strcat(res, str2);
Packit Service db8eaa
				}
Packit Service db8eaa
			} else {
Packit Service db8eaa
				res = malloc(strlen(list->cardname) + strlen(str2) + strlen(str1) + 6);
Packit Service db8eaa
				if (res != NULL) {
Packit Service db8eaa
					strcpy(res, list->cardname);
Packit Service db8eaa
					strcat(res, ", ");
Packit Service db8eaa
					strcat(res, str2);
Packit Service db8eaa
					strcat(res, " / ");
Packit Service db8eaa
					strcat(res, str1);
Packit Service db8eaa
				}
Packit Service db8eaa
			}
Packit Service db8eaa
			free(str2);
Packit Service db8eaa
			free(str1);
Packit Service db8eaa
			return res;
Packit Service db8eaa
		} else {
Packit Service db8eaa
			if (str1 != NULL) {
Packit Service db8eaa
				str2 = "Input";
Packit Service db8eaa
			} else {
Packit Service db8eaa
				str1 = str2;
Packit Service db8eaa
				str2 = "Output";
Packit Service db8eaa
			}
Packit Service db8eaa
			res = malloc(strlen(list->cardname) + strlen(str1) + 19);
Packit Service db8eaa
			if (res == NULL) {
Packit Service db8eaa
				free(str1);
Packit Service db8eaa
				return NULL;
Packit Service db8eaa
			}
Packit Service db8eaa
			strcpy(res, list->cardname);
Packit Service db8eaa
			strcat(res, ", ");
Packit Service db8eaa
			strcat(res, str1);
Packit Service db8eaa
			strcat(res, "|IOID");
Packit Service db8eaa
			strcat(res, str2);
Packit Service db8eaa
			free(str1);
Packit Service db8eaa
			return res;
Packit Service db8eaa
		}
Packit Service db8eaa
	}
Packit Service db8eaa
	/* if the specified device doesn't exist, skip this entry */
Packit Service db8eaa
	if (list->device >= 0 || list->device_input >= 0 || list->device_output >= 0)
Packit Service db8eaa
		return NULL;
Packit Service db8eaa
	return strdup(list->cardname);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
#ifndef DOC_HIDDEN
Packit Service db8eaa
#define BUF_SIZE 128
Packit Service db8eaa
#endif
Packit Service db8eaa
Packit Service db8eaa
static int try_config(snd_config_t *config,
Packit Service db8eaa
		      struct hint_list *list,
Packit Service db8eaa
		      const char *base,
Packit Service db8eaa
		      const char *name)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_local_error_handler_t eh;
Packit Service db8eaa
	snd_config_t *res = NULL, *cfg, *cfg1, *n;
Packit Service db8eaa
	snd_config_iterator_t i, next;
Packit Service db8eaa
	char *buf, *buf1 = NULL, *buf2;
Packit Service db8eaa
	const char *str;
Packit Service db8eaa
	int err = 0, level;
Packit Service db8eaa
	long dev = list->device;
Packit Service db8eaa
	int cleanup_res = 0;
Packit Service db8eaa
Packit Service db8eaa
	list->device_input = -1;
Packit Service db8eaa
	list->device_output = -1;
Packit Service db8eaa
	buf = malloc(BUF_SIZE);
Packit Service db8eaa
	if (buf == NULL)
Packit Service db8eaa
		return -ENOMEM;
Packit Service db8eaa
	sprintf(buf, "%s.%s", base, name);
Packit Service db8eaa
	/* look for redirection */
Packit Service db8eaa
	if (snd_config_search(config, buf, &cfg) >= 0 &&
Packit Service db8eaa
	    snd_config_get_string(cfg, &str) >= 0 &&
Packit Service db8eaa
	    ((strncmp(base, str, strlen(base)) == 0 &&
Packit Service db8eaa
	     str[strlen(base)] == '.') || strchr(str, '.') == NULL))
Packit Service db8eaa
	     	goto __skip_add;
Packit Service db8eaa
	if (list->card >= 0 && list->device >= 0)
Packit Service db8eaa
		sprintf(buf, "%s:CARD=%s,DEV=%i", name, snd_ctl_card_info_get_id(list->info), list->device);
Packit Service db8eaa
	else if (list->card >= 0)
Packit Service db8eaa
		sprintf(buf, "%s:CARD=%s", name, snd_ctl_card_info_get_id(list->info));
Packit Service db8eaa
	else
Packit Service db8eaa
		strcpy(buf, name);
Packit Service db8eaa
	eh = snd_lib_error_set_local(&zero_handler);
Packit Service db8eaa
	err = snd_config_search_definition(config, base, buf, &res;;
Packit Service db8eaa
	snd_lib_error_set_local(eh);
Packit Service db8eaa
	if (err < 0)
Packit Service db8eaa
		goto __skip_add;
Packit Service db8eaa
	cleanup_res = 1;
Packit Service db8eaa
	err = -EINVAL;
Packit Service db8eaa
	if (snd_config_get_type(res) != SND_CONFIG_TYPE_COMPOUND)
Packit Service db8eaa
		goto __cleanup;
Packit Service db8eaa
	if (snd_config_search(res, "type", NULL) < 0)
Packit Service db8eaa
		goto __cleanup;
Packit Service db8eaa
Packit Service db8eaa
#if 0	/* for debug purposes */
Packit Service db8eaa
		{
Packit Service db8eaa
			snd_output_t *out;
Packit Service db8eaa
			fprintf(stderr, "********* PCM '%s':\n", buf);
Packit Service db8eaa
			snd_output_stdio_attach(&out, stderr, 0);
Packit Service db8eaa
			snd_config_save(res, out);
Packit Service db8eaa
			snd_output_close(out);
Packit Service db8eaa
			fprintf(stderr, "\n");
Packit Service db8eaa
		}
Packit Service db8eaa
#endif
Packit Service db8eaa
Packit Service db8eaa
	cfg1 = res;
Packit Service db8eaa
	level = 0;
Packit Service db8eaa
      __hint:
Packit Service db8eaa
      	level++;
Packit Service db8eaa
	if (snd_config_search(cfg1, "type", &cfg) >= 0 &&
Packit Service db8eaa
	    snd_config_get_string(cfg, &str) >= 0 &&
Packit Service db8eaa
	    strcmp(str, "hw") == 0) {
Packit Service db8eaa
		if (snd_config_search(cfg1, "device", &cfg) >= 0) {
Packit Service db8eaa
			if (snd_config_get_integer(cfg, &dev) < 0) {
Packit Service db8eaa
				SNDERR("(%s) device must be an integer", buf);
Packit Service db8eaa
				err = -EINVAL;
Packit Service db8eaa
				goto __cleanup;
Packit Service db8eaa
			}
Packit Service db8eaa
		}
Packit Service db8eaa
	}
Packit Service db8eaa
      	
Packit Service db8eaa
	if (snd_config_search(cfg1, "hint", &cfg) >= 0) {
Packit Service db8eaa
		if (snd_config_get_type(cfg) != SND_CONFIG_TYPE_COMPOUND) {
Packit Service db8eaa
			SNDERR("hint (%s) must be a compound", buf);
Packit Service db8eaa
			err = -EINVAL;
Packit Service db8eaa
			goto __cleanup;
Packit Service db8eaa
		}
Packit Service db8eaa
		if (list->card < 0 &&
Packit Service db8eaa
		    snd_config_search(cfg, "omit_noargs", &n) >= 0 &&
Packit Service db8eaa
		    snd_config_get_bool(n) > 0)
Packit Service db8eaa
			goto __skip_add;
Packit Service db8eaa
		if (level == 1 &&
Packit Service db8eaa
		    snd_config_search(cfg, "show", &n) >= 0 &&
Packit Service db8eaa
		    snd_config_get_bool(n) <= 0)
Packit Service db8eaa
			goto __skip_add;
Packit Service db8eaa
		if (buf1 == NULL &&
Packit Service db8eaa
		    snd_config_search(cfg, "description", &n) >= 0 &&
Packit Service db8eaa
		    snd_config_get_string(n, &str) >= 0) {
Packit Service db8eaa
			buf1 = strdup(str);
Packit Service db8eaa
			if (buf1 == NULL) {
Packit Service db8eaa
				err = -ENOMEM;
Packit Service db8eaa
				goto __cleanup;
Packit Service db8eaa
			}
Packit Service db8eaa
		}
Packit Service db8eaa
		if (snd_config_search(cfg, "device", &n) >= 0) {
Packit Service db8eaa
			if (snd_config_get_integer(n, &dev) < 0) {
Packit Service db8eaa
				SNDERR("(%s) device must be an integer", buf);
Packit Service db8eaa
				err = -EINVAL;
Packit Service db8eaa
				goto __cleanup;
Packit Service db8eaa
			}
Packit Service db8eaa
			list->device_input = dev;
Packit Service db8eaa
			list->device_output = dev;
Packit Service db8eaa
		}
Packit Service db8eaa
		if (snd_config_search(cfg, "device_input", &n) >= 0) {
Packit Service db8eaa
			if (snd_config_get_integer(n, &list->device_input) < 0) {
Packit Service db8eaa
				SNDERR("(%s) device_input must be an integer", buf);
Packit Service db8eaa
				err = -EINVAL;
Packit Service db8eaa
				goto __cleanup;
Packit Service db8eaa
			}
Packit Service db8eaa
			/* skip the counterpart if only a single direction is defined */
Packit Service db8eaa
			if (list->device_output < 0)
Packit Service db8eaa
				list->device_output = DEV_SKIP;
Packit Service db8eaa
		}
Packit Service db8eaa
		if (snd_config_search(cfg, "device_output", &n) >= 0) {
Packit Service db8eaa
			if (snd_config_get_integer(n, &list->device_output) < 0) {
Packit Service db8eaa
				SNDERR("(%s) device_output must be an integer", buf);
Packit Service db8eaa
				err = -EINVAL;
Packit Service db8eaa
				goto __cleanup;
Packit Service db8eaa
			}
Packit Service db8eaa
			/* skip the counterpart if only a single direction is defined */
Packit Service db8eaa
			if (list->device_input < 0)
Packit Service db8eaa
				list->device_input = DEV_SKIP;
Packit Service db8eaa
		}
Packit Service db8eaa
	} else if (level == 1 && !list->show_all)
Packit Service db8eaa
		goto __skip_add;
Packit Service db8eaa
	if (snd_config_search(cfg1, "slave", &cfg) >= 0 &&
Packit Service db8eaa
	    snd_config_search(cfg, base, &cfg1) >= 0)
Packit Service db8eaa
	    	goto __hint;
Packit Service db8eaa
	snd_config_delete(res);
Packit Service db8eaa
	res = NULL;
Packit Service db8eaa
	cleanup_res = 0;
Packit Service db8eaa
	if (strchr(buf, ':') != NULL)
Packit Service db8eaa
		goto __ok;
Packit Service db8eaa
	/* find, if all parameters have a default, */
Packit Service db8eaa
	/* otherwise filter this definition */
Packit Service db8eaa
	eh = snd_lib_error_set_local(&zero_handler);
Packit Service db8eaa
	err = snd_config_search_alias_hooks(config, base, buf, &res;;
Packit Service db8eaa
	snd_lib_error_set_local(eh);
Packit Service db8eaa
	if (err < 0)
Packit Service db8eaa
		goto __cleanup;
Packit Service db8eaa
	if (snd_config_search(res, "@args", &cfg) >= 0) {
Packit Service db8eaa
		snd_config_for_each(i, next, cfg) {
Packit Service db8eaa
			/* skip the argument list */
Packit Service db8eaa
			if (snd_config_get_id(snd_config_iterator_entry(i), &str) < 0)
Packit Service db8eaa
				continue;
Packit Service db8eaa
			while (*str && *str >= '0' && *str <= '9') str++;
Packit Service db8eaa
			if (*str == '\0')
Packit Service db8eaa
				continue;
Packit Service db8eaa
			/* the argument definition must have the default */
Packit Service db8eaa
			if (snd_config_search(snd_config_iterator_entry(i),
Packit Service db8eaa
					      "default", NULL) < 0) {
Packit Service db8eaa
				err = -EINVAL;
Packit Service db8eaa
				goto __cleanup;
Packit Service db8eaa
			}
Packit Service db8eaa
		}
Packit Service db8eaa
	}
Packit Service db8eaa
      __ok:
Packit Service db8eaa
	err = 0;
Packit Service db8eaa
      __cleanup:
Packit Service db8eaa
      	if (err >= 0) {
Packit Service db8eaa
      		list->device = dev;
Packit Service db8eaa
 		str = list->card >= 0 ? get_dev_name(list) : NULL;
Packit Service db8eaa
      		if (str != NULL) {
Packit Service db8eaa
      			level = (buf1 == NULL ? 0 : strlen(buf1)) + 1 + strlen(str);
Packit Service db8eaa
      			buf2 = realloc((char *)str, level + 1);
Packit Service db8eaa
      			if (buf2 != NULL) {
Packit Service db8eaa
      				if (buf1 != NULL) {
Packit Service db8eaa
      					str = strchr(buf2, '|');
Packit Service db8eaa
      					if (str != NULL)
Packit Service db8eaa
						memmove(buf2 + (level - strlen(str)), str, strlen(str));
Packit Service db8eaa
					else
Packit Service db8eaa
						str = buf2 + strlen(buf2);
Packit Service db8eaa
      					*(char *)str++ = '\n';
Packit Service db8eaa
	      				memcpy((char *)str, buf1, strlen(buf1));
Packit Service db8eaa
	      				buf2[level] = '\0';
Packit Service db8eaa
					free(buf1);
Packit Service db8eaa
				}
Packit Service db8eaa
				buf1 = buf2;
Packit Service db8eaa
			} else {
Packit Service db8eaa
				free((char *)str);
Packit Service db8eaa
			}
Packit Service db8eaa
      		} else if (list->device >= 0)
Packit Service db8eaa
      			goto __skip_add;
Packit Service db8eaa
	      	err = hint_list_add(list, buf, buf1);
Packit Service db8eaa
	}
Packit Service db8eaa
      __skip_add:
Packit Service db8eaa
	if (res && cleanup_res)
Packit Service db8eaa
	      	snd_config_delete(res);
Packit Service db8eaa
	if (buf1)
Packit Service db8eaa
		free(buf1);
Packit Service db8eaa
      	free(buf);
Packit Service db8eaa
	return err;
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
#ifndef DOC_HIDDEN
Packit Service db8eaa
#define IFACE(v, fcn) [SND_CTL_ELEM_IFACE_##v] = (next_devices_t)fcn
Packit Service db8eaa
Packit Service db8eaa
typedef int (*next_devices_t)(snd_ctl_t *, int *);
Packit Service db8eaa
Packit Service db8eaa
static const next_devices_t next_devices[] = {
Packit Service db8eaa
	IFACE(CARD, NULL),
Packit Service db8eaa
	IFACE(HWDEP, snd_ctl_hwdep_next_device),
Packit Service db8eaa
	IFACE(MIXER, NULL),
Packit Service db8eaa
	IFACE(PCM, snd_ctl_pcm_next_device),
Packit Service db8eaa
	IFACE(RAWMIDI, snd_ctl_rawmidi_next_device),
Packit Service db8eaa
	IFACE(TIMER, NULL),
Packit Service db8eaa
	IFACE(SEQUENCER, NULL)
Packit Service db8eaa
};
Packit Service db8eaa
#endif
Packit Service db8eaa
Packit Service db8eaa
static int add_card(snd_config_t *config, snd_config_t *rw_config, struct hint_list *list, int card)
Packit Service db8eaa
{
Packit Service db8eaa
	int err, ok;
Packit Service db8eaa
	snd_config_t *conf, *n;
Packit Service db8eaa
	snd_config_iterator_t i, next;
Packit Service db8eaa
	const char *str;
Packit Service db8eaa
	char ctl_name[16];
Packit Service db8eaa
	snd_ctl_card_info_t info = {0};
Packit Service db8eaa
	int device, max_device = 0;
Packit Service db8eaa
	
Packit Service db8eaa
	list->info = &info;
Packit Service db8eaa
	err = snd_config_search(config, list->siface, &conf;;
Packit Service db8eaa
	if (err < 0)
Packit Service db8eaa
		return err;
Packit Service db8eaa
	sprintf(ctl_name, "hw:%i", card);
Packit Service db8eaa
	err = snd_ctl_open(&list->ctl, ctl_name, 0);
Packit Service db8eaa
	if (err < 0)
Packit Service db8eaa
		return err;
Packit Service db8eaa
	err = snd_ctl_card_info(list->ctl, &info;;
Packit Service db8eaa
	if (err < 0)
Packit Service db8eaa
		goto __error;
Packit Service db8eaa
	snd_config_for_each(i, next, conf) {
Packit Service db8eaa
		n = snd_config_iterator_entry(i);
Packit Service db8eaa
		if (snd_config_get_id(n, &str) < 0)
Packit Service db8eaa
			continue;
Packit Service db8eaa
		
Packit Service db8eaa
		if (next_devices[list->iface] != NULL) {
Packit Service db8eaa
			list->card = card;
Packit Service db8eaa
			device = max_device = -1;
Packit Service db8eaa
			err = next_devices[list->iface](list->ctl, &device);
Packit Service db8eaa
			if (device < 0)
Packit Service db8eaa
				err = -EINVAL;
Packit Service db8eaa
			else
Packit Service db8eaa
				max_device = device;
Packit Service db8eaa
			while (err >= 0 && device >= 0) {
Packit Service db8eaa
				err = next_devices[list->iface](list->ctl, &device);
Packit Service db8eaa
				if (err >= 0 && device > max_device)
Packit Service db8eaa
					max_device = device;
Packit Service db8eaa
			}
Packit Service db8eaa
			ok = 0;
Packit Service db8eaa
			for (device = 0; err >= 0 && device <= max_device; device++) {
Packit Service db8eaa
				list->device = device;
Packit Service db8eaa
				err = try_config(rw_config, list, list->siface, str);
Packit Service db8eaa
				if (err < 0)
Packit Service db8eaa
					break;
Packit Service db8eaa
				ok++;
Packit Service db8eaa
			}
Packit Service db8eaa
			if (ok)
Packit Service db8eaa
				continue;
Packit Service db8eaa
		} else {
Packit Service db8eaa
			err = -EINVAL;
Packit Service db8eaa
		}
Packit Service db8eaa
		if (err == -EXDEV)
Packit Service db8eaa
			continue;
Packit Service db8eaa
		if (err < 0) {
Packit Service db8eaa
			list->card = card;
Packit Service db8eaa
			list->device = -1;
Packit Service db8eaa
			err = try_config(rw_config, list, list->siface, str);
Packit Service db8eaa
		}
Packit Service db8eaa
		if (err == -ENOMEM)
Packit Service db8eaa
			goto __error;
Packit Service db8eaa
	}
Packit Service db8eaa
	err = 0;
Packit Service db8eaa
      __error:
Packit Service db8eaa
      	snd_ctl_close(list->ctl);
Packit Service db8eaa
	return err;
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
static int get_card_name(struct hint_list *list, int card)
Packit Service db8eaa
{
Packit Service db8eaa
	char scard[16], *s;
Packit Service db8eaa
	int err;
Packit Service db8eaa
Packit Service db8eaa
	free(list->cardname);
Packit Service db8eaa
	list->cardname = NULL;
Packit Service db8eaa
	err = snd_card_get_name(card, &list->cardname);
Packit Service db8eaa
	if (err <= 0)
Packit Service db8eaa
		return 0;
Packit Service db8eaa
	sprintf(scard, " #%i", card);
Packit Service db8eaa
	s = realloc(list->cardname, strlen(list->cardname) + strlen(scard) + 1);
Packit Service db8eaa
	if (s == NULL)
Packit Service db8eaa
		return -ENOMEM;
Packit Service db8eaa
	list->cardname = s;
Packit Service db8eaa
	return 0;
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
static int add_software_devices(snd_config_t *config, snd_config_t *rw_config,
Packit Service db8eaa
				struct hint_list *list)
Packit Service db8eaa
{
Packit Service db8eaa
	int err;
Packit Service db8eaa
	snd_config_t *conf, *n;
Packit Service db8eaa
	snd_config_iterator_t i, next;
Packit Service db8eaa
	const char *str;
Packit Service db8eaa
Packit Service db8eaa
	err = snd_config_search(config, list->siface, &conf;;
Packit Service db8eaa
	if (err < 0)
Packit Service db8eaa
		return err;
Packit Service db8eaa
	snd_config_for_each(i, next, conf) {
Packit Service db8eaa
		n = snd_config_iterator_entry(i);
Packit Service db8eaa
		if (snd_config_get_id(n, &str) < 0)
Packit Service db8eaa
			continue;
Packit Service db8eaa
		list->card = -1;
Packit Service db8eaa
		list->device = -1;
Packit Service db8eaa
		err = try_config(rw_config, list, list->siface, str);
Packit Service db8eaa
		if (err == -ENOMEM)
Packit Service db8eaa
			return -ENOMEM;
Packit Service db8eaa
	}
Packit Service db8eaa
	return 0;
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
/**
Packit Service db8eaa
 * \brief Get a set of device name hints
Packit Service db8eaa
 * \param card Card number or -1 (means all cards)
Packit Service db8eaa
 * \param iface Interface identification (like "pcm", "rawmidi", "timer", "seq")
Packit Service db8eaa
 * \param hints Result - array of device name hints
Packit Service db8eaa
 * \result zero if success, otherwise a negative error code
Packit Service db8eaa
 *
Packit Service db8eaa
 * hints will receive a NULL-terminated array of device name hints,
Packit Service db8eaa
 * which can be passed to #snd_device_name_get_hint to extract usable
Packit Service db8eaa
 * values. When no longer needed, hints should be passed to
Packit Service db8eaa
 * #snd_device_name_free_hint to release resources.
Packit Service db8eaa
 *
Packit Service db8eaa
 * User-defined hints are gathered from namehint.IFACE tree like:
Packit Service db8eaa
 *
Packit Service db8eaa
 * 
Packit Service db8eaa
 * namehint.pcm [
Packit Service db8eaa
 *   myfile "file:FILE=/tmp/soundwave.raw|Save sound output to /tmp/soundwave.raw"
Packit Service db8eaa
 *   myplug "plug:front|Do all conversions for front speakers"
Packit Service db8eaa
 * ]
Packit Service db8eaa
 * 
Packit Service db8eaa
 *
Packit Service db8eaa
 * Note: The device description is separated with '|' char.
Packit Service db8eaa
 *
Packit Service db8eaa
 * Special variables: defaults.namehint.showall specifies if all device
Packit Service db8eaa
 * definitions are accepted (boolean type).
Packit Service db8eaa
 */
Packit Service db8eaa
int snd_device_name_hint(int card, const char *iface, void ***hints)
Packit Service db8eaa
{
Packit Service db8eaa
	struct hint_list list;
Packit Service db8eaa
	char ehints[24];
Packit Service db8eaa
	const char *str;
Packit Service db8eaa
	snd_config_t *conf, *local_config = NULL, *local_config_rw = NULL;
Packit Service db8eaa
	snd_config_update_t *local_config_update = NULL;
Packit Service db8eaa
	snd_config_iterator_t i, next;
Packit Service db8eaa
	int err;
Packit Service db8eaa
Packit Service db8eaa
	if (hints == NULL)
Packit Service db8eaa
		return -EINVAL;
Packit Service db8eaa
	err = snd_config_update_r(&local_config, &local_config_update, NULL);
Packit Service db8eaa
	if (err < 0)
Packit Service db8eaa
		return err;
Packit Service db8eaa
	err = snd_config_copy(&local_config_rw, local_config);
Packit Service db8eaa
	if (err < 0)
Packit Service db8eaa
		return err;
Packit Service db8eaa
	list.list = NULL;
Packit Service db8eaa
	list.count = list.allocated = 0;
Packit Service db8eaa
	list.siface = iface;
Packit Service db8eaa
	list.show_all = 0;
Packit Service db8eaa
	list.cardname = NULL;
Packit Service db8eaa
	if (strcmp(iface, "card") == 0)
Packit Service db8eaa
		list.iface = SND_CTL_ELEM_IFACE_CARD;
Packit Service db8eaa
	else if (strcmp(iface, "pcm") == 0)
Packit Service db8eaa
		list.iface = SND_CTL_ELEM_IFACE_PCM;
Packit Service db8eaa
	else if (strcmp(iface, "rawmidi") == 0)
Packit Service db8eaa
		list.iface = SND_CTL_ELEM_IFACE_RAWMIDI;
Packit Service db8eaa
	else if (strcmp(iface, "timer") == 0)
Packit Service db8eaa
		list.iface = SND_CTL_ELEM_IFACE_TIMER;
Packit Service db8eaa
	else if (strcmp(iface, "seq") == 0)
Packit Service db8eaa
		list.iface = SND_CTL_ELEM_IFACE_SEQUENCER;
Packit Service db8eaa
	else if (strcmp(iface, "hwdep") == 0)
Packit Service db8eaa
		list.iface = SND_CTL_ELEM_IFACE_HWDEP;
Packit Service db8eaa
	else if (strcmp(iface, "ctl") == 0)
Packit Service db8eaa
		list.iface = SND_CTL_ELEM_IFACE_MIXER;
Packit Service db8eaa
	else {
Packit Service db8eaa
		err = -EINVAL;
Packit Service db8eaa
		goto __error;
Packit Service db8eaa
	}
Packit Service db8eaa
Packit Service db8eaa
	if (snd_config_search(local_config, "defaults.namehint.showall", &conf) >= 0)
Packit Service db8eaa
		list.show_all = snd_config_get_bool(conf) > 0;
Packit Service db8eaa
	if (card >= 0) {
Packit Service db8eaa
		err = get_card_name(&list, card);
Packit Service db8eaa
		if (err >= 0)
Packit Service db8eaa
			err = add_card(local_config, local_config_rw, &list, card);
Packit Service db8eaa
	} else {
Packit Service db8eaa
		add_software_devices(local_config, local_config_rw, &list);
Packit Service db8eaa
		err = snd_card_next(&card;;
Packit Service db8eaa
		if (err < 0)
Packit Service db8eaa
			goto __error;
Packit Service db8eaa
		while (card >= 0) {
Packit Service db8eaa
			err = get_card_name(&list, card);
Packit Service db8eaa
			if (err < 0)
Packit Service db8eaa
				goto __error;
Packit Service db8eaa
			err = add_card(local_config, local_config_rw, &list, card);
Packit Service db8eaa
			if (err < 0)
Packit Service db8eaa
				goto __error;
Packit Service db8eaa
			err = snd_card_next(&card;;
Packit Service db8eaa
			if (err < 0)
Packit Service db8eaa
				goto __error;
Packit Service db8eaa
		}
Packit Service db8eaa
	}
Packit Service db8eaa
	sprintf(ehints, "namehint.%s", list.siface);
Packit Service db8eaa
	err = snd_config_search(local_config, ehints, &conf;;
Packit Service db8eaa
	if (err >= 0) {
Packit Service db8eaa
		snd_config_for_each(i, next, conf) {
Packit Service db8eaa
			if (snd_config_get_string(snd_config_iterator_entry(i),
Packit Service db8eaa
						  &str) < 0)
Packit Service db8eaa
				continue;
Packit Service db8eaa
			err = hint_list_add_custom(&list, str);
Packit Service db8eaa
			if (err < 0)
Packit Service db8eaa
				goto __error;
Packit Service db8eaa
		}
Packit Service db8eaa
	}
Packit Service db8eaa
	err = 0;
Packit Service db8eaa
      __error:
Packit Service db8eaa
	/* add an empty entry if nothing has been added yet; the caller
Packit Service db8eaa
	 * expects non-NULL return
Packit Service db8eaa
	 */
Packit Service db8eaa
	if (!err && !list.list)
Packit Service db8eaa
		err = hint_list_add(&list, NULL, NULL);
Packit Service db8eaa
	if (err < 0)
Packit Service db8eaa
      		snd_device_name_free_hint((void **)list.list);
Packit Service db8eaa
	else
Packit Service db8eaa
      		*hints = (void **)list.list;
Packit Service db8eaa
	free(list.cardname);
Packit Service db8eaa
	if (local_config_rw)
Packit Service db8eaa
		snd_config_delete(local_config_rw);
Packit Service db8eaa
	if (local_config)
Packit Service db8eaa
		snd_config_delete(local_config);
Packit Service db8eaa
	if (local_config_update)
Packit Service db8eaa
		snd_config_update_free(local_config_update);
Packit Service db8eaa
	return err;
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
/**
Packit Service db8eaa
 * \brief Free a list of device name hints.
Packit Service db8eaa
 * \param hints List to free
Packit Service db8eaa
 * \result zero if success, otherwise a negative error code
Packit Service db8eaa
 */
Packit Service db8eaa
int snd_device_name_free_hint(void **hints)
Packit Service db8eaa
{
Packit Service db8eaa
	char **h;
Packit Service db8eaa
Packit Service db8eaa
	if (hints == NULL)
Packit Service db8eaa
		return 0;
Packit Service db8eaa
	h = (char **)hints;
Packit Service db8eaa
	while (*h) {
Packit Service db8eaa
		free(*h);
Packit Service db8eaa
		h++;
Packit Service db8eaa
	}
Packit Service db8eaa
	free(hints);
Packit Service db8eaa
	return 0;
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
/**
Packit Service db8eaa
 * \brief Extract a value from a hint
Packit Service db8eaa
 * \param hint A pointer to hint
Packit Service db8eaa
 * \param id Hint value to extract ("NAME", "DESC", or "IOID", see below)
Packit Service db8eaa
 * \result an allocated ASCII string if success, otherwise NULL
Packit Service db8eaa
 *
Packit Service db8eaa
 * List of valid IDs:
Packit Service db8eaa
 * NAME - name of device
Packit Service db8eaa
 * DESC - description of device
Packit Service db8eaa
 * IOID - input / output identification ("Input" or "Output"), NULL means both
Packit Service db8eaa
 *
Packit Service db8eaa
 * The return value should be freed when no longer needed.
Packit Service db8eaa
 */
Packit Service db8eaa
char *snd_device_name_get_hint(const void *hint, const char *id)
Packit Service db8eaa
{
Packit Service db8eaa
	const char *hint1 = (const char *)hint, *delim;
Packit Service db8eaa
	char *res;
Packit Service db8eaa
	unsigned size;
Packit Service db8eaa
Packit Service db8eaa
	if (strlen(id) != 4)
Packit Service db8eaa
		return NULL;
Packit Service db8eaa
	while (*hint1 != '\0') {
Packit Service db8eaa
		delim = strchr(hint1, '|');
Packit Service db8eaa
		if (memcmp(id, hint1, 4) != 0) {
Packit Service db8eaa
			if (delim == NULL)
Packit Service db8eaa
				return NULL;
Packit Service db8eaa
			hint1 = delim + 1;
Packit Service db8eaa
			continue;
Packit Service db8eaa
		} 
Packit Service db8eaa
		if (delim == NULL)
Packit Service db8eaa
			return strdup(hint1 + 4);
Packit Service db8eaa
		size = delim - hint1 - 4;
Packit Service db8eaa
		res = malloc(size + 1);
Packit Service db8eaa
		if (res != NULL) {
Packit Service db8eaa
			memcpy(res, hint1 + 4, size);
Packit Service db8eaa
			res[size] = '\0';
Packit Service db8eaa
		}
Packit Service db8eaa
		return res;
Packit Service db8eaa
	}
Packit Service db8eaa
	return NULL;
Packit Service db8eaa
}