Blame src/pcm/pcm_generic.c

Packit Service db8eaa
/**
Packit Service db8eaa
 * \file pcm/pcm_generic.c
Packit Service db8eaa
 * \ingroup PCM
Packit Service db8eaa
 * \brief PCM Interface
Packit Service db8eaa
 * \author Jaroslav Kysela <perex@perex.cz>
Packit Service db8eaa
 * \date 2004
Packit Service db8eaa
 */
Packit Service db8eaa
/*
Packit Service db8eaa
 *  PCM - Common generic plugin code
Packit Service db8eaa
 *  Copyright (c) 2004 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 <sys/ioctl.h>
Packit Service db8eaa
#include <limits.h>
Packit Service db8eaa
#include "pcm_local.h"
Packit Service db8eaa
#include "pcm_generic.h"
Packit Service db8eaa
Packit Service db8eaa
#ifndef DOC_HIDDEN
Packit Service db8eaa
Packit Service db8eaa
int snd_pcm_generic_close(snd_pcm_t *pcm)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	int err = 0;
Packit Service db8eaa
	if (generic->close_slave)
Packit Service db8eaa
		err = snd_pcm_close(generic->slave);
Packit Service db8eaa
	free(generic);
Packit Service db8eaa
	return err;
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
int snd_pcm_generic_nonblock(snd_pcm_t *pcm, int nonblock)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return snd_pcm_nonblock(generic->slave, nonblock);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
int snd_pcm_generic_async(snd_pcm_t *pcm, int sig, pid_t pid)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return snd_pcm_async(generic->slave, sig, pid);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
int snd_pcm_generic_poll_descriptors_count(snd_pcm_t *pcm)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return snd_pcm_poll_descriptors_count(generic->slave);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
int snd_pcm_generic_poll_descriptors(snd_pcm_t *pcm, struct pollfd *pfds, unsigned int space)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return snd_pcm_poll_descriptors(generic->slave, pfds, space);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
int snd_pcm_generic_poll_revents(snd_pcm_t *pcm, struct pollfd *pfds, unsigned int nfds, unsigned short *revents)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return snd_pcm_poll_descriptors_revents(generic->slave, pfds, nfds, revents);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
int snd_pcm_generic_info(snd_pcm_t *pcm, snd_pcm_info_t * info)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return snd_pcm_info(generic->slave, info);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
int snd_pcm_generic_hw_free(snd_pcm_t *pcm)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return snd_pcm_hw_free(generic->slave);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
int snd_pcm_generic_sw_params(snd_pcm_t *pcm, snd_pcm_sw_params_t *params)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return snd_pcm_sw_params(generic->slave, params);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
int snd_pcm_generic_hw_refine(snd_pcm_t *pcm, snd_pcm_hw_params_t *params)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return snd_pcm_hw_refine(generic->slave, params);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
int snd_pcm_generic_hw_params(snd_pcm_t *pcm, snd_pcm_hw_params_t *params)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return _snd_pcm_hw_params_internal(generic->slave, params);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
int snd_pcm_generic_prepare(snd_pcm_t *pcm)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return snd_pcm_prepare(generic->slave);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
int snd_pcm_generic_channel_info(snd_pcm_t *pcm, snd_pcm_channel_info_t *info)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	if (pcm->mmap_shadow) {
Packit Service db8eaa
		/* No own buffer is required - the plugin won't change
Packit Service db8eaa
		 * the data on the buffer, or do safely on-the-place
Packit Service db8eaa
		 * conversion
Packit Service db8eaa
		 */
Packit Service db8eaa
		return snd_pcm_channel_info(generic->slave, info);
Packit Service db8eaa
	} else {
Packit Service db8eaa
		/* Allocate own buffer */
Packit Service db8eaa
		return snd_pcm_channel_info_shm(pcm, info, -1);
Packit Service db8eaa
	}
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
int snd_pcm_generic_status(snd_pcm_t *pcm, snd_pcm_status_t * status)
Packit Service db8eaa
{ 
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return snd_pcm_status(generic->slave, status);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
snd_pcm_state_t snd_pcm_generic_state(snd_pcm_t *pcm)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return snd_pcm_state(generic->slave);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
int snd_pcm_generic_hwsync(snd_pcm_t *pcm)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return snd_pcm_hwsync(generic->slave);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
int snd_pcm_generic_reset(snd_pcm_t *pcm)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return snd_pcm_reset(generic->slave);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
int snd_pcm_generic_start(snd_pcm_t *pcm)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return snd_pcm_start(generic->slave);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
int snd_pcm_generic_drop(snd_pcm_t *pcm)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return snd_pcm_drop(generic->slave);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
int snd_pcm_generic_drain(snd_pcm_t *pcm)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return snd_pcm_drain(generic->slave);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
int snd_pcm_generic_pause(snd_pcm_t *pcm, int enable)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return snd_pcm_pause(generic->slave, enable);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
int snd_pcm_generic_resume(snd_pcm_t *pcm)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return snd_pcm_resume(generic->slave);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
int snd_pcm_generic_delay(snd_pcm_t *pcm, snd_pcm_sframes_t *delayp)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return snd_pcm_delay(generic->slave, delayp);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
snd_pcm_sframes_t snd_pcm_generic_forwardable(snd_pcm_t *pcm)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return snd_pcm_forwardable(generic->slave);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
snd_pcm_sframes_t snd_pcm_generic_forward(snd_pcm_t *pcm, snd_pcm_uframes_t frames)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return INTERNAL(snd_pcm_forward)(generic->slave, frames);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
snd_pcm_sframes_t snd_pcm_generic_rewindable(snd_pcm_t *pcm)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return snd_pcm_rewindable(generic->slave);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
snd_pcm_sframes_t snd_pcm_generic_rewind(snd_pcm_t *pcm, snd_pcm_uframes_t frames)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return snd_pcm_rewind(generic->slave, frames);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
int snd_pcm_generic_link(snd_pcm_t *pcm1, snd_pcm_t *pcm2)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm1->private_data;
Packit Service db8eaa
	if (generic->slave->fast_ops->link)
Packit Service db8eaa
		return generic->slave->fast_ops->link(generic->slave->fast_op_arg, pcm2);
Packit Service db8eaa
	return -ENOSYS;
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
int snd_pcm_generic_link_slaves(snd_pcm_t *pcm, snd_pcm_t *master)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	if (generic->slave->fast_ops->link_slaves)
Packit Service db8eaa
		return generic->slave->fast_ops->link_slaves(generic->slave->fast_op_arg, master);
Packit Service db8eaa
	return -ENOSYS;
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
int snd_pcm_generic_unlink(snd_pcm_t *pcm)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	if (generic->slave->fast_ops->unlink)
Packit Service db8eaa
		return generic->slave->fast_ops->unlink(generic->slave->fast_op_arg);
Packit Service db8eaa
	return -ENOSYS;
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
snd_pcm_sframes_t snd_pcm_generic_writei(snd_pcm_t *pcm, const void *buffer, snd_pcm_uframes_t size)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return _snd_pcm_writei(generic->slave, buffer, size);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
snd_pcm_sframes_t snd_pcm_generic_writen(snd_pcm_t *pcm, void **bufs, snd_pcm_uframes_t size)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return _snd_pcm_writen(generic->slave, bufs, size);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
snd_pcm_sframes_t snd_pcm_generic_readi(snd_pcm_t *pcm, void *buffer, snd_pcm_uframes_t size)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return _snd_pcm_readi(generic->slave, buffer, size);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
snd_pcm_sframes_t snd_pcm_generic_readn(snd_pcm_t *pcm, void **bufs, snd_pcm_uframes_t size)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return _snd_pcm_readn(generic->slave, bufs, size);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
snd_pcm_sframes_t snd_pcm_generic_mmap_commit(snd_pcm_t *pcm, 
Packit Service db8eaa
					      snd_pcm_uframes_t offset,
Packit Service db8eaa
					      snd_pcm_uframes_t size)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return snd_pcm_mmap_commit(generic->slave, offset, size);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
snd_pcm_sframes_t snd_pcm_generic_avail_update(snd_pcm_t *pcm)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return snd_pcm_avail_update(generic->slave);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
int snd_pcm_generic_htimestamp(snd_pcm_t *pcm, snd_pcm_uframes_t *avail,
Packit Service db8eaa
			       snd_htimestamp_t *tstamp)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return snd_pcm_htimestamp(generic->slave, avail, tstamp);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
/* stand-alone version - similar like snd_pcm_hw_htimestamp but
Packit Service db8eaa
 * taking the tstamp via gettimestamp().
Packit Service db8eaa
 */
Packit Service db8eaa
int snd_pcm_generic_real_htimestamp(snd_pcm_t *pcm, snd_pcm_uframes_t *avail,
Packit Service db8eaa
				    snd_htimestamp_t *tstamp)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_sframes_t avail1;
Packit Service db8eaa
	int ok = 0;
Packit Service db8eaa
Packit Service db8eaa
	while (1) {
Packit Service db8eaa
		avail1 = __snd_pcm_avail_update(pcm);
Packit Service db8eaa
		if (avail1 < 0)
Packit Service db8eaa
			return avail1;
Packit Service db8eaa
		if (ok && (snd_pcm_uframes_t)avail1 == *avail)
Packit Service db8eaa
			break;
Packit Service db8eaa
		*avail = avail1;
Packit Service db8eaa
		gettimestamp(tstamp, pcm->tstamp_type);
Packit Service db8eaa
		ok = 1;
Packit Service db8eaa
	}
Packit Service db8eaa
	return 0;
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
int snd_pcm_generic_mmap(snd_pcm_t *pcm)
Packit Service db8eaa
{
Packit Service db8eaa
	if (pcm->mmap_shadow) {
Packit Service db8eaa
		/* Copy the slave mmapped buffer data */
Packit Service db8eaa
		snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
		pcm->mmap_channels = generic->slave->mmap_channels;
Packit Service db8eaa
		pcm->running_areas = generic->slave->running_areas;
Packit Service db8eaa
		pcm->stopped_areas = generic->slave->stopped_areas;
Packit Service db8eaa
	}
Packit Service db8eaa
	return 0;
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
int snd_pcm_generic_munmap(snd_pcm_t *pcm)
Packit Service db8eaa
{
Packit Service db8eaa
	if (pcm->mmap_shadow) {
Packit Service db8eaa
		/* Clean up */
Packit Service db8eaa
		pcm->mmap_channels = NULL;
Packit Service db8eaa
		pcm->running_areas = NULL;
Packit Service db8eaa
		pcm->stopped_areas = NULL;
Packit Service db8eaa
	}
Packit Service db8eaa
	return 0;
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
snd_pcm_chmap_query_t **snd_pcm_generic_query_chmaps(snd_pcm_t *pcm)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return snd_pcm_query_chmaps(generic->slave);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
snd_pcm_chmap_t *snd_pcm_generic_get_chmap(snd_pcm_t *pcm)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return snd_pcm_get_chmap(generic->slave);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
int snd_pcm_generic_set_chmap(snd_pcm_t *pcm, const snd_pcm_chmap_t *map)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return snd_pcm_set_chmap(generic->slave, map);
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
int snd_pcm_generic_may_wait_for_avail_min(snd_pcm_t *pcm, snd_pcm_uframes_t avail ATTRIBUTE_UNUSED)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_pcm_generic_t *generic = pcm->private_data;
Packit Service db8eaa
	return snd_pcm_may_wait_for_avail_min(generic->slave, snd_pcm_mmap_avail(generic->slave));
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
#endif /* DOC_HIDDEN */