Blame src/control/setup.c

Packit 4a16fb
/**
Packit 4a16fb
 * \file control/setup.c
Packit 4a16fb
 * \brief Routines to setup control primitives from configuration
Packit 4a16fb
 * \author Abramo Bagnara <abramo@alsa-project.org>
Packit 4a16fb
 * \author Jaroslav Kysela <perex@perex.cz>
Packit 4a16fb
 * \date 2001
Packit 4a16fb
 *
Packit 4a16fb
 * Routines to setup control primitives from configuration
Packit 4a16fb
 */
Packit 4a16fb
/*
Packit 4a16fb
 *  Control Interface - routines for setup from configuration
Packit 4a16fb
 *  Copyright (c) 2001 by Abramo Bagnara <abramo@alsa-project.org>
Packit 4a16fb
 *			  Jaroslav Kysela <perex@perex.cz>
Packit 4a16fb
 *
Packit 4a16fb
 *
Packit 4a16fb
 *   This library is free software; you can redistribute it and/or modify
Packit 4a16fb
 *   it under the terms of the GNU Lesser General Public License as
Packit 4a16fb
 *   published by the Free Software Foundation; either version 2.1 of
Packit 4a16fb
 *   the License, or (at your option) any later version.
Packit 4a16fb
 *
Packit 4a16fb
 *   This program is distributed in the hope that it will be useful,
Packit 4a16fb
 *   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 4a16fb
 *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit 4a16fb
 *   GNU Lesser General Public License for more details.
Packit 4a16fb
 *
Packit 4a16fb
 *   You should have received a copy of the GNU Lesser General Public
Packit 4a16fb
 *   License along with this library; if not, write to the Free Software
Packit 4a16fb
 *   Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
Packit 4a16fb
 *
Packit 4a16fb
 */
Packit 4a16fb
Packit 4a16fb
#include <stdio.h>
Packit 4a16fb
#include <stdlib.h>
Packit 4a16fb
#include <stdarg.h>
Packit 4a16fb
#include <unistd.h>
Packit 4a16fb
#include <string.h>
Packit 4a16fb
#include <ctype.h>
Packit 4a16fb
#include "local.h"
Packit 4a16fb
Packit 4a16fb
#ifndef DOC_HIDDEN
Packit 4a16fb
typedef struct {
Packit 4a16fb
	unsigned int lock: 1;
Packit 4a16fb
	unsigned int preserve: 1;
Packit 4a16fb
	snd_ctl_elem_id_t *id;
Packit 4a16fb
	snd_ctl_elem_info_t *info;
Packit 4a16fb
	snd_ctl_elem_value_t *val;
Packit 4a16fb
	snd_ctl_elem_value_t *mask;
Packit 4a16fb
	snd_ctl_elem_value_t *old;
Packit 4a16fb
	struct list_head list;
Packit 4a16fb
} snd_sctl_elem_t;
Packit 4a16fb
Packit 4a16fb
struct _snd_sctl {
Packit 4a16fb
	int mode;
Packit 4a16fb
	snd_ctl_t *ctl;
Packit 4a16fb
	struct list_head elems;
Packit 4a16fb
};
Packit 4a16fb
#endif /* DOC_HIDDEN */
Packit 4a16fb
Packit 4a16fb
static int free_elems(snd_sctl_t *h)
Packit 4a16fb
{
Packit 4a16fb
	int err = 0;
Packit 4a16fb
	while (!list_empty(&h->elems)) {
Packit 4a16fb
		snd_sctl_elem_t *elem = list_entry(h->elems.next, snd_sctl_elem_t, list);
Packit 4a16fb
		snd_ctl_elem_id_free(elem->id);
Packit 4a16fb
		snd_ctl_elem_info_free(elem->info);
Packit 4a16fb
		snd_ctl_elem_value_free(elem->val);
Packit 4a16fb
		snd_ctl_elem_value_free(elem->mask);
Packit 4a16fb
		snd_ctl_elem_value_free(elem->old);
Packit 4a16fb
		list_del(&elem->list);
Packit 4a16fb
		free(elem);
Packit 4a16fb
	}
Packit 4a16fb
	if ((h->mode & SND_SCTL_NOFREE) == 0)
Packit 4a16fb
		err = snd_ctl_close(h->ctl);
Packit 4a16fb
	free(h);
Packit 4a16fb
	return err;
Packit 4a16fb
}
Packit 4a16fb
Packit 4a16fb
/**
Packit 4a16fb
 * \brief Install given values to control elements
Packit 4a16fb
 * \param h Setup control handle
Packit 4a16fb
 * \result zero if success, otherwise a negative error code
Packit 4a16fb
 */
