/* Copyright 2016 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. */ #include <stdlib.h> #include <string.h> #include <syslog.h> #include "cras_alsa_mixer_name.h" #include "utlist.h" struct mixer_name *mixer_name_add(struct mixer_name *names, const char *name, enum CRAS_STREAM_DIRECTION dir, mixer_name_type type) { struct mixer_name *m_name; if (!name) return names; m_name = (struct mixer_name *)calloc(1, sizeof(struct mixer_name)); if (!m_name) return names; m_name->name = strdup(name); if (!m_name->name) { free(m_name); return names; } m_name->dir = dir; m_name->type = type; DL_APPEND(names, m_name); return names; } struct mixer_name *mixer_name_add_array(struct mixer_name *names, const char * const *name_array, size_t name_array_size, enum CRAS_STREAM_DIRECTION dir, mixer_name_type type) { size_t i; for (i = 0; i < name_array_size; i++) names = mixer_name_add(names, name_array[i], dir, type); return names; } void mixer_name_free(struct mixer_name *names) { struct mixer_name *m_name; DL_FOREACH(names, m_name) { DL_DELETE(names, m_name); free((void*)m_name->name); free(m_name); } } struct mixer_name *mixer_name_find(struct mixer_name *names, const char *name, enum CRAS_STREAM_DIRECTION dir, mixer_name_type type) { if (!name && type == MIXER_NAME_UNDEFINED) return NULL; struct mixer_name *m_name; DL_FOREACH(names, m_name) { /* Match the direction. */ if (dir != m_name->dir) continue; /* Match the type unless the type is UNDEFINED. */ if (type != MIXER_NAME_UNDEFINED && type != m_name->type) continue; /* Match the name if it is non-NULL, or return the first * item with the correct type when the name is not defined. */ if ((type != MIXER_NAME_UNDEFINED && !name) || (name && !strcmp(m_name->name, name))) return m_name; } return NULL; } static const char *mixer_name_type_str(enum CRAS_STREAM_DIRECTION dir, mixer_name_type type) { switch (dir) { case CRAS_STREAM_OUTPUT: switch (type) { case MIXER_NAME_VOLUME: return "output volume"; case MIXER_NAME_MAIN_VOLUME: return "main volume"; case MIXER_NAME_UNDEFINED: break; } break; case CRAS_STREAM_INPUT: switch (type) { case MIXER_NAME_VOLUME: return "input volume"; case MIXER_NAME_MAIN_VOLUME: return "main capture"; case MIXER_NAME_UNDEFINED: break; } break; case CRAS_STREAM_UNDEFINED: case CRAS_STREAM_POST_MIX_PRE_DSP: case CRAS_NUM_DIRECTIONS: break; } return "undefined"; } void mixer_name_dump(struct mixer_name *names, const char *message) { struct mixer_name *m_name; if (!names) { syslog(LOG_DEBUG, "%s: empty", message); return; } syslog(LOG_DEBUG, "%s:", message); DL_FOREACH(names, m_name) { const char *type_str = mixer_name_type_str(m_name->dir, m_name->type); syslog(LOG_DEBUG, " %s %s", m_name->name, type_str); } }