blob: 9c32b1b94c6fa9f91d79ae1bc22412a3ed52c7a4 [file] [log] [blame]
/**
* \file pcm/pcm_asym.c
* \ingroup PCM_Plugins
* \brief PCM Asymmetrical Plugin Interface
* \author Takashi Iwai <tiwai@suse.de>
* \date 2003
*/
#include "pcm_local.h"
#ifndef PIC
/* entry for static linking */
const char *_snd_module_pcm_asym = "";
#endif
/*! \page pcm_plugins
\section pcm_plugins_asym Plugin: asym
This plugin is a combination of playback and capture PCM streams.
Slave PCMs can be defined asymmetrically for both directions.
\code
pcm.name {
type asym # Asym PCM
playback STR # Playback slave name
# or
playback { # Playback slave definition
pcm STR # Slave PCM name
# or
pcm { } # Slave PCM definition
}
capture STR # Capture slave name
# or
capture { # Capture slave definition
pcm STR # Slave PCM name
# or
pcm { } # Slave PCM definition
}
}
\endcode
For example, you can combine a dmix plugin and a dsnoop plugin as
as a single PCM for playback and capture directions, respectively.
\code
pcm.duplex {
type asym
playback.pcm "dmix"
capture.pcm "dsnoop"
}
\endcode
By defining only a single direction, the resultant PCM becomes
half-duplex.
\subsection pcm_plugins_asym_funcref Function reference
<UL>
<LI>_snd_pcm_asym_open()
</UL>
*/
/**
* \brief Creates a new asym stream PCM
* \param pcmp Returns created PCM handle
* \param name Name of PCM
* \param root Root configuration node
* \param conf Configuration node with copy PCM description
* \param stream Stream type
* \param mode Stream mode
* \retval zero on success otherwise a negative error code
* \warning Using of this function might be dangerous in the sense
* of compatibility reasons. The prototype might be freely
* changed in future.
*/
int _snd_pcm_asym_open(snd_pcm_t **pcmp, const char *name ATTRIBUTE_UNUSED,
snd_config_t *root, snd_config_t *conf,
snd_pcm_stream_t stream, int mode)
{
snd_config_iterator_t i, next;
int err;
snd_config_t *slave = NULL, *sconf;
snd_config_for_each(i, next, conf) {
snd_config_t *n = snd_config_iterator_entry(i);
const char *id;
if (snd_config_get_id(n, &id) < 0)
continue;
if (snd_pcm_conf_generic_id(id))
continue;
if (strcmp(id, "playback") == 0) {
if (stream == SND_PCM_STREAM_PLAYBACK)
slave = n;
continue;
}
if (strcmp(id, "capture") == 0) {
if (stream == SND_PCM_STREAM_CAPTURE)
slave = n;
continue;
}
SNDERR("Unknown field %s", id);
return -EINVAL;
}
if (! slave) {
SNDERR("%s slave is not defined",
stream == SND_PCM_STREAM_PLAYBACK ? "playback" : "capture");
return -EINVAL;
}
err = snd_pcm_slave_conf(root, slave, &sconf, 0);
if (err < 0)
return err;
err = snd_pcm_open_named_slave(pcmp, name, root, sconf, stream,
mode, conf);
snd_config_delete(sconf);
return err;
}
#ifndef DOC_HIDDEN
SND_DLSYM_BUILD_VERSION(_snd_pcm_asym_open, SND_PCM_DLSYM_VERSION);
#endif