Blame mix/pcm_upmix.c

Packit Service cd2a00
/*
Packit Service cd2a00
 * Automatic upmix plugin
Packit Service cd2a00
 *
Packit Service cd2a00
 * Copyright (c) 2006 by Takashi Iwai <tiwai@suse.de>
Packit Service cd2a00
 *
Packit Service cd2a00
 * This library is free software; you can redistribute it and/or modify
Packit Service cd2a00
 * it under the terms of the GNU Lesser General Public License as
Packit Service cd2a00
 * published by the Free Software Foundation; either version 2.1 of
Packit Service cd2a00
 * the License, or (at your option) any later version.
Packit Service cd2a00
 *
Packit Service cd2a00
 * This program is distributed in the hope that it will be useful,
Packit Service cd2a00
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service cd2a00
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit Service cd2a00
 * GNU Lesser General Public License for more details.
Packit Service cd2a00
 *
Packit Service cd2a00
 * You should have received a copy of the GNU Lesser General Public
Packit Service cd2a00
 * License along with this library; if not, write to the Free Software
Packit Service cd2a00
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
Packit Service cd2a00
 */
Packit Service cd2a00
Packit Service cd2a00
#include <alsa/asoundlib.h>
Packit Service cd2a00
#include <alsa/pcm_external.h>
Packit Service cd2a00
Packit Service cd2a00
typedef struct snd_pcm_upmix snd_pcm_upmix_t;
Packit Service cd2a00
Packit Service cd2a00
typedef void (*upmixer_t)(snd_pcm_upmix_t *mix,
Packit Service cd2a00
			  const snd_pcm_channel_area_t *dst_areas,
Packit Service cd2a00
			  snd_pcm_uframes_t dst_offset,
Packit Service cd2a00
			  const snd_pcm_channel_area_t *src_areas,
Packit Service cd2a00
			  snd_pcm_uframes_t src_offset,
Packit Service cd2a00
			  snd_pcm_uframes_t size);
Packit Service cd2a00
Packit Service cd2a00
struct snd_pcm_upmix {
Packit Service cd2a00
	snd_pcm_extplug_t ext;
Packit Service cd2a00
	/* setup */
Packit Service cd2a00
	int delay_ms;
Packit Service cd2a00
	/* privates */
Packit Service cd2a00
	upmixer_t upmix;
Packit Service cd2a00
	unsigned int curpos;
Packit Service cd2a00
	int delay;
Packit Service cd2a00
	short *delayline[2];
Packit Service cd2a00
};
Packit Service cd2a00
Packit Service cd2a00
static inline void *area_addr(const snd_pcm_channel_area_t *area,
Packit Service cd2a00
			      snd_pcm_uframes_t offset)
Packit Service cd2a00
{
Packit Service cd2a00
	unsigned int bitofs = area->first + area->step * offset;
Packit Service cd2a00
	return (char *) area->addr + bitofs / 8;
Packit Service cd2a00
}
Packit Service cd2a00
Packit Service cd2a00
static inline unsigned int area_step(const snd_pcm_channel_area_t *area)
Packit Service cd2a00
{
Packit Service cd2a00
	return area->step / 8;
Packit Service cd2a00
}
Packit Service cd2a00
Packit Service cd2a00
/* Delayed copy SL & SR */
Packit Service cd2a00
static void delayed_copy(snd_pcm_upmix_t *mix,
Packit Service cd2a00
			 const snd_pcm_channel_area_t *dst_areas,
Packit Service cd2a00
			 snd_pcm_uframes_t dst_offset,
Packit Service cd2a00
			 const snd_pcm_channel_area_t *src_areas,
Packit Service cd2a00
			 snd_pcm_uframes_t src_offset,
Packit Service cd2a00
			 unsigned int size)