Packit 4a16fb
int snd_sctl_install(snd_sctl_t *h)
Packit 4a16fb
{
Packit 4a16fb
	struct list_head *pos;
Packit 4a16fb
	int err;
Packit 4a16fb
	unsigned int k;
Packit 4a16fb
	assert(h);
Packit 4a16fb
	list_for_each(pos, &h->elems) {
Packit 4a16fb
		snd_sctl_elem_t *elem = list_entry(pos, snd_sctl_elem_t, list);
Packit 4a16fb
		unsigned int count;
Packit 4a16fb
		snd_ctl_elem_type_t type;
Packit 4a16fb
		if (elem->lock) {
Packit 4a16fb
			err = snd_ctl_elem_lock(h->ctl, elem->id);
Packit 4a16fb
			if (err < 0) {
Packit 4a16fb
				SNDERR("Cannot lock ctl elem");
Packit 4a16fb
				return err;
Packit 4a16fb
			}
Packit 4a16fb
		}
Packit 4a16fb
		err = snd_ctl_elem_read(h->ctl, elem->old);
Packit 4a16fb
		if (err < 0) {
Packit 4a16fb
			SNDERR("Cannot read ctl elem");
Packit 4a16fb
			return err;
Packit 4a16fb
		}
Packit 4a16fb
		count = snd_ctl_elem_info_get_count(elem->info);
Packit 4a16fb
		type = snd_ctl_elem_info_get_type(elem->info);
Packit 4a16fb
		switch (type) {
Packit 4a16fb
		case SND_CTL_ELEM_TYPE_BOOLEAN:
Packit 4a16fb
			for (k = 0; k < count; ++k) {
Packit 4a16fb
				int old, val, mask;
Packit 4a16fb
				old = snd_ctl_elem_value_get_boolean(elem->old, k);
Packit 4a16fb
				mask = snd_ctl_elem_value_get_boolean(elem->mask, k);
Packit 4a16fb
				old &= ~mask;
Packit 4a16fb
				if (old) {
Packit 4a16fb
					val = snd_ctl_elem_value_get_boolean(elem->val, k);
Packit 4a16fb
					val |= old;
Packit 4a16fb
					snd_ctl_elem_value_set_boolean(elem->val, k, val);
Packit 4a16fb
				}
Packit 4a16fb
			}
Packit 4a16fb
			break;
Packit 4a16fb
		case SND_CTL_ELEM_TYPE_INTEGER:
Packit 4a16fb
			for (k = 0; k < count; ++k) {
Packit 4a16fb
				long old, val, mask;
Packit 4a16fb
				old = snd_ctl_elem_value_get_integer(elem->old, k);
Packit 4a16fb
				mask = snd_ctl_elem_value_get_integer(elem->mask, k);
Packit 4a16fb
				old &= ~mask;
Packit 4a16fb
				if (old) {
Packit 4a16fb
					val = snd_ctl_elem_value_get_integer(elem->val, k);
Packit 4a16fb
					val |= old;
Packit 4a16fb
					snd_ctl_elem_value_set_integer(elem->val, k, val);
Packit 4a16fb
				}
Packit 4a16fb
			}
Packit 4a16fb
			break;
Packit 4a16fb
		case SND_CTL_ELEM_TYPE_ENUMERATED:
Packit 4a16fb
			for (k = 0; k < count; ++k) {
Packit 4a16fb
				unsigned int old, val, mask;
Packit 4a16fb
				old = snd_ctl_elem_value_get_enumerated(elem->old, k);
Packit 4a16fb
				mask = snd_ctl_elem_value_get_enumerated(elem->mask, k);
Packit 4a16fb
				old &= ~mask;
Packit 4a16fb
				if (old) {
Packit 4a16fb
					val = snd_ctl_elem_value_get_enumerated(elem->val, k);
Packit 4a16fb
					val |= old;
Packit 4a16fb
					snd_ctl_elem_value_set_enumerated(elem->val, k, val);
Packit 4a16fb
				}
Packit 4a16fb
			}
Packit 4a16fb
			break;
Packit 4a16fb
		case SND_CTL_ELEM_TYPE_IEC958:
Packit 4a16fb
			count = sizeof(snd_aes_iec958_t);
Packit 4a16fb
			/* Fall through */
Packit 4a16fb
		case SND_CTL_ELEM_TYPE_BYTES:
Packit 4a16fb
			for (k = 0; k < count; ++k) {
Packit 4a16fb
				unsigned char old, val, mask;
Packit 4a16fb
				old = snd_ctl_elem_value_get_byte(elem->old, k);
Packit 4a16fb
				mask = snd_ctl_elem_value_get_byte(elem->mask, k);
Packit 4a16fb
				old &= ~mask;
Packit 4a16fb
				if (old) {
Packit 4a16fb
					val = snd_ctl_elem_value_get_byte(elem->val, k);
Packit 4a16fb
					val |= old;
Packit 4a16fb
					snd_ctl_elem_value_set_byte(elem->val, k, val);
Packit 4a16fb
				}
Packit 4a16fb
			}
Packit 4a16fb
			break;
Packit 4a16fb
		default:
Packit 4a16fb
			assert(0);
Packit 4a16fb
			break;
Packit 4a16fb
		}
Packit 4a16fb
		err = snd_ctl_elem_write(h->ctl, elem->val);
Packit 4a16fb
		if (err < 0) {
Packit 4a16fb
			SNDERR("Cannot write ctl elem");
Packit 4a16fb
			return err;
Packit 4a16fb
		}
Packit 4a16fb
	}
Packit 4a16fb
	return 0;
Packit 4a16fb
}
Packit 4a16fb
Packit 4a16fb
/**
Packit 4a16fb
 * \brief Remove (restore) previous values from control elements
Packit 4a16fb
 * \param h Setup control handle
Packit 4a16fb
 * \result zero if success, otherwise a negative error code
Packit 4a16fb
 */
Packit 4a16fb
int snd_sctl_remove(snd_sctl_t *h)
Packit 4a16fb
{
Packit 4a16fb
	struct list_head *pos;
Packit 4a16fb
	int err;
Packit 4a16fb
	assert(h);
Packit 4a16fb
	list_for_each(pos, &h->elems) {
Packit 4a16fb
		snd_sctl_elem_t *elem = list_entry(pos, snd_sctl_elem_t, list);
Packit 4a16fb
		if (elem->lock) {
Packit 4a16fb
			err = snd_ctl_elem_unlock(h->ctl, elem->id);
Packit 4a16fb
			if (err < 0) {
Packit 4a16fb
				SNDERR("Cannot unlock ctl elem");
Packit 4a16fb
				return err;
Packit 4a16fb
			}
Packit 4a16fb
		}
Packit 4a16fb
		/* Only restore the old value if it differs from the requested
Packit 4a16fb
		 * value, because if it has changed restoring the old value
Packit 4a16fb
		 * overrides the change.  Take for example, a voice modem with
Packit 4a16fb
		 * a .conf that sets preserve off-hook.  Start playback (on-hook
Packit 4a16fb
		 * to off-hook), start record (off-hook to off-hook), stop
Packit 4a16fb
		 * playback (off-hook to restore on-hook), stop record (on-hook
Packit 4a16fb
		 * to restore off-hook), Clearly you don't want to leave the
Packit 4a16fb
		 * modem "on the phone" now that there isn't any playback or
Packit 4a16fb
		 * recording active.
Packit 4a16fb
		 */
Packit 4a16fb
		if (elem->preserve && snd_ctl_elem_value_compare(elem->val, elem->old)) {
Packit 4a16fb
			err = snd_ctl_elem_write(h->ctl, elem->old);
Packit 4a16fb
			if (err < 0) {
Packit 4a16fb
				SNDERR("Cannot restore ctl elem");
Packit 4a16fb
				return err;
Packit 4a16fb
			}
Packit 4a16fb
		}
Packit 4a16fb
	}
Packit 4a16fb
	return 0;
Packit 4a16fb
}
Packit 4a16fb
Packit 4a16fb
static int snd_config_get_ctl_elem_enumerated(snd_config_t *n, snd_ctl_t *ctl,
Packit 4a16fb
					      snd_ctl_elem_info_t *info)
Packit 4a16fb
{
Packit 4a16fb
	const char *str;
Packit 4a16fb
	long val;
Packit 4a16fb
	unsigned int idx, items;
Packit 4a16fb
	switch (snd_config_get_type(n)) {
Packit 4a16fb
	case SND_CONFIG_TYPE_INTEGER:
Packit 4a16fb
		snd_config_get_integer(n, &val;;
Packit 4a16fb
		return val;
Packit 4a16fb
	case SND_CONFIG_TYPE_STRING:
Packit 4a16fb
		snd_config_get_string(n, &str);
Packit 4a16fb
		break;
Packit 4a16fb
	default:
Packit 4a16fb
		return -1;
Packit 4a16fb
	}
Packit 4a16fb
	items = snd_ctl_elem_info_get_items(info);
Packit 4a16fb
	for (idx = 0; idx < items; idx++) {
Packit 4a16fb
		int err;
Packit 4a16fb
		snd_ctl_elem_info_set_item(info, idx);
Packit 4a16fb
		err = snd_ctl_elem_info(ctl, info);
Packit 4a16fb
		if (err < 0) {
Packit 4a16fb
			SNDERR("Cannot obtain info for CTL elem");
Packit 4a16fb
			return err;
Packit 4a16fb
		}
Packit 4a16fb
		if (strcmp(str, snd_ctl_elem_info_get_item_name(info)) == 0)
Packit 4a16fb
			return idx;
Packit 4a16fb
	}
Packit 4a16fb
	return -1;
Packit 4a16fb
}
Packit 4a16fb
Packit 4a16fb
static int snd_config_get_ctl_elem_value(snd_config_t *conf,
Packit 4a16fb
					 snd_ctl_t *ctl,
Packit 4a16fb
					 snd_ctl_elem_value_t *val,
Packit 4a16fb
					 snd_ctl_elem_value_t *mask,
Packit 4a16fb
					 snd_ctl_elem_info_t *info)
Packit 4a16fb
{
Packit 4a16fb
	int err;
Packit 4a16fb
	snd_config_iterator_t i, next;
Packit 4a16fb
	snd_ctl_elem_id_t id = {0};
Packit 4a16fb
	snd_ctl_elem_type_t type;
Packit 4a16fb
	unsigned int count;
Packit 4a16fb
	long v;
Packit 4a16fb
	long idx;
Packit 4a16fb
	snd_ctl_elem_value_get_id(val, &id;;
Packit 4a16fb
	count = snd_ctl_elem_info_get_count(info);
Packit 4a16fb
	type = snd_ctl_elem_info_get_type(info);
Packit 4a16fb
	if (count == 1) {
Packit 4a16fb
		switch (type) {
Packit 4a16fb
		case SND_CTL_ELEM_TYPE_BOOLEAN:
Packit 4a16fb
			v = snd_config_get_bool(conf);
Packit 4a16fb
			if (v >= 0) {
Packit 4a16fb
				snd_ctl_elem_value_set_boolean(val, 0, v);
Packit 4a16fb
				if (mask)
Packit 4a16fb
					snd_ctl_elem_value_set_boolean(mask, 0, 1);
Packit 4a16fb
				return 0;
Packit 4a16fb
			}
Packit 4a16fb
			break;
Packit 4a16fb
		case SND_CTL_ELEM_TYPE_INTEGER:
Packit 4a16fb
			err = snd_config_get_integer(conf, &v);
Packit 4a16fb
			if (err == 0) {
Packit 4a16fb
				snd_ctl_elem_value_set_integer(val, 0, v);
Packit 4a16fb
				if (mask)
Packit 4a16fb
					snd_ctl_elem_value_set_integer(mask, 0, ~0L);
Packit 4a16fb
				return 0;
Packit 4a16fb
			}
Packit 4a16fb
			break;
Packit 4a16fb
		case SND_CTL_ELEM_TYPE_ENUMERATED:
Packit 4a16fb
			v = snd_config_get_ctl_elem_enumerated(conf, ctl, info);
Packit 4a16fb
			if (v >= 0) {
Packit 4a16fb
				snd_ctl_elem_value_set_enumerated(val, 0, v);
Packit 4a16fb
				if (mask)
Packit 4a16fb
					snd_ctl_elem_value_set_enumerated(mask, 0, ~0);
Packit 4a16fb
				return 0;
Packit 4a16fb
			}
Packit 4a16fb
			break;
Packit 4a16fb
		case SND_CTL_ELEM_TYPE_BYTES:
Packit 4a16fb
		case SND_CTL_ELEM_TYPE_IEC958:
Packit 4a16fb
			break;
Packit 4a16fb
		default:
Packit 4a16fb
			SNDERR("Unknown control type: %d", type);
Packit 4a16fb
			return -EINVAL;
Packit 4a16fb
		}
Packit 4a16fb
	}
Packit 4a16fb
	switch (type) {
Packit 4a16fb
	case SND_CTL_ELEM_TYPE_IEC958:
Packit 4a16fb
		count = sizeof(snd_aes_iec958_t);
Packit 4a16fb
		/* Fall through */
Packit 4a16fb
	case SND_CTL_ELEM_TYPE_BYTES:
Packit 4a16fb
	{
Packit 4a16fb
		const char *buf;
Packit 4a16fb
		err = snd_config_get_string(conf, &buf;;
Packit 4a16fb
		if (err >= 0) {
Packit 4a16fb
			int c1 = 0;
Packit 4a16fb
			unsigned int len = strlen(buf);
Packit 4a16fb
			unsigned int idx = 0;
Packit 4a16fb
			if (len % 2 != 0 || len > count * 2) {
Packit 4a16fb
			_bad_content:
Packit 4a16fb
				SNDERR("bad value content\n");
Packit 4a16fb
				return -EINVAL;
Packit 4a16fb
			}
Packit 4a16fb
			while (*buf) {
Packit 4a16fb
				int c = *buf++;
Packit 4a16fb
				if (c >= '0' && c <= '9')
Packit 4a16fb
					c -= '0';
Packit 4a16fb
				else if (c >= 'a' && c <= 'f')
Packit 4a16fb
					c = c - 'a' + 10;
Packit 4a16fb
				else if (c >= 'A' && c <= 'F')
Packit 4a16fb
					c = c - 'A' + 10;
Packit 4a16fb
				else {
Packit 4a16fb
					goto _bad_content;
Packit 4a16fb
				}
Packit 4a16fb
				if (idx % 2 == 1) {
Packit 4a16fb
					snd_ctl_elem_value_set_byte(val, idx / 2, c1 << 4 | c);
Packit 4a16fb
					if (mask)
Packit 4a16fb
						snd_ctl_elem_value_set_byte(mask, idx / 2, 0xff);
Packit 4a16fb
				} else
Packit 4a16fb
					c1 = c;
Packit 4a16fb
				idx++;
Packit 4a16fb
			}
Packit 4a16fb
			return 0;
Packit 4a16fb
		}
Packit 4a16fb
	}
Packit 4a16fb
	default:
Packit 4a16fb
		break;
Packit 4a16fb
	}
Packit 4a16fb
	if (snd_config_get_type(conf) != SND_CONFIG_TYPE_COMPOUND) {
Packit 4a16fb
		SNDERR("bad value type");
Packit 4a16fb
		return -EINVAL;
Packit 4a16fb
	}
Packit 4a16fb
Packit 4a16fb
	snd_config_for_each(i, next, conf) {
Packit 4a16fb
		snd_config_t *n = snd_config_iterator_entry(i);
Packit 4a16fb
		const char *id;
Packit 4a16fb
		if (snd_config_get_id(n, &id) < 0)
Packit 4a16fb
			continue;
Packit 4a16fb
		err = safe_strtol(id, &idx);
Packit 4a16fb
		if (err < 0 || idx < 0 || (unsigned int) idx >= count) {
Packit 4a16fb
			SNDERR("bad value index");
Packit 4a16fb
			return -EINVAL;
Packit 4a16fb
		}
Packit 4a16fb
		switch (type) {
Packit 4a16fb
		case SND_CTL_ELEM_TYPE_BOOLEAN:
Packit 4a16fb
			v = snd_config_get_bool(n);
Packit 4a16fb
			if (v < 0)
Packit 4a16fb
				goto _bad_content;
Packit 4a16fb
			snd_ctl_elem_value_set_boolean(val, idx, v);
Packit 4a16fb
			if (mask)
Packit 4a16fb
				snd_ctl_elem_value_set_boolean(mask, idx, 1);
Packit 4a16fb
			break;
Packit 4a16fb
		case SND_CTL_ELEM_TYPE_INTEGER:
Packit 4a16fb
			err = snd_config_get_integer(n, &v);
Packit 4a16fb
			if (err < 0)
Packit 4a16fb
				goto _bad_content;
Packit 4a16fb
			snd_ctl_elem_value_set_integer(val, idx, v);
Packit 4a16fb
			if (mask)
Packit 4a16fb
				snd_ctl_elem_value_set_integer(mask, idx, ~0L);
Packit 4a16fb
			break;
Packit 4a16fb
		case SND_CTL_ELEM_TYPE_ENUMERATED:
Packit 4a16fb
			v = snd_config_get_ctl_elem_enumerated(n, ctl, info);
Packit 4a16fb
			if (v < 0)
Packit 4a16fb
				goto _bad_content;
Packit 4a16fb
			snd_ctl_elem_value_set_enumerated(val, idx, v);
Packit 4a16fb
			if (mask)
Packit 4a16fb
				snd_ctl_elem_value_set_enumerated(mask, idx, ~0);
Packit 4a16fb
			break;
Packit 4a16fb
		case SND_CTL_ELEM_TYPE_BYTES:
Packit 4a16fb
		case SND_CTL_ELEM_TYPE_IEC958:
Packit 4a16fb
			err = snd_config_get_integer(n, &v);
Packit 4a16fb
			if (err < 0 || v < 0 || v > 255)
Packit 4a16fb
				goto _bad_content;
Packit 4a16fb
			snd_ctl_elem_value_set_byte(val, idx, v);
Packit 4a16fb
			if (mask)
Packit 4a16fb
				snd_ctl_elem_value_set_byte(mask, idx, 0xff);
Packit 4a16fb
			break;
Packit 4a16fb
		default:
Packit 4a16fb
			break;
Packit 4a16fb
		}
Packit 4a16fb
	}
Packit 4a16fb
	return 0;
Packit 4a16fb
}
Packit 4a16fb
Packit 4a16fb
static int add_elem(snd_sctl_t *h, snd_config_t *_conf, snd_config_t *private_data, int *quit)
Packit 4a16fb
{
Packit 4a16fb
	snd_config_t *conf;
Packit 4a16fb
	snd_config_iterator_t i, next;
Packit 4a16fb
	int iface = SND_CTL_ELEM_IFACE_MIXER;
Packit 4a16fb
	const char *name = NULL;
Packit 4a16fb
	long index = 0;
Packit 4a16fb
	long device = -1;
Packit 4a16fb
	long subdevice = -1;
Packit 4a16fb
	int lock = 0;
Packit 4a16fb
	int preserve = 0;
Packit 4a16fb
	int optional = 0;
Packit 4a16fb
	int skip_rest = 0;
Packit 4a16fb
	snd_config_t *value = NULL, *mask = NULL;
Packit 4a16fb
	snd_sctl_elem_t *elem = NULL;
Packit 4a16fb
	int err;
Packit 4a16fb
	err = snd_config_expand(_conf, _conf, NULL, private_data, &conf;;
Packit 4a16fb
	if (err < 0)
Packit 4a16fb
		return err;
Packit 4a16fb
	snd_config_for_each(i, next, conf) {
Packit 4a16fb
		snd_config_t *n = snd_config_iterator_entry(i);
Packit 4a16fb
		const char *id;
Packit 4a16fb
		if (snd_config_get_id(n, &id) < 0)
Packit 4a16fb
			continue;
Packit 4a16fb
		if (strcmp(id, "comment") == 0)
Packit 4a16fb
			continue;
Packit 4a16fb
		if (strcmp(id, "iface") == 0 || strcmp(id, "interface") == 0) {
Packit 4a16fb
			const char *ptr;
Packit 4a16fb
			if ((err = snd_config_get_string(n, &ptr)) < 0) {
Packit 4a16fb
				SNDERR("field %s is not a string", id);
Packit 4a16fb
				goto _err;
Packit 4a16fb
			}
Packit 4a16fb
			if ((err = snd_config_get_ctl_iface_ascii(ptr)) < 0) {
Packit 4a16fb
				SNDERR("Invalid value for '%s'", id);
Packit 4a16fb
				goto _err;
Packit 4a16fb
			}
Packit 4a16fb
			iface = err;
Packit 4a16fb
			continue;
Packit 4a16fb
		}
Packit 4a16fb
		if (strcmp(id, "name") == 0) {
Packit 4a16fb
			if ((err = snd_config_get_string(n, &name)) < 0) {
Packit 4a16fb
				SNDERR("field %s is not a string", id);
Packit 4a16fb
				goto _err;
Packit 4a16fb
			}
Packit 4a16fb
			continue;
Packit 4a16fb
		}
Packit 4a16fb
		if (strcmp(id, "index") == 0) {
Packit 4a16fb
			if ((err = snd_config_get_integer(n, &index)) < 0) {
Packit 4a16fb
				SNDERR("field %s is not an integer", id);
Packit 4a16fb
				goto _err;
Packit 4a16fb
			}
Packit 4a16fb
			continue;
Packit 4a16fb
		}
Packit 4a16fb
		if (strcmp(id, "device") == 0) {
Packit 4a16fb
			if ((err = snd_config_get_integer(n, &device)) < 0) {
Packit 4a16fb
				SNDERR("field %s is not an integer", id);
Packit 4a16fb
				goto _err;
Packit 4a16fb
			}
Packit 4a16fb
			continue;
Packit 4a16fb
		}
Packit 4a16fb
		if (strcmp(id, "subdevice") == 0) {
Packit 4a16fb
			if ((err = snd_config_get_integer(n, &subdevice)) < 0) {
Packit 4a16fb
				SNDERR("field %s is not an integer", id);
Packit 4a16fb
				goto _err;
Packit 4a16fb
			}
Packit 4a16fb
			continue;
Packit 4a16fb
		}
Packit 4a16fb
		if (strcmp(id, "lock") == 0) {
Packit 4a16fb
			err = snd_config_get_bool(n);
Packit 4a16fb
			if (err < 0)
Packit 4a16fb
				goto _err;
Packit 4a16fb
			lock = err;
Packit 4a16fb
			continue;
Packit 4a16fb
		}
Packit 4a16fb
		if (strcmp(id, "preserve") == 0) {
Packit 4a16fb
			err = snd_config_get_bool(n);
Packit 4a16fb
			if (err < 0)
Packit 4a16fb
				goto _err;
Packit 4a16fb
			preserve = err;
Packit 4a16fb
			continue;
Packit 4a16fb
		}
Packit 4a16fb
		if (strcmp(id, "value") == 0) {
Packit 4a16fb
			value = n;
Packit 4a16fb
			continue;
Packit 4a16fb
		}
Packit 4a16fb
		if (strcmp(id, "mask") == 0) {
Packit 4a16fb
			mask = n;
Packit 4a16fb
			continue;
Packit 4a16fb
		}
Packit 4a16fb
		if (strcmp(id, "optional") == 0) {
Packit 4a16fb
			err = snd_config_get_bool(n);
Packit 4a16fb
			if (err < 0)
Packit 4a16fb
				goto _err;
Packit 4a16fb
			optional = err;
Packit 4a16fb
			continue;
Packit 4a16fb
		}
Packit 4a16fb
		if (strcmp(id, "skip_rest") == 0) {
Packit 4a16fb
			err = snd_config_get_bool(n);
Packit 4a16fb
			if (err < 0)
Packit 4a16fb
				goto _err;
Packit 4a16fb
			skip_rest = err;
Packit 4a16fb
			continue;
Packit 4a16fb
		}
Packit 4a16fb
		SNDERR("Unknown field %s", id);
Packit 4a16fb
		return -EINVAL;
Packit 4a16fb
	}
Packit 4a16fb
	if (name == NULL) {
Packit 4a16fb
		SNDERR("Missing control name");
Packit 4a16fb
		err = -EINVAL;
Packit 4a16fb
		goto _err;
Packit 4a16fb
	}
Packit 4a16fb
	if (value == NULL) {
Packit 4a16fb
		SNDERR("Missing control value");
Packit 4a16fb
		err = -EINVAL;
Packit 4a16fb
		goto _err;
Packit 4a16fb
	}
Packit 4a16fb
	if (device < 0)
Packit 4a16fb
		device = 0;
Packit 4a16fb
	if (subdevice < 0)
Packit 4a16fb
		subdevice = 0;
Packit 4a16fb
	elem = calloc(1, sizeof(*elem));
Packit 4a16fb
	if (!elem)
Packit 4a16fb
		return -ENOMEM;
Packit 4a16fb
	err = snd_ctl_elem_id_malloc(&elem->id);
Packit 4a16fb
	if (err < 0)
Packit 4a16fb
		goto _err;
Packit 4a16fb
	err = snd_ctl_elem_info_malloc(&elem->info);
Packit 4a16fb
	if (err < 0)
Packit 4a16fb
		goto _err;
Packit 4a16fb
	err = snd_ctl_elem_value_malloc(&elem->val);
Packit 4a16fb
	if (err < 0)
Packit 4a16fb
		goto _err;
Packit 4a16fb
	err = snd_ctl_elem_value_malloc(&elem->mask);
Packit 4a16fb
	if (err < 0)
Packit 4a16fb
		goto _err;
Packit 4a16fb
	err = snd_ctl_elem_value_malloc(&elem->old);
Packit 4a16fb
	if (err < 0)
Packit 4a16fb
		goto _err;
Packit 4a16fb
	elem->lock = lock;
Packit 4a16fb
	elem->preserve = preserve;
Packit 4a16fb
	snd_ctl_elem_id_set_interface(elem->id, iface);
Packit 4a16fb
	snd_ctl_elem_id_set_name(elem->id, name);
Packit 4a16fb
	snd_ctl_elem_id_set_index(elem->id, index);
Packit 4a16fb
	snd_ctl_elem_id_set_device(elem->id, device);
Packit 4a16fb
	snd_ctl_elem_id_set_subdevice(elem->id, subdevice);
Packit 4a16fb
	snd_ctl_elem_info_set_id(elem->info, elem->id);
Packit 4a16fb
	err = snd_ctl_elem_info(h->ctl, elem->info);
Packit 4a16fb
	if (err < 0) {
Packit 4a16fb
		if (! optional)
Packit 4a16fb
			SNDERR("Cannot obtain info for CTL elem (%s,'%s',%li,%li,%li): %s", snd_ctl_elem_iface_name(iface), name, index, device, subdevice, snd_strerror(err));
Packit 4a16fb
		goto _err;
Packit 4a16fb
	} else {
Packit 4a16fb
		if (skip_rest)
Packit 4a16fb
			*quit = 1;
Packit 4a16fb
	}
Packit 4a16fb
	snd_ctl_elem_value_set_id(elem->val, elem->id);
Packit 4a16fb
	snd_ctl_elem_value_set_id(elem->old, elem->id);
Packit 4a16fb
	if (mask) {
Packit 4a16fb
		err = snd_config_get_ctl_elem_value(value, h->ctl, elem->val, NULL, elem->info);
Packit 4a16fb
		if (err < 0)
Packit 4a16fb
			goto _err;
Packit 4a16fb
		err = snd_config_get_ctl_elem_value(mask, h->ctl, elem->mask, NULL, elem->info);
Packit 4a16fb
		if (err < 0)
Packit 4a16fb
			goto _err;
Packit 4a16fb
	} else {
Packit 4a16fb
		err = snd_config_get_ctl_elem_value(value, h->ctl, elem->val, elem->mask, elem->info);
Packit 4a16fb
		if (err < 0)
Packit 4a16fb
			goto _err;
Packit 4a16fb
	}
Packit 4a16fb
		
Packit 4a16fb
	err = snd_config_get_ctl_elem_value(value, h->ctl, elem->val, elem->mask, elem->info);
Packit 4a16fb
	if (err < 0)
Packit 4a16fb
		goto _err;
Packit 4a16fb
	list_add_tail(&elem->list, &h->elems);
Packit 4a16fb
Packit 4a16fb
 _err:
Packit 4a16fb
 	if (err < 0 && elem) {
Packit 4a16fb
		if (elem->id)
Packit 4a16fb
			snd_ctl_elem_id_free(elem->id);
Packit 4a16fb
		if (elem->info)
Packit 4a16fb
			snd_ctl_elem_info_free(elem->info);
Packit 4a16fb
		if (elem->val)
Packit 4a16fb
			snd_ctl_elem_value_free(elem->val);
Packit 4a16fb
		if (elem->mask)
Packit 4a16fb
			snd_ctl_elem_value_free(elem->mask);
Packit 4a16fb
		if (elem->old)
Packit 4a16fb
			snd_ctl_elem_value_free(elem->old);
Packit 4a16fb
		free(elem);
Packit 4a16fb
		if (err != -ENOMEM && optional)
Packit 4a16fb
			err = 0; /* ignore the error */
Packit 4a16fb
	}
Packit 4a16fb
	if (conf)
Packit 4a16fb
		snd_config_delete(conf);
Packit 4a16fb
	return err;
Packit 4a16fb
}
Packit 4a16fb
Packit 4a16fb
/**
Packit 4a16fb
 * \brief Build setup control handle
Packit 4a16fb
 * \param sctl Result - setup control handle
Packit 4a16fb
 * \param handle Master control handle
Packit 4a16fb
 * \param conf Setup configuration
Packit 4a16fb
 * \param private_data Private data for runtime evaluation
Packit 4a16fb
 * \param mode Build mode - SND_SCTL_xxxx
Packit 4a16fb
 * \result zero if success, otherwise a negative error code
Packit 4a16fb
 */
Packit 4a16fb
int snd_sctl_build(snd_sctl_t **sctl, snd_ctl_t *handle, snd_config_t *conf, snd_config_t *private_data, int mode)
Packit 4a16fb
{
Packit 4a16fb
	snd_sctl_t *h;
Packit 4a16fb
	snd_config_iterator_t i, next;
Packit 4a16fb
	int err, quit = 0;
Packit 4a16fb
Packit 4a16fb
	assert(sctl);
Packit 4a16fb
	assert(handle);
Packit 4a16fb
	assert(conf);
Packit 4a16fb
	*sctl = NULL;
Packit 4a16fb
	if (snd_config_get_type(conf) != SND_CONFIG_TYPE_COMPOUND)
Packit 4a16fb
		return -EINVAL;
Packit 4a16fb
	h = calloc(1, sizeof(*h));
Packit 4a16fb
	if (!h) {
Packit 4a16fb
		if (mode & SND_SCTL_NOFREE)
Packit 4a16fb
			return -ENOMEM;
Packit 4a16fb
		snd_ctl_close(handle);
Packit 4a16fb
		return -ENOMEM;
Packit 4a16fb
	}
Packit 4a16fb
	h->mode = mode;
Packit 4a16fb
	h->ctl = handle;
Packit 4a16fb
	INIT_LIST_HEAD(&h->elems);
Packit 4a16fb
	snd_config_for_each(i, next, conf) {
Packit 4a16fb
		snd_config_t *n = snd_config_iterator_entry(i);
Packit 4a16fb
		err = add_elem(h, n, private_data, &quit);
Packit 4a16fb
		if (err < 0) {
Packit 4a16fb
			free_elems(h);
Packit 4a16fb
			return err;
Packit 4a16fb
		}
Packit 4a16fb
		if (quit)
Packit 4a16fb
			break;
Packit 4a16fb
	}
Packit 4a16fb
	*sctl = h;
Packit 4a16fb
	return 0;
Packit 4a16fb
}
Packit 4a16fb
Packit 4a16fb
/**
Packit 4a16fb
 * \brief Free setup control handle
Packit 4a16fb
 * \param sctl Setup control handle
Packit 4a16fb
 * \result zero if success, otherwise a negative error code
Packit 4a16fb
 */
Packit 4a16fb
int snd_sctl_free(snd_sctl_t *sctl)
Packit 4a16fb
{
Packit 4a16fb
	assert(sctl);
Packit 4a16fb
	return free_elems(sctl);
Packit 4a16fb
}