Packit Service cd2a00
{
Packit Service cd2a00
	unsigned int i, p, delay, curpos, dst_step, src_step;
Packit Service cd2a00
	short *dst, *src;
Packit Service cd2a00
Packit Service cd2a00
	if (! mix->delay_ms) {
Packit Service cd2a00
		snd_pcm_areas_copy(dst_areas, dst_offset, src_areas, src_offset,
Packit Service cd2a00
				   2, size, SND_PCM_FORMAT_S16);
Packit Service cd2a00
		return;
Packit Service cd2a00
	}
Packit Service cd2a00
Packit Service cd2a00
	delay = mix->delay;
Packit Service cd2a00
	if (delay > size)
Packit Service cd2a00
		delay = size;
Packit Service cd2a00
	for (i = 0; i < 2; i++) {
Packit Service cd2a00
		dst = (short *)area_addr(dst_areas + i, dst_offset);
Packit Service cd2a00
		dst_step = area_step(dst_areas + i) / 2;
Packit Service cd2a00
		curpos = mix->curpos;
Packit Service cd2a00
		for (p = 0; p < delay; p++) {
Packit Service cd2a00
			*dst = mix->delayline[i][curpos];
Packit Service cd2a00
			dst += dst_step;
Packit Service cd2a00
			curpos = (curpos + 1) % mix->delay;
Packit Service cd2a00
		}
Packit Service cd2a00
		snd_pcm_area_copy(dst_areas + i, dst_offset + delay,
Packit Service cd2a00
				  src_areas + i, src_offset,
Packit Service cd2a00
				  size - delay, SND_PCM_FORMAT_S16);
Packit Service cd2a00
		src = (short *)area_addr(src_areas + i,
Packit Service cd2a00
					 src_offset + size - delay);
Packit Service cd2a00
		src_step = area_step(src_areas + i) / 2;
Packit Service cd2a00
		curpos = mix->curpos;
Packit Service cd2a00
		for (p = 0; p < delay; p++) {
Packit Service cd2a00
			mix->delayline[i][curpos] = *src;
Packit Service cd2a00
			src += src_step;
Packit Service cd2a00
			curpos = (curpos + 1) % mix->delay;
Packit Service cd2a00
		}
Packit Service cd2a00
	}
Packit Service cd2a00
	mix->curpos = curpos;
Packit Service cd2a00
}
Packit Service cd2a00
Packit Service cd2a00
/* Average of L+R -> C and LFE */
Packit Service cd2a00
static void average_copy(const snd_pcm_channel_area_t *dst_areas,
Packit Service cd2a00
			 snd_pcm_uframes_t dst_offset,
Packit Service cd2a00
			 const snd_pcm_channel_area_t *src_areas,
Packit Service cd2a00
			 snd_pcm_uframes_t src_offset,
Packit Service cd2a00
			 unsigned int nchns,
Packit Service cd2a00
			 unsigned int size)
Packit Service cd2a00
{
Packit Service cd2a00
	short *dst[2], *src[2];
Packit Service cd2a00
	unsigned int i, dst_step[2], src_step[2];
Packit Service cd2a00
Packit Service cd2a00
	for (i = 0; i < nchns; i++) {
Packit Service cd2a00
		dst[i] = (short *)area_addr(dst_areas + i, dst_offset);
Packit Service cd2a00
		dst_step[i] = area_step(dst_areas + i) / 2;
Packit Service cd2a00
	}
Packit Service cd2a00
	for (i = 0; i < 2; i++) {
Packit Service cd2a00
		src[i] = (short *)area_addr(src_areas + i, src_offset);
Packit Service cd2a00
		src_step[i] = area_step(src_areas + i) / 2;
Packit Service cd2a00
	}
Packit Service cd2a00
	while (size--) {
Packit Service cd2a00
		short val = (*src[0] >> 1) + (*src[1] >> 1);
Packit Service cd2a00
		for (i = 0; i < nchns; i++) {
Packit Service cd2a00
			*dst[i] = val;
Packit Service cd2a00
			dst[i] += dst_step[i];
Packit Service cd2a00
		}
Packit Service cd2a00
		src[0] += src_step[0];
Packit Service cd2a00
		src[1] += src_step[1];
Packit Service cd2a00
	}
Packit Service cd2a00
}
Packit Service cd2a00
Packit Service cd2a00
static void upmix_1_to_71(snd_pcm_upmix_t *mix ATTRIBUTE_UNUSED,
Packit Service cd2a00
			  const snd_pcm_channel_area_t *dst_areas,
Packit Service cd2a00
			  snd_pcm_uframes_t dst_offset,
Packit Service cd2a00
			  const snd_pcm_channel_area_t *src_areas,
Packit Service cd2a00
			  snd_pcm_uframes_t src_offset,
Packit Service cd2a00
			  snd_pcm_uframes_t size)
Packit Service cd2a00
{
Packit Service cd2a00
	int i;
Packit Service cd2a00
	for (i = 0; i < 8; i++)
Packit Service cd2a00
		snd_pcm_area_copy(dst_areas + i, dst_offset,
Packit Service cd2a00
				  src_areas, src_offset,
Packit Service cd2a00
				  size, SND_PCM_FORMAT_S16);
Packit Service cd2a00
}
Packit Service cd2a00
Packit Service cd2a00
static void upmix_1_to_51(snd_pcm_upmix_t *mix ATTRIBUTE_UNUSED,
Packit Service cd2a00
			  const snd_pcm_channel_area_t *dst_areas,
Packit Service cd2a00
			  snd_pcm_uframes_t dst_offset,
Packit Service cd2a00
			  const snd_pcm_channel_area_t *src_areas,
Packit Service cd2a00
			  snd_pcm_uframes_t src_offset,
Packit Service cd2a00
			  snd_pcm_uframes_t size)
Packit Service cd2a00
{
Packit Service cd2a00
	int i;
Packit Service cd2a00
	for (i = 0; i < 6; i++)
Packit Service cd2a00
		snd_pcm_area_copy(dst_areas + i, dst_offset,
Packit Service cd2a00
				  src_areas, src_offset,
Packit Service cd2a00
				  size, SND_PCM_FORMAT_S16);
Packit Service cd2a00
}
Packit Service cd2a00
Packit Service cd2a00
static void upmix_1_to_40(snd_pcm_upmix_t *mix ATTRIBUTE_UNUSED,
Packit Service cd2a00
			  const snd_pcm_channel_area_t *dst_areas,
Packit Service cd2a00
			  snd_pcm_uframes_t dst_offset,
Packit Service cd2a00
			  const snd_pcm_channel_area_t *src_areas,
Packit Service cd2a00
			  snd_pcm_uframes_t src_offset,
Packit Service cd2a00
			  snd_pcm_uframes_t size)
Packit Service cd2a00
{
Packit Service cd2a00
	int i;
Packit Service cd2a00
	for (i = 0; i < 4; i++)
Packit Service cd2a00
		snd_pcm_area_copy(dst_areas + i, dst_offset,
Packit Service cd2a00
				  src_areas, src_offset,
Packit Service cd2a00
				  size, SND_PCM_FORMAT_S16);
Packit Service cd2a00
}
Packit Service cd2a00
Packit Service cd2a00
static void upmix_2_to_71(snd_pcm_upmix_t *mix,
Packit Service cd2a00
			  const snd_pcm_channel_area_t *dst_areas,
Packit Service cd2a00
			  snd_pcm_uframes_t dst_offset,
Packit Service cd2a00
			  const snd_pcm_channel_area_t *src_areas,
Packit Service cd2a00
			  snd_pcm_uframes_t src_offset,
Packit Service cd2a00
			  snd_pcm_uframes_t size)
Packit Service cd2a00
{
Packit Service cd2a00
	snd_pcm_areas_copy(dst_areas, dst_offset, src_areas, src_offset,
Packit Service cd2a00
			   2, size, SND_PCM_FORMAT_S16);
Packit Service cd2a00
	delayed_copy(mix, dst_areas + 2, dst_offset, src_areas, src_offset,
Packit Service cd2a00
		     size);
Packit Service cd2a00
	average_copy(dst_areas + 4, dst_offset, src_areas, src_offset,
Packit Service cd2a00
		     2, size);
Packit Service cd2a00
	snd_pcm_areas_copy(dst_areas + 6, dst_offset, src_areas, src_offset,
Packit Service cd2a00
			   2, size, SND_PCM_FORMAT_S16);
Packit Service cd2a00
	
Packit Service cd2a00
}
Packit Service cd2a00
Packit Service cd2a00
static void upmix_2_to_51(snd_pcm_upmix_t *mix,
Packit Service cd2a00
			  const snd_pcm_channel_area_t *dst_areas,
Packit Service cd2a00
			  snd_pcm_uframes_t dst_offset,
Packit Service cd2a00
			  const snd_pcm_channel_area_t *src_areas,
Packit Service cd2a00
			  snd_pcm_uframes_t src_offset,
Packit Service cd2a00
			  snd_pcm_uframes_t size)
Packit Service cd2a00
{
Packit Service cd2a00
	snd_pcm_areas_copy(dst_areas, dst_offset, src_areas, src_offset,
Packit Service cd2a00
			   2, size, SND_PCM_FORMAT_S16);
Packit Service cd2a00
	delayed_copy(mix, dst_areas + 2, dst_offset, src_areas, src_offset,
Packit Service cd2a00
		     size);
Packit Service cd2a00
	average_copy(dst_areas + 4, dst_offset, src_areas, src_offset,
Packit Service cd2a00
		     2, size);
Packit Service cd2a00
}
Packit Service cd2a00
Packit Service cd2a00
static void upmix_2_to_40(snd_pcm_upmix_t *mix,
Packit Service cd2a00
			  const snd_pcm_channel_area_t *dst_areas,
Packit Service cd2a00
			  snd_pcm_uframes_t dst_offset,
Packit Service cd2a00
			  const snd_pcm_channel_area_t *src_areas,
Packit Service cd2a00
			  snd_pcm_uframes_t src_offset,
Packit Service cd2a00
			  snd_pcm_uframes_t size)
Packit Service cd2a00
{
Packit Service cd2a00
	snd_pcm_areas_copy(dst_areas, dst_offset, src_areas, src_offset,
Packit Service cd2a00
			   2, size, SND_PCM_FORMAT_S16);
Packit Service cd2a00
	delayed_copy(mix, dst_areas + 2, dst_offset, src_areas, src_offset,
Packit Service cd2a00
		     size);
Packit Service cd2a00
}
Packit Service cd2a00
Packit Service cd2a00
static void upmix_3_to_51(snd_pcm_upmix_t *mix,
Packit Service cd2a00
			  const snd_pcm_channel_area_t *dst_areas,
Packit Service cd2a00
			  snd_pcm_uframes_t dst_offset,
Packit Service cd2a00
			  const snd_pcm_channel_area_t *src_areas,
Packit Service cd2a00
			  snd_pcm_uframes_t src_offset,
Packit Service cd2a00
			  snd_pcm_uframes_t size)
Packit Service cd2a00
{
Packit Service cd2a00
	snd_pcm_areas_copy(dst_areas, dst_offset, src_areas, src_offset,
Packit Service cd2a00
			   2, size, SND_PCM_FORMAT_S16);
Packit Service cd2a00
	delayed_copy(mix, dst_areas + 2, dst_offset, src_areas, src_offset,
Packit Service cd2a00
		     size);
Packit Service cd2a00
	snd_pcm_areas_copy(dst_areas + 4, dst_offset, src_areas, src_offset,
Packit Service cd2a00
			   2, size, SND_PCM_FORMAT_S16);
Packit Service cd2a00
}
Packit Service cd2a00
Packit Service cd2a00
static void upmix_3_to_40(snd_pcm_upmix_t *mix,
Packit Service cd2a00
			  const snd_pcm_channel_area_t *dst_areas,
Packit Service cd2a00
			  snd_pcm_uframes_t dst_offset,
Packit Service cd2a00
			  const snd_pcm_channel_area_t *src_areas,
Packit Service cd2a00
			  snd_pcm_uframes_t src_offset,
Packit Service cd2a00
			  snd_pcm_uframes_t size)
Packit Service cd2a00
{
Packit Service cd2a00
	snd_pcm_areas_copy(dst_areas, dst_offset, src_areas, src_offset,
Packit Service cd2a00
			   2, size, SND_PCM_FORMAT_S16);
Packit Service cd2a00
	delayed_copy(mix, dst_areas + 2, dst_offset, src_areas, src_offset,
Packit Service cd2a00
		     size);
Packit Service cd2a00
}
Packit Service cd2a00
Packit Service cd2a00
static void upmix_4_to_51(snd_pcm_upmix_t *mix ATTRIBUTE_UNUSED,
Packit Service cd2a00
			  const snd_pcm_channel_area_t *dst_areas,
Packit Service cd2a00
			  snd_pcm_uframes_t dst_offset,
Packit Service cd2a00
			  const snd_pcm_channel_area_t *src_areas,
Packit Service cd2a00
			  snd_pcm_uframes_t src_offset,
Packit Service cd2a00
			  snd_pcm_uframes_t size)
Packit Service cd2a00
{
Packit Service cd2a00
	snd_pcm_areas_copy(dst_areas, dst_offset, src_areas, src_offset,
Packit Service cd2a00
			   4, size, SND_PCM_FORMAT_S16);
Packit Service cd2a00
	snd_pcm_areas_copy(dst_areas + 4, dst_offset, src_areas, src_offset,
Packit Service cd2a00
			   2, size, SND_PCM_FORMAT_S16);
Packit Service cd2a00
}
Packit Service cd2a00
Packit Service cd2a00
static void upmix_4_to_40(snd_pcm_upmix_t *mix ATTRIBUTE_UNUSED,
Packit Service cd2a00
			  const snd_pcm_channel_area_t *dst_areas,
Packit Service cd2a00
			  snd_pcm_uframes_t dst_offset,
Packit Service cd2a00
			  const snd_pcm_channel_area_t *src_areas,
Packit Service cd2a00
			  snd_pcm_uframes_t src_offset,
Packit Service cd2a00
			  snd_pcm_uframes_t size)
Packit Service cd2a00
{
Packit Service cd2a00
	snd_pcm_areas_copy(dst_areas, dst_offset, src_areas, src_offset,
Packit Service cd2a00
			   4, size, SND_PCM_FORMAT_S16);
Packit Service cd2a00
}
Packit Service cd2a00
Packit Service cd2a00
static void upmix_5_to_51(snd_pcm_upmix_t *mix ATTRIBUTE_UNUSED,
Packit Service cd2a00
			  const snd_pcm_channel_area_t *dst_areas,
Packit Service cd2a00
			  snd_pcm_uframes_t dst_offset,
Packit Service cd2a00
			  const snd_pcm_channel_area_t *src_areas,
Packit Service cd2a00
			  snd_pcm_uframes_t src_offset,
Packit Service cd2a00
			  snd_pcm_uframes_t size)
Packit Service cd2a00
{
Packit Service cd2a00
	snd_pcm_areas_copy(dst_areas, dst_offset, src_areas, src_offset,
Packit Service cd2a00
			   5, size, SND_PCM_FORMAT_S16);
Packit Service cd2a00
	snd_pcm_area_copy(dst_areas + 5, dst_offset, src_areas + 4, src_offset,
Packit Service cd2a00
			  size, SND_PCM_FORMAT_S16);
Packit Service cd2a00
}
Packit Service cd2a00
Packit Service cd2a00
static void upmix_6_to_51(snd_pcm_upmix_t *mix ATTRIBUTE_UNUSED,
Packit Service cd2a00
			  const snd_pcm_channel_area_t *dst_areas,
Packit Service cd2a00
			  snd_pcm_uframes_t dst_offset,
Packit Service cd2a00
			  const snd_pcm_channel_area_t *src_areas,
Packit Service cd2a00
			  snd_pcm_uframes_t src_offset,
Packit Service cd2a00
			  snd_pcm_uframes_t size)
Packit Service cd2a00
{
Packit Service cd2a00
	snd_pcm_areas_copy(dst_areas, dst_offset, src_areas, src_offset,
Packit Service cd2a00
			   6, size, SND_PCM_FORMAT_S16);
Packit Service cd2a00
}
Packit Service cd2a00
Packit Service cd2a00
static void upmix_8_to_71(snd_pcm_upmix_t *mix ATTRIBUTE_UNUSED,
Packit Service cd2a00
			  const snd_pcm_channel_area_t *dst_areas,
Packit Service cd2a00
			  snd_pcm_uframes_t dst_offset,
Packit Service cd2a00
			  const snd_pcm_channel_area_t *src_areas,
Packit Service cd2a00
			  snd_pcm_uframes_t src_offset,
Packit Service cd2a00
			  snd_pcm_uframes_t size)
Packit Service cd2a00
{
Packit Service cd2a00
	snd_pcm_areas_copy(dst_areas, dst_offset, src_areas, src_offset,
Packit Service cd2a00
			   8, size, SND_PCM_FORMAT_S16);
Packit Service cd2a00
}
Packit Service cd2a00
Packit Service cd2a00
static const upmixer_t do_upmix[8][3] = {
Packit Service cd2a00
	{ upmix_1_to_40, upmix_1_to_51, upmix_1_to_71 },
Packit Service cd2a00
	{ upmix_2_to_40, upmix_2_to_51, upmix_2_to_71 },
Packit Service cd2a00
	{ upmix_3_to_40, upmix_3_to_51, upmix_3_to_51 },
Packit Service cd2a00
	{ upmix_4_to_40, upmix_4_to_51, upmix_4_to_51 },
Packit Service cd2a00
	{ upmix_4_to_40, upmix_5_to_51, upmix_5_to_51 },
Packit Service cd2a00
	{ upmix_4_to_40, upmix_6_to_51, upmix_6_to_51 },
Packit Service cd2a00
	{ upmix_4_to_40, upmix_6_to_51, upmix_6_to_51 },
Packit Service cd2a00
	{ upmix_4_to_40, upmix_6_to_51, upmix_8_to_71 },
Packit Service cd2a00
};
Packit Service cd2a00
Packit Service cd2a00
static snd_pcm_sframes_t
Packit Service cd2a00
upmix_transfer(snd_pcm_extplug_t *ext,
Packit Service cd2a00
	       const snd_pcm_channel_area_t *dst_areas,
Packit Service cd2a00
	       snd_pcm_uframes_t dst_offset,
Packit Service cd2a00
	       const snd_pcm_channel_area_t *src_areas,
Packit Service cd2a00
	       snd_pcm_uframes_t src_offset,
Packit Service cd2a00
	       snd_pcm_uframes_t size)
Packit Service cd2a00
{
Packit Service cd2a00
	snd_pcm_upmix_t *mix = (snd_pcm_upmix_t *)ext;
Packit Service cd2a00
	mix->upmix(mix, dst_areas, dst_offset,
Packit Service cd2a00
		   src_areas, src_offset, size);
Packit Service cd2a00
	return size;
Packit Service cd2a00
}
Packit Service cd2a00
Packit Service cd2a00
static int upmix_init(snd_pcm_extplug_t *ext)
Packit Service cd2a00
{
Packit Service cd2a00
	snd_pcm_upmix_t *mix = (snd_pcm_upmix_t *)ext;
Packit Service cd2a00
	int ctype, stype;
Packit Service cd2a00
Packit Service cd2a00
	switch (ext->slave_channels) {
Packit Service cd2a00
		case	6:
Packit Service cd2a00
			stype = 1;
Packit Service cd2a00
			break;
Packit Service cd2a00
		case 8:
Packit Service cd2a00
			stype = 2;
Packit Service cd2a00
			break;
Packit Service cd2a00
		default:
Packit Service cd2a00
			stype = 0;
Packit Service cd2a00
	}
Packit Service cd2a00
	ctype = ext->channels - 1;
Packit Service cd2a00
	if (ctype < 0 || ctype > 7) {
Packit Service cd2a00
		SNDERR("Invalid channel numbers for upmix: %d", ctype + 1);
Packit Service cd2a00
		return -EINVAL;
Packit Service cd2a00
	}
Packit Service cd2a00
	mix->upmix = do_upmix[ctype][stype];
Packit Service cd2a00
Packit Service cd2a00
	if (mix->delay_ms) {
Packit Service cd2a00
		free(mix->delayline[0]);
Packit Service cd2a00
		free(mix->delayline[1]);
Packit Service cd2a00
		mix->delay = ext->rate * mix->delay_ms / 1000;
Packit Service cd2a00
		mix->delayline[0] = calloc(2, mix->delay);
Packit Service cd2a00
		mix->delayline[1] = calloc(2, mix->delay);
Packit Service cd2a00
		if (! mix->delayline[0] || ! mix->delayline[1])
Packit Service cd2a00
			return -ENOMEM;
Packit Service cd2a00
		mix->curpos = 0;
Packit Service cd2a00
	}
Packit Service cd2a00
	return 0;
Packit Service cd2a00
}
Packit Service cd2a00
Packit Service cd2a00
static int upmix_close(snd_pcm_extplug_t *ext)
Packit Service cd2a00
{
Packit Service cd2a00
	snd_pcm_upmix_t *mix = (snd_pcm_upmix_t *)ext;
Packit Service cd2a00
	free(mix->delayline[0]);
Packit Service cd2a00
	free(mix->delayline[1]);
Packit Service cd2a00
	return 0;
Packit Service cd2a00
}
Packit Service cd2a00
Packit Service cd2a00
#if SND_PCM_EXTPLUG_VERSION >= 0x10002
Packit Service cd2a00
static unsigned int chmap[8][8] = {
Packit Service cd2a00
	{ SND_CHMAP_MONO },
Packit Service cd2a00
	{ SND_CHMAP_FL, SND_CHMAP_FR },
Packit Service cd2a00
	{ SND_CHMAP_FL, SND_CHMAP_FR, SND_CHMAP_FC },
Packit Service cd2a00
	{ SND_CHMAP_FL, SND_CHMAP_FR, SND_CHMAP_RL, SND_CHMAP_RR },
Packit Service cd2a00
	{ SND_CHMAP_FL, SND_CHMAP_FR, SND_CHMAP_RL, SND_CHMAP_RR, SND_CHMAP_FC },
Packit Service cd2a00
	{ SND_CHMAP_FL, SND_CHMAP_FR, SND_CHMAP_RL, SND_CHMAP_RR, SND_CHMAP_FC, SND_CHMAP_LFE },
Packit Service cd2a00
	{ SND_CHMAP_FL, SND_CHMAP_FR, SND_CHMAP_RL, SND_CHMAP_RR, SND_CHMAP_FC, SND_CHMAP_LFE, SND_CHMAP_UNKNOWN },
Packit Service cd2a00
	{ SND_CHMAP_FL, SND_CHMAP_FR, SND_CHMAP_RL, SND_CHMAP_RR, SND_CHMAP_FC, SND_CHMAP_LFE, SND_CHMAP_SL, SND_CHMAP_SR },
Packit Service cd2a00
};
Packit Service cd2a00
Packit Service cd2a00
static snd_pcm_chmap_query_t **upmix_query_chmaps(snd_pcm_extplug_t *ext ATTRIBUTE_UNUSED)
Packit Service cd2a00
{
Packit Service cd2a00
	snd_pcm_chmap_query_t **maps;
Packit Service cd2a00
	int i;
Packit Service cd2a00
Packit Service cd2a00
	maps = calloc(9, sizeof(void *));
Packit Service cd2a00
	if (!maps)
Packit Service cd2a00
		return NULL;
Packit Service cd2a00
	for (i = 0; i < 8; i++) {
Packit Service cd2a00
		snd_pcm_chmap_query_t *p;
Packit Service cd2a00
		p = maps[i] = calloc(i + 1 + 2, sizeof(int));
Packit Service cd2a00
		if (!p) {
Packit Service cd2a00
			snd_pcm_free_chmaps(maps);
Packit Service cd2a00
			return NULL;
Packit Service cd2a00
		}
Packit Service cd2a00
		p->type = SND_CHMAP_TYPE_FIXED;
Packit Service cd2a00
		p->map.channels = i + 1;
Packit Service cd2a00
		memcpy(p->map.pos, &chmap[i][0], (i + 1) * sizeof(int));
Packit Service cd2a00
	}
Packit Service cd2a00
	return maps;
Packit Service cd2a00
}
Packit Service cd2a00
Packit Service cd2a00
static snd_pcm_chmap_t *upmix_get_chmap(snd_pcm_extplug_t *ext)
Packit Service cd2a00
{
Packit Service cd2a00
	snd_pcm_chmap_t *map;
Packit Service cd2a00
Packit Service cd2a00
	if (ext->channels < 1 || ext->channels > 8)
Packit Service cd2a00
		return NULL;
Packit Service cd2a00
	map = malloc((ext->channels + 1) * sizeof(int));
Packit Service cd2a00
	if (!map)
Packit Service cd2a00
		return NULL;
Packit Service cd2a00
	map->channels = ext->channels;
Packit Service cd2a00
	memcpy(map->pos, &chmap[ext->channels - 1][0], ext->channels * sizeof(int));
Packit Service cd2a00
	return map;
Packit Service cd2a00
}
Packit Service cd2a00
#endif /* SND_PCM_EXTPLUG_VERSION >= 0x10002 */
Packit Service cd2a00
Packit Service cd2a00
static const snd_pcm_extplug_callback_t upmix_callback = {
Packit Service cd2a00
	.transfer = upmix_transfer,
Packit Service cd2a00
	.init = upmix_init,
Packit Service cd2a00
	.close = upmix_close,
Packit Service cd2a00
#if SND_PCM_EXTPLUG_VERSION >= 0x10002
Packit Service cd2a00
	.query_chmaps = upmix_query_chmaps,
Packit Service cd2a00
	.get_chmap = upmix_get_chmap,
Packit Service cd2a00
#endif
Packit Service cd2a00
};
Packit Service cd2a00
Packit Service cd2a00
SND_PCM_PLUGIN_DEFINE_FUNC(upmix)
Packit Service cd2a00
{
Packit Service cd2a00
	snd_config_iterator_t i, next;
Packit Service cd2a00
	snd_pcm_upmix_t *mix;
Packit Service cd2a00
	snd_config_t *sconf = NULL;
Packit Service cd2a00
	static const unsigned int chlist[3] = {4, 6, 8};
Packit Service cd2a00
	unsigned int channels = 0;
Packit Service cd2a00
	int delay = 10;
Packit Service cd2a00
	int err;
Packit Service cd2a00
Packit Service cd2a00
	snd_config_for_each(i, next, conf) {
Packit Service cd2a00
		snd_config_t *n = snd_config_iterator_entry(i);
Packit Service cd2a00
		const char *id;
Packit Service cd2a00
		if (snd_config_get_id(n, &id) < 0)
Packit Service cd2a00
			continue;
Packit Service cd2a00
		if (strcmp(id, "comment") == 0 || strcmp(id, "type") == 0 || strcmp(id, "hint") == 0)
Packit Service cd2a00
			continue;
Packit Service cd2a00
		if (strcmp(id, "slave") == 0) {
Packit Service cd2a00
			sconf = n;
Packit Service cd2a00
			continue;
Packit Service cd2a00
		}
Packit Service cd2a00
		if (strcmp(id, "delay") == 0) {
Packit Service cd2a00
			long val;
Packit Service cd2a00
			err = snd_config_get_integer(n, &val;;
Packit Service cd2a00
			if (err < 0) {
Packit Service cd2a00
				SNDERR("Invalid value for %s", id);
Packit Service cd2a00
				return err;
Packit Service cd2a00
			}
Packit Service cd2a00
			delay = val;
Packit Service cd2a00
			continue;
Packit Service cd2a00
		}
Packit Service cd2a00
		if (strcmp(id, "channels") == 0) {
Packit Service cd2a00
			long val;
Packit Service cd2a00
			err = snd_config_get_integer(n, &val;;
Packit Service cd2a00
			if (err < 0) {
Packit Service cd2a00
				SNDERR("Invalid value for %s", id);
Packit Service cd2a00
				return err;
Packit Service cd2a00
			}
Packit Service cd2a00
			channels = val;
Packit Service cd2a00
			if (channels != 4 && channels != 6 && channels != 0 && channels != 8) {
Packit Service cd2a00
				SNDERR("channels must be 4, 6, 8 or 0");
Packit Service cd2a00
				return -EINVAL;
Packit Service cd2a00
			}
Packit Service cd2a00
			continue;
Packit Service cd2a00
		}
Packit Service cd2a00
		SNDERR("Unknown field %s", id);
Packit Service cd2a00
		return -EINVAL;
Packit Service cd2a00
	}
Packit Service cd2a00
Packit Service cd2a00
	if (! sconf) {
Packit Service cd2a00
		SNDERR("No slave configuration for filrmix pcm");
Packit Service cd2a00
		return -EINVAL;
Packit Service cd2a00
	}
Packit Service cd2a00
Packit Service cd2a00
	mix = calloc(1, sizeof(*mix));
Packit Service cd2a00
	if (mix == NULL)
Packit Service cd2a00
		return -ENOMEM;
Packit Service cd2a00
Packit Service cd2a00
	mix->ext.version = SND_PCM_EXTPLUG_VERSION;
Packit Service cd2a00
	mix->ext.name = "Upmix Plugin";
Packit Service cd2a00
	mix->ext.callback = &upmix_callback;
Packit Service cd2a00
	mix->ext.private_data = mix;
Packit Service cd2a00
	if (delay < 0)
Packit Service cd2a00
		delay = 0;
Packit Service cd2a00
	else if (delay > 1000)
Packit Service cd2a00
		delay = 1000;
Packit Service cd2a00
	mix->delay_ms = delay;
Packit Service cd2a00
Packit Service cd2a00
	err = snd_pcm_extplug_create(&mix->ext, name, root, sconf, stream, mode);
Packit Service cd2a00
	if (err < 0) {
Packit Service cd2a00
		free(mix);
Packit Service cd2a00
		return err;
Packit Service cd2a00
	}
Packit Service cd2a00
Packit Service cd2a00
	snd_pcm_extplug_set_param_minmax(&mix->ext,
Packit Service cd2a00
					 SND_PCM_EXTPLUG_HW_CHANNELS,
Packit Service cd2a00
					 1, 8);
Packit Service cd2a00
	if (channels)
Packit Service cd2a00
		snd_pcm_extplug_set_slave_param_minmax(&mix->ext,
Packit Service cd2a00
						       SND_PCM_EXTPLUG_HW_CHANNELS,
Packit Service cd2a00
						       channels, channels);
Packit Service cd2a00
	else
Packit Service cd2a00
		snd_pcm_extplug_set_slave_param_list(&mix->ext,
Packit Service cd2a00
						     SND_PCM_EXTPLUG_HW_CHANNELS,
Packit Service cd2a00
						     3, chlist);
Packit Service cd2a00
	snd_pcm_extplug_set_param(&mix->ext, SND_PCM_EXTPLUG_HW_FORMAT,
Packit Service cd2a00
				  SND_PCM_FORMAT_S16);
Packit Service cd2a00
	snd_pcm_extplug_set_slave_param(&mix->ext, SND_PCM_EXTPLUG_HW_FORMAT,
Packit Service cd2a00
					SND_PCM_FORMAT_S16);
Packit Service cd2a00
Packit Service cd2a00
	*pcmp = mix->ext.pcm;
Packit Service cd2a00
	return 0;
Packit Service cd2a00
}
Packit Service cd2a00
Packit Service cd2a00
SND_PCM_PLUGIN_SYMBOL(upmix);