mirror of
				https://github.com/alsa-project/alsa-lib.git
				synced 2025-11-03 09:01:52 -05:00 
			
		
		
		
	
		
			
				
	
	
		
			959 lines
		
	
	
	
		
			24 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			959 lines
		
	
	
	
		
			24 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 *  PCM - Linear conversion
 | 
						|
 *  Copyright (c) 2000 by Abramo Bagnara <abramo@alsa-project.org>
 | 
						|
 *
 | 
						|
 *
 | 
						|
 *   This library is free software; you can redistribute it and/or modify
 | 
						|
 *   it under the terms of the GNU Library General Public License as
 | 
						|
 *   published by the Free Software Foundation; either version 2 of
 | 
						|
 *   the License, or (at your option) any later version.
 | 
						|
 *
 | 
						|
 *   This program is distributed in the hope that it will be useful,
 | 
						|
 *   but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
						|
 *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
						|
 *   GNU Library General Public License for more details.
 | 
						|
 *
 | 
						|
 *   You should have received a copy of the GNU Library General Public
 | 
						|
 *   License along with this library; if not, write to the Free Software
 | 
						|
 *   Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
 | 
						|
 *
 | 
						|
 */
 | 
						|
  
 | 
						|
#include <byteswap.h>
 | 
						|
#include <math.h>
 | 
						|
#include "pcm_local.h"
 | 
						|
#include "pcm_plugin.h"
 | 
						|
 | 
						|
/* The best possible hack to support missing optimization in gcc 2.7.2.3 */
 | 
						|
#if ROUTE_PLUGIN_RESOLUTION & (ROUTE_PLUGIN_RESOLUTION - 1) != 0
 | 
						|
#define div(a) a /= ROUTE_PLUGIN_RESOLUTION
 | 
						|
#elif ROUTE_PLUGIN_RESOLUTION == 16
 | 
						|
#define div(a) a >>= 4
 | 
						|
#else
 | 
						|
#error "Add some code here"
 | 
						|
#endif
 | 
						|
 | 
						|
typedef struct {
 | 
						|
	int channel;
 | 
						|
	int as_int;
 | 
						|
#if ROUTE_PLUGIN_FLOAT
 | 
						|
	float as_float;
 | 
						|
#endif
 | 
						|
} ttable_src_t;
 | 
						|
 | 
						|
typedef struct ttable_dst ttable_dst_t;
 | 
						|
 | 
						|
typedef struct {
 | 
						|
	enum {UINT32=0, UINT64=1, FLOAT=2} sum_idx;
 | 
						|
	int get_idx;
 | 
						|
	int put_idx;
 | 
						|
	int conv_idx;
 | 
						|
	int src_size;
 | 
						|
	int dst_sfmt;
 | 
						|
	size_t ndsts;
 | 
						|
	ttable_dst_t *dsts;
 | 
						|
} route_params_t;
 | 
						|
 | 
						|
 | 
						|
typedef void (*route_f)(const snd_pcm_channel_area_t *src_areas,
 | 
						|
			size_t src_offset,
 | 
						|
			const snd_pcm_channel_area_t *dst_area,
 | 
						|
			size_t dst_offset,
 | 
						|
			size_t frames,
 | 
						|
			const ttable_dst_t *ttable,
 | 
						|
			const route_params_t *params);
 | 
						|
 | 
						|
struct ttable_dst {
 | 
						|
	int att;	/* Attenuated */
 | 
						|
	unsigned int nsrcs;
 | 
						|
	ttable_src_t* srcs;
 | 
						|
	route_f func;
 | 
						|
};
 | 
						|
 | 
						|
typedef union {
 | 
						|
	u_int32_t as_uint32;
 | 
						|
	u_int64_t as_uint64;
 | 
						|
#if ROUTE_PLUGIN_FLOAT
 | 
						|
	float as_float;
 | 
						|
#endif
 | 
						|
} sum_t;
 | 
						|
 | 
						|
typedef struct {
 | 
						|
	/* This field need to be the first */
 | 
						|
	snd_pcm_plugin_t plug;
 | 
						|
	int sformat;
 | 
						|
	int schannels;
 | 
						|
	route_params_t params;
 | 
						|
} snd_pcm_route_t;
 | 
						|
 | 
						|
 | 
						|
static void route1_zero(const snd_pcm_channel_area_t *src_areas ATTRIBUTE_UNUSED,
 | 
						|
			size_t src_offset ATTRIBUTE_UNUSED,
 | 
						|
			const snd_pcm_channel_area_t *dst_area,
 | 
						|
			size_t dst_offset,
 | 
						|
			size_t frames,
 | 
						|
			const ttable_dst_t* ttable ATTRIBUTE_UNUSED,
 | 
						|
			const route_params_t *params)
 | 
						|
{
 | 
						|
#if 0
 | 
						|
	if (dst_area->wanted)
 | 
						|
		snd_pcm_area_silence(dst_area, dst_offset, frames, params->dst_sfmt);
 | 
						|
	dsts_area->enabled = 0;
 | 
						|
#else
 | 
						|
	snd_pcm_area_silence(dst_area, dst_offset, frames, params->dst_sfmt);
 | 
						|
#endif
 | 
						|
}
 | 
						|
 | 
						|
static void route1_one(const snd_pcm_channel_area_t *src_areas,
 | 
						|
		       size_t src_offset,
 | 
						|
		       const snd_pcm_channel_area_t *dst_area,
 | 
						|
		       size_t dst_offset,
 | 
						|
		       size_t frames,
 | 
						|
		       const ttable_dst_t* ttable,
 | 
						|
		       const route_params_t *params)
 | 
						|
{
 | 
						|
#define CONV_LABELS
 | 
						|
#include "plugin_ops.h"
 | 
						|
#undef CONV_LABELS
 | 
						|
	void *conv;
 | 
						|
	const snd_pcm_channel_area_t *src_area = 0;
 | 
						|
	unsigned int srcidx;
 | 
						|
	char *src, *dst;
 | 
						|
	int src_step, dst_step;
 | 
						|
	for (srcidx = 0; srcidx < ttable->nsrcs; ++srcidx) {
 | 
						|
		src_area = &src_areas[ttable->srcs[srcidx].channel];
 | 
						|
		if (src_area->addr != NULL)
 | 
						|
			break;
 | 
						|
	}
 | 
						|
	if (srcidx == ttable->nsrcs) {
 | 
						|
		route1_zero(src_areas, src_offset, dst_area, dst_offset, frames, ttable, params);
 | 
						|
		return;
 | 
						|
	}
 | 
						|
	
 | 
						|
#if 0
 | 
						|
	dst_area->enabled = 1;
 | 
						|
#endif
 | 
						|
	conv = conv_labels[params->conv_idx];
 | 
						|
	src = snd_pcm_channel_area_addr(src_area, src_offset);
 | 
						|
	dst = snd_pcm_channel_area_addr(dst_area, dst_offset);
 | 
						|
	src_step = snd_pcm_channel_area_step(src_area);
 | 
						|
	dst_step = snd_pcm_channel_area_step(dst_area);
 | 
						|
	while (frames-- > 0) {
 | 
						|
		goto *conv;
 | 
						|
#define CONV_END after
 | 
						|
#include "plugin_ops.h"
 | 
						|
#undef CONV_END
 | 
						|
	after:
 | 
						|
		src += src_step;
 | 
						|
		dst += dst_step;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static void route1_many(const snd_pcm_channel_area_t *src_areas,
 | 
						|
			size_t src_offset,
 | 
						|
			const snd_pcm_channel_area_t *dst_area,
 | 
						|
			size_t dst_offset,
 | 
						|
			size_t frames,
 | 
						|
			const ttable_dst_t* ttable,
 | 
						|
			const route_params_t *params)
 | 
						|
{
 | 
						|
#define GET_LABELS
 | 
						|
#define PUT32_LABELS
 | 
						|
#include "plugin_ops.h"
 | 
						|
#undef GET_LABELS
 | 
						|
#undef PUT32_LABELS
 | 
						|
	static void *zero_labels[3] = { &&zero_int32, &&zero_int64,
 | 
						|
#if ROUTE_PLUGIN_FLOAT
 | 
						|
				 &&zero_float
 | 
						|
#endif
 | 
						|
	};
 | 
						|
	/* sum_type att */
 | 
						|
	static void *add_labels[3 * 2] = { &&add_int32_noatt, &&add_int32_att,
 | 
						|
				    &&add_int64_noatt, &&add_int64_att,
 | 
						|
#if ROUTE_PLUGIN_FLOAT
 | 
						|
				    &&add_float_noatt, &&add_float_att
 | 
						|
#endif
 | 
						|
	};
 | 
						|
	/* sum_type att shift */
 | 
						|
	static void *norm_labels[3 * 2 * 4] = { 0,
 | 
						|
					 &&norm_int32_8_noatt,
 | 
						|
					 &&norm_int32_16_noatt,
 | 
						|
					 &&norm_int32_24_noatt,
 | 
						|
					 0,
 | 
						|
					 &&norm_int32_8_att,
 | 
						|
					 &&norm_int32_16_att,
 | 
						|
					 &&norm_int32_24_att,
 | 
						|
					 &&norm_int64_0_noatt,
 | 
						|
					 &&norm_int64_8_noatt,
 | 
						|
					 &&norm_int64_16_noatt,
 | 
						|
					 &&norm_int64_24_noatt,
 | 
						|
					 &&norm_int64_0_att,
 | 
						|
					 &&norm_int64_8_att,
 | 
						|
					 &&norm_int64_16_att,
 | 
						|
					 &&norm_int64_24_att,
 | 
						|
#if ROUTE_PLUGIN_FLOAT
 | 
						|
					 &&norm_float_0,
 | 
						|
					 &&norm_float_8,
 | 
						|
					 &&norm_float_16,
 | 
						|
					 &&norm_float_24,
 | 
						|
					 &&norm_float_0,
 | 
						|
					 &&norm_float_8,
 | 
						|
					 &&norm_float_16,
 | 
						|
					 &&norm_float_24,
 | 
						|
#endif
 | 
						|
	};
 | 
						|
	void *zero, *get, *add, *norm, *put32;
 | 
						|
	int nsrcs = ttable->nsrcs;
 | 
						|
	char *dst;
 | 
						|
	int dst_step;
 | 
						|
	char *srcs[nsrcs];
 | 
						|
	int src_steps[nsrcs];
 | 
						|
	ttable_src_t src_tt[nsrcs];
 | 
						|
	u_int32_t sample = 0;
 | 
						|
	int srcidx, srcidx1 = 0;
 | 
						|
	for (srcidx = 0; srcidx < nsrcs; ++srcidx) {
 | 
						|
		const snd_pcm_channel_area_t *src_area = &src_areas[ttable->srcs[srcidx].channel];
 | 
						|
#if 0
 | 
						|
		if (!src_area->enabled)
 | 
						|
			continue;
 | 
						|
#endif
 | 
						|
		srcs[srcidx1] = snd_pcm_channel_area_addr(src_area, src_offset);
 | 
						|
		src_steps[srcidx1] = snd_pcm_channel_area_step(src_area);
 | 
						|
		src_tt[srcidx1] = ttable->srcs[srcidx];
 | 
						|
		srcidx1++;
 | 
						|
	}
 | 
						|
	nsrcs = srcidx1;
 | 
						|
	if (nsrcs == 0) {
 | 
						|
		route1_zero(src_areas, src_offset, dst_area, dst_offset, frames, ttable, params);
 | 
						|
		return;
 | 
						|
	} else if (nsrcs == 1 && src_tt[0].as_int == ROUTE_PLUGIN_RESOLUTION) {
 | 
						|
		route1_one(src_areas, src_offset, dst_area, dst_offset, frames, ttable, params);
 | 
						|
		return;
 | 
						|
	}
 | 
						|
 | 
						|
#if 0
 | 
						|
	dst_area->enabled = 1;
 | 
						|
#endif
 | 
						|
	zero = zero_labels[params->sum_idx];
 | 
						|
	get = get_labels[params->get_idx];
 | 
						|
	add = add_labels[params->sum_idx * 2 + ttable->att];
 | 
						|
	norm = norm_labels[params->sum_idx * 8 + ttable->att * 4 + 4 - params->src_size];
 | 
						|
	put32 = put32_labels[params->put_idx];
 | 
						|
	dst = snd_pcm_channel_area_addr(dst_area, dst_offset);
 | 
						|
	dst_step = snd_pcm_channel_area_step(dst_area);
 | 
						|
 | 
						|
	while (frames-- > 0) {
 | 
						|
		ttable_src_t *ttp = src_tt;
 | 
						|
		sum_t sum;
 | 
						|
 | 
						|
		/* Zero sum */
 | 
						|
		goto *zero;
 | 
						|
	zero_int32:
 | 
						|
		sum.as_uint32 = 0;
 | 
						|
		goto zero_end;
 | 
						|
	zero_int64: 
 | 
						|
		sum.as_uint64 = 0;
 | 
						|
		goto zero_end;
 | 
						|
#if ROUTE_PLUGIN_FLOAT
 | 
						|
	zero_float:
 | 
						|
		sum.as_float = 0.0;
 | 
						|
		goto zero_end;
 | 
						|
#endif
 | 
						|
	zero_end:
 | 
						|
		for (srcidx = 0; srcidx < nsrcs; ++srcidx) {
 | 
						|
			char *src = srcs[srcidx];
 | 
						|
			
 | 
						|
			/* Get sample */
 | 
						|
			goto *get;
 | 
						|
#define GET_END after_get
 | 
						|
#include "plugin_ops.h"
 | 
						|
#undef GET_END
 | 
						|
		after_get:
 | 
						|
 | 
						|
			/* Sum */
 | 
						|
			goto *add;
 | 
						|
		add_int32_att:
 | 
						|
			sum.as_uint32 += sample * ttp->as_int;
 | 
						|
			goto after_sum;
 | 
						|
		add_int32_noatt:
 | 
						|
			if (ttp->as_int)
 | 
						|
				sum.as_uint32 += sample;
 | 
						|
			goto after_sum;
 | 
						|
		add_int64_att:
 | 
						|
			sum.as_uint64 += (u_int64_t) sample * ttp->as_int;
 | 
						|
			goto after_sum;
 | 
						|
		add_int64_noatt:
 | 
						|
			if (ttp->as_int)
 | 
						|
				sum.as_uint64 += sample;
 | 
						|
			goto after_sum;
 | 
						|
#if ROUTE_PLUGIN_FLOAT
 | 
						|
		add_float_att:
 | 
						|
			sum.as_float += sample * ttp->as_float;
 | 
						|
			goto after_sum;
 | 
						|
		add_float_noatt:
 | 
						|
			if (ttp->as_int)
 | 
						|
				sum.as_float += sample;
 | 
						|
			goto after_sum;
 | 
						|
#endif
 | 
						|
		after_sum:
 | 
						|
			srcs[srcidx] += src_steps[srcidx];
 | 
						|
			ttp++;
 | 
						|
		}
 | 
						|
		
 | 
						|
		/* Normalization */
 | 
						|
		goto *norm;
 | 
						|
	norm_int32_8_att:
 | 
						|
		sum.as_uint64 = sum.as_uint32;
 | 
						|
	norm_int64_8_att:
 | 
						|
		sum.as_uint64 <<= 8;
 | 
						|
	norm_int64_0_att:
 | 
						|
		div(sum.as_uint64);
 | 
						|
		goto norm_int;
 | 
						|
 | 
						|
	norm_int32_16_att:
 | 
						|
		sum.as_uint64 = sum.as_uint32;
 | 
						|
	norm_int64_16_att:
 | 
						|
		sum.as_uint64 <<= 16;
 | 
						|
		div(sum.as_uint64);
 | 
						|
		goto norm_int;
 | 
						|
 | 
						|
	norm_int32_24_att:
 | 
						|
		sum.as_uint64 = sum.as_uint32;
 | 
						|
	norm_int64_24_att:
 | 
						|
		sum.as_uint64 <<= 24;
 | 
						|
		div(sum.as_uint64);
 | 
						|
		goto norm_int;
 | 
						|
 | 
						|
	norm_int32_8_noatt:
 | 
						|
		sum.as_uint64 = sum.as_uint32;
 | 
						|
	norm_int64_8_noatt:
 | 
						|
		sum.as_uint64 <<= 8;
 | 
						|
		goto norm_int;
 | 
						|
 | 
						|
	norm_int32_16_noatt:
 | 
						|
		sum.as_uint64 = sum.as_uint32;
 | 
						|
	norm_int64_16_noatt:
 | 
						|
		sum.as_uint64 <<= 16;
 | 
						|
		goto norm_int;
 | 
						|
 | 
						|
	norm_int32_24_noatt:
 | 
						|
		sum.as_uint64 = sum.as_uint32;
 | 
						|
	norm_int64_24_noatt:
 | 
						|
		sum.as_uint64 <<= 24;
 | 
						|
		goto norm_int;
 | 
						|
 | 
						|
	norm_int64_0_noatt:
 | 
						|
	norm_int:
 | 
						|
		if (sum.as_uint64 > (u_int32_t)0xffffffff)
 | 
						|
			sample = (u_int32_t)0xffffffff;
 | 
						|
		else
 | 
						|
			sample = sum.as_uint64;
 | 
						|
		goto after_norm;
 | 
						|
 | 
						|
#if ROUTE_PLUGIN_FLOAT
 | 
						|
	norm_float_8:
 | 
						|
		sum.as_float *= 1 << 8;
 | 
						|
		goto norm_float;
 | 
						|
	norm_float_16:
 | 
						|
		sum.as_float *= 1 << 16;
 | 
						|
		goto norm_float;
 | 
						|
	norm_float_24:
 | 
						|
		sum.as_float *= 1 << 24;
 | 
						|
		goto norm_float;
 | 
						|
	norm_float_0:
 | 
						|
	norm_float:
 | 
						|
		sum.as_float = floor(sum.as_float + 0.5);
 | 
						|
		if (sum.as_float > (u_int32_t)0xffffffff)
 | 
						|
			sample = (u_int32_t)0xffffffff;
 | 
						|
		else
 | 
						|
			sample = sum.as_float;
 | 
						|
		goto after_norm;
 | 
						|
#endif
 | 
						|
	after_norm:
 | 
						|
		
 | 
						|
		/* Put sample */
 | 
						|
		goto *put32;
 | 
						|
#define PUT32_END after_put32
 | 
						|
#include "plugin_ops.h"
 | 
						|
#undef PUT32_END
 | 
						|
	after_put32:
 | 
						|
		
 | 
						|
		dst += dst_step;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static void route_transfer(const snd_pcm_channel_area_t *src_areas,
 | 
						|
			   size_t src_offset,
 | 
						|
			   const snd_pcm_channel_area_t *dst_areas,
 | 
						|
			   size_t dst_offset,
 | 
						|
			   size_t dst_channels,
 | 
						|
			   size_t frames,
 | 
						|
			   route_params_t *params)
 | 
						|
{
 | 
						|
	size_t dst_channel;
 | 
						|
	ttable_dst_t *dstp;
 | 
						|
	const snd_pcm_channel_area_t *dst_area;
 | 
						|
 | 
						|
	dstp = params->dsts;
 | 
						|
	dst_area = dst_areas;
 | 
						|
	for (dst_channel = 0; dst_channel < dst_channels; ++dst_channel) {
 | 
						|
		if (dst_channel >= params->ndsts)
 | 
						|
			route1_zero(src_areas, src_offset, dst_area, dst_offset, frames, dstp, params);
 | 
						|
		else
 | 
						|
			dstp->func(src_areas, src_offset, dst_area, dst_offset, frames, dstp, params);
 | 
						|
		dstp++;
 | 
						|
		dst_area++;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static int snd_pcm_route_close(snd_pcm_t *pcm)
 | 
						|
{
 | 
						|
	snd_pcm_route_t *route = pcm->private;
 | 
						|
	route_params_t *params = &route->params;
 | 
						|
	int err = 0;
 | 
						|
	size_t dst_channel;
 | 
						|
	if (route->plug.close_slave)
 | 
						|
		err = snd_pcm_close(route->plug.slave);
 | 
						|
	if (params->dsts) {
 | 
						|
		for (dst_channel = 0; dst_channel < params->ndsts; ++dst_channel) {
 | 
						|
			if (params->dsts[dst_channel].srcs != NULL)
 | 
						|
				free(params->dsts[dst_channel].srcs);
 | 
						|
		}
 | 
						|
		free(params->dsts);
 | 
						|
	}
 | 
						|
	free(route);
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static int snd_pcm_route_hw_refine(snd_pcm_t *pcm, snd_pcm_hw_params_t *params)
 | 
						|
{
 | 
						|
	snd_pcm_route_t *route = pcm->private;
 | 
						|
	snd_pcm_t *slave = route->plug.slave;
 | 
						|
	int err;
 | 
						|
	snd_pcm_hw_params_t sparams;
 | 
						|
	unsigned int links = (SND_PCM_HW_PARBIT_CHANNELS |
 | 
						|
			      SND_PCM_HW_PARBIT_FRAGMENTS |
 | 
						|
			      SND_PCM_HW_PARBIT_FRAGMENT_SIZE |
 | 
						|
			      SND_PCM_HW_PARBIT_FRAGMENT_LENGTH |
 | 
						|
			      SND_PCM_HW_PARBIT_BUFFER_SIZE |
 | 
						|
			      SND_PCM_HW_PARBIT_BUFFER_LENGTH);
 | 
						|
	mask_t *access_mask = alloca(mask_sizeof());
 | 
						|
	mask_t *format_mask = alloca(mask_sizeof());
 | 
						|
	mask_t *saccess_mask = alloca(mask_sizeof());
 | 
						|
	mask_load(access_mask, SND_PCM_ACCBIT_PLUGIN);
 | 
						|
	mask_load(format_mask, SND_PCM_FMTBIT_LINEAR);
 | 
						|
	mask_load(saccess_mask, SND_PCM_ACCBIT_MMAP);
 | 
						|
	err = _snd_pcm_hw_param_mask(params, 1, SND_PCM_HW_PARAM_ACCESS,
 | 
						|
				      access_mask);
 | 
						|
	if (err < 0)
 | 
						|
		return err;
 | 
						|
	err = _snd_pcm_hw_param_mask(params, 1, SND_PCM_HW_PARAM_FORMAT,
 | 
						|
				      format_mask);
 | 
						|
	if (err < 0)
 | 
						|
		return err;
 | 
						|
	err = _snd_pcm_hw_param_min(params, 1, SND_PCM_HW_PARAM_CHANNELS, 1);
 | 
						|
	if (err < 0)
 | 
						|
		return err;
 | 
						|
 | 
						|
	_snd_pcm_hw_params_any(&sparams);
 | 
						|
	_snd_pcm_hw_param_mask(&sparams, 0, SND_PCM_HW_PARAM_ACCESS,
 | 
						|
				saccess_mask);
 | 
						|
	if (route->sformat >= 0) {
 | 
						|
		_snd_pcm_hw_param_set(&sparams, 0, SND_PCM_HW_PARAM_FORMAT,
 | 
						|
				       route->sformat);
 | 
						|
		_snd_pcm_hw_param_set(&sparams, 0, SND_PCM_HW_PARAM_SUBFORMAT,
 | 
						|
				       SND_PCM_SUBFORMAT_STD);
 | 
						|
	} else
 | 
						|
		links |= (SND_PCM_HW_PARBIT_FORMAT | 
 | 
						|
			  SND_PCM_HW_PARBIT_SUBFORMAT |
 | 
						|
			  SND_PCM_HW_PARBIT_SAMPLE_BITS);
 | 
						|
	if (route->schannels >= 0) {
 | 
						|
		_snd_pcm_hw_param_set(&sparams, 0, SND_PCM_HW_PARAM_CHANNELS,
 | 
						|
				       route->schannels);
 | 
						|
	} else {
 | 
						|
		links |= SND_PCM_HW_PARBIT_CHANNELS;
 | 
						|
		if (route->sformat < 0)
 | 
						|
			links |= (SND_PCM_HW_PARBIT_FRAME_BITS |
 | 
						|
				  SND_PCM_HW_PARBIT_FRAGMENT_BYTES |
 | 
						|
				  SND_PCM_HW_PARBIT_BUFFER_BYTES);
 | 
						|
	}
 | 
						|
		
 | 
						|
	err = snd_pcm_hw_refine2(params, &sparams,
 | 
						|
				 snd_pcm_hw_refine, slave, links);
 | 
						|
	if (err < 0)
 | 
						|
		return err;
 | 
						|
	params->info &= ~(SND_PCM_INFO_MMAP | SND_PCM_INFO_MMAP_VALID);
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static int snd_pcm_route_hw_params(snd_pcm_t *pcm, snd_pcm_hw_params_t * params)
 | 
						|
{
 | 
						|
	snd_pcm_route_t *route = pcm->private;
 | 
						|
	snd_pcm_t *slave = route->plug.slave;
 | 
						|
	int err;
 | 
						|
	snd_pcm_hw_params_t sparams;
 | 
						|
	unsigned int links = (SND_PCM_HW_PARBIT_FRAGMENTS |
 | 
						|
			      SND_PCM_HW_PARBIT_FRAGMENT_SIZE |
 | 
						|
			      SND_PCM_HW_PARBIT_FRAGMENT_LENGTH |
 | 
						|
			      SND_PCM_HW_PARBIT_BUFFER_SIZE |
 | 
						|
			      SND_PCM_HW_PARBIT_BUFFER_LENGTH);
 | 
						|
	unsigned int src_format, dst_format;
 | 
						|
	mask_t *saccess_mask = alloca(mask_sizeof());
 | 
						|
	mask_load(saccess_mask, SND_PCM_ACCBIT_MMAP);
 | 
						|
 | 
						|
	_snd_pcm_hw_params_any(&sparams);
 | 
						|
	_snd_pcm_hw_param_mask(&sparams, 0, SND_PCM_HW_PARAM_ACCESS,
 | 
						|
				saccess_mask);
 | 
						|
	if (route->sformat >= 0) {
 | 
						|
		_snd_pcm_hw_param_set(&sparams, 0, SND_PCM_HW_PARAM_FORMAT,
 | 
						|
				       route->sformat);
 | 
						|
		_snd_pcm_hw_param_set(&sparams, 0, SND_PCM_HW_PARAM_SUBFORMAT,
 | 
						|
				       SND_PCM_SUBFORMAT_STD);
 | 
						|
	} else
 | 
						|
		links |= (SND_PCM_HW_PARBIT_FORMAT | 
 | 
						|
			  SND_PCM_HW_PARBIT_SUBFORMAT |
 | 
						|
			  SND_PCM_HW_PARBIT_SAMPLE_BITS);
 | 
						|
	if (route->schannels >= 0) {
 | 
						|
		_snd_pcm_hw_param_set(&sparams, 0, SND_PCM_HW_PARAM_CHANNELS,
 | 
						|
				      route->schannels);
 | 
						|
	} else {
 | 
						|
		links |= SND_PCM_HW_PARBIT_CHANNELS;
 | 
						|
		if (route->sformat < 0)
 | 
						|
			links |= (SND_PCM_HW_PARBIT_FRAME_BITS |
 | 
						|
				  SND_PCM_HW_PARBIT_FRAGMENT_BYTES |
 | 
						|
				  SND_PCM_HW_PARBIT_BUFFER_BYTES);
 | 
						|
	}
 | 
						|
 | 
						|
	err = snd_pcm_hw_params2(params, &sparams, 
 | 
						|
				 snd_pcm_hw_params, slave, links);
 | 
						|
	if (err < 0)
 | 
						|
		return err;
 | 
						|
	params->info &= ~(SND_PCM_INFO_MMAP | SND_PCM_INFO_MMAP_VALID);
 | 
						|
	if (pcm->stream == SND_PCM_STREAM_PLAYBACK) {
 | 
						|
		src_format = snd_pcm_hw_param_value(params, SND_PCM_HW_PARAM_FORMAT);
 | 
						|
		dst_format = slave->format;
 | 
						|
	} else {
 | 
						|
		src_format = slave->format;
 | 
						|
		dst_format = snd_pcm_hw_param_value(params, SND_PCM_HW_PARAM_FORMAT);
 | 
						|
	}
 | 
						|
	route->params.get_idx = get_index(src_format, SND_PCM_FORMAT_U16);
 | 
						|
	route->params.put_idx = put_index(SND_PCM_FORMAT_U32, dst_format);
 | 
						|
	route->params.conv_idx = conv_index(src_format, dst_format);
 | 
						|
	route->params.src_size = snd_pcm_format_width(src_format) / 8;
 | 
						|
	route->params.dst_sfmt = dst_format;
 | 
						|
#if ROUTE_PLUGIN_FLOAT
 | 
						|
	route->params.sum_idx = FLOAT;
 | 
						|
#else
 | 
						|
	if (src_size == 4)
 | 
						|
		route->params.sum_idx = UINT64;
 | 
						|
	else
 | 
						|
		route->params.sum_idx = UINT32;
 | 
						|
#endif
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static ssize_t snd_pcm_route_write_areas(snd_pcm_t *pcm,
 | 
						|
					 const snd_pcm_channel_area_t *areas,
 | 
						|
					 size_t offset,
 | 
						|
					 size_t size,
 | 
						|
					 size_t *slave_sizep)
 | 
						|
{
 | 
						|
	snd_pcm_route_t *route = pcm->private;
 | 
						|
	snd_pcm_t *slave = route->plug.slave;
 | 
						|
	size_t xfer = 0;
 | 
						|
	ssize_t err = 0;
 | 
						|
	if (slave_sizep && *slave_sizep < size)
 | 
						|
		size = *slave_sizep;
 | 
						|
	assert(size > 0);
 | 
						|
	while (xfer < size) {
 | 
						|
		size_t frames = snd_pcm_mmap_playback_xfer(slave, size - xfer);
 | 
						|
		route_transfer(areas, offset, 
 | 
						|
			       snd_pcm_mmap_areas(slave), snd_pcm_mmap_offset(slave),
 | 
						|
			       slave->channels, frames, &route->params);
 | 
						|
		err = snd_pcm_mmap_forward(slave, frames);
 | 
						|
		if (err < 0)
 | 
						|
			break;
 | 
						|
		assert((size_t)err == frames);
 | 
						|
		offset += err;
 | 
						|
		xfer += err;
 | 
						|
		snd_pcm_mmap_hw_forward(pcm, err);
 | 
						|
	}
 | 
						|
	if (xfer > 0) {
 | 
						|
		if (slave_sizep)
 | 
						|
			*slave_sizep = xfer;
 | 
						|
		return xfer;
 | 
						|
	}
 | 
						|
	return err;
 | 
						|
}
 | 
						|
 | 
						|
static ssize_t snd_pcm_route_read_areas(snd_pcm_t *pcm,
 | 
						|
					 const snd_pcm_channel_area_t *areas,
 | 
						|
					 size_t offset,
 | 
						|
					 size_t size,
 | 
						|
					 size_t *slave_sizep)
 | 
						|
{
 | 
						|
	snd_pcm_route_t *route = pcm->private;
 | 
						|
	snd_pcm_t *slave = route->plug.slave;
 | 
						|
	size_t xfer = 0;
 | 
						|
	ssize_t err = 0;
 | 
						|
	if (slave_sizep && *slave_sizep < size)
 | 
						|
		size = *slave_sizep;
 | 
						|
	assert(size > 0);
 | 
						|
	while (xfer < size) {
 | 
						|
		size_t frames = snd_pcm_mmap_capture_xfer(slave, size - xfer);
 | 
						|
		route_transfer(snd_pcm_mmap_areas(slave), snd_pcm_mmap_offset(slave),
 | 
						|
			       areas, offset, 
 | 
						|
			       pcm->channels, frames, &route->params);
 | 
						|
		err = snd_pcm_mmap_forward(slave, frames);
 | 
						|
		if (err < 0)
 | 
						|
			break;
 | 
						|
		assert((size_t)err == frames);
 | 
						|
		offset += err;
 | 
						|
		xfer += err;
 | 
						|
		snd_pcm_mmap_hw_forward(pcm, err);
 | 
						|
	}
 | 
						|
	if (xfer > 0) {
 | 
						|
		if (slave_sizep)
 | 
						|
			*slave_sizep = xfer;
 | 
						|
		return xfer;
 | 
						|
	}
 | 
						|
	return err;
 | 
						|
}
 | 
						|
 | 
						|
static void snd_pcm_route_dump(snd_pcm_t *pcm, FILE *fp)
 | 
						|
{
 | 
						|
	snd_pcm_route_t *route = pcm->private;
 | 
						|
	unsigned int dst;
 | 
						|
	if (route->sformat < 0)
 | 
						|
		fprintf(fp, "Route conversion PCM\n");
 | 
						|
	else
 | 
						|
		fprintf(fp, "Route conversion PCM (sformat=%s)\n", 
 | 
						|
			snd_pcm_format_name(route->sformat));
 | 
						|
	fputs("Transformation table:\n", fp);
 | 
						|
	for (dst = 0; dst < route->params.ndsts; dst++) {
 | 
						|
		ttable_dst_t *d = &route->params.dsts[dst];
 | 
						|
		unsigned int src;
 | 
						|
		if (d->nsrcs == 0)
 | 
						|
			continue;
 | 
						|
		fprintf(fp, "%d <- ", dst);
 | 
						|
		src = 0;
 | 
						|
		while (1) {
 | 
						|
			ttable_src_t *s = &d->srcs[src];
 | 
						|
			if (d->att)
 | 
						|
				fprintf(fp, "%d*%g", s->channel, s->as_float);
 | 
						|
			else
 | 
						|
				fprintf(fp, "%d", s->channel);
 | 
						|
			src++;
 | 
						|
			if (src == d->nsrcs)
 | 
						|
				break;
 | 
						|
			fputs(" + ", fp);
 | 
						|
		}
 | 
						|
		putc('\n', fp);
 | 
						|
	}
 | 
						|
	if (pcm->setup) {
 | 
						|
		fprintf(fp, "Its setup is:\n");
 | 
						|
		snd_pcm_dump_setup(pcm, fp);
 | 
						|
	}
 | 
						|
	fprintf(fp, "Slave: ");
 | 
						|
	snd_pcm_dump(route->plug.slave, fp);
 | 
						|
}
 | 
						|
 | 
						|
snd_pcm_ops_t snd_pcm_route_ops = {
 | 
						|
	close: snd_pcm_route_close,
 | 
						|
	card: snd_pcm_plugin_card,
 | 
						|
	info: snd_pcm_plugin_info,
 | 
						|
	hw_refine: snd_pcm_route_hw_refine,
 | 
						|
	hw_params: snd_pcm_route_hw_params,
 | 
						|
	sw_params: snd_pcm_plugin_sw_params,
 | 
						|
	channel_info: snd_pcm_plugin_channel_info,
 | 
						|
	dump: snd_pcm_route_dump,
 | 
						|
	nonblock: snd_pcm_plugin_nonblock,
 | 
						|
	async: snd_pcm_plugin_async,
 | 
						|
	mmap: snd_pcm_plugin_mmap,
 | 
						|
	munmap: snd_pcm_plugin_munmap,
 | 
						|
};
 | 
						|
 | 
						|
int route_load_ttable(route_params_t *params, int stream,
 | 
						|
		      unsigned int tt_ssize,
 | 
						|
		      ttable_entry_t *ttable,
 | 
						|
		      unsigned int tt_cused, unsigned int tt_sused)
 | 
						|
{
 | 
						|
	unsigned int src_channel, dst_channel;
 | 
						|
	ttable_dst_t *dptr;
 | 
						|
	unsigned int sused, dused, smul, dmul;
 | 
						|
	if (stream == SND_PCM_STREAM_PLAYBACK) {
 | 
						|
		sused = tt_cused;
 | 
						|
		dused = tt_sused;
 | 
						|
		smul = tt_ssize;
 | 
						|
		dmul = 1;
 | 
						|
	} else {
 | 
						|
		sused = tt_sused;
 | 
						|
		dused = tt_cused;
 | 
						|
		smul = 1;
 | 
						|
		dmul = tt_ssize;
 | 
						|
	}
 | 
						|
	params->ndsts = dused;
 | 
						|
	dptr = calloc(dused, sizeof(*params->dsts));
 | 
						|
	if (!dptr)
 | 
						|
		return -ENOMEM;
 | 
						|
	params->dsts = dptr;
 | 
						|
	for (dst_channel = 0; dst_channel < dused; ++dst_channel) {
 | 
						|
		ttable_entry_t t = 0;
 | 
						|
		int att = 0;
 | 
						|
		int nsrcs = 0;
 | 
						|
		ttable_src_t srcs[sused];
 | 
						|
		for (src_channel = 0; src_channel < sused; ++src_channel) {
 | 
						|
			ttable_entry_t v;
 | 
						|
			v = ttable[src_channel * smul + dst_channel * dmul];
 | 
						|
			assert(v >= 0 && v <= FULL);
 | 
						|
			if (v != 0) {
 | 
						|
				srcs[nsrcs].channel = src_channel;
 | 
						|
#if ROUTE_PLUGIN_FLOAT
 | 
						|
				/* Also in user space for non attenuated */
 | 
						|
				srcs[nsrcs].as_int = (v == FULL ? ROUTE_PLUGIN_RESOLUTION : 0);
 | 
						|
				srcs[nsrcs].as_float = v;
 | 
						|
#else
 | 
						|
				srcs[nsrcs].as_int = v;
 | 
						|
#endif
 | 
						|
				if (v != FULL)
 | 
						|
					att = 1;
 | 
						|
				t += v;
 | 
						|
				nsrcs++;
 | 
						|
			}
 | 
						|
		}
 | 
						|
#if 0
 | 
						|
		assert(t <= FULL);
 | 
						|
#endif
 | 
						|
		dptr->att = att;
 | 
						|
		dptr->nsrcs = nsrcs;
 | 
						|
		if (nsrcs == 0)
 | 
						|
			dptr->func = route1_zero;
 | 
						|
		else if (nsrcs == 1 && !att)
 | 
						|
			dptr->func = route1_one;
 | 
						|
		else
 | 
						|
			dptr->func = route1_many;
 | 
						|
		if (nsrcs > 0) {
 | 
						|
			dptr->srcs = calloc(nsrcs, sizeof(*srcs));
 | 
						|
			if (!dptr->srcs)
 | 
						|
				return -ENOMEM;
 | 
						|
			memcpy(dptr->srcs, srcs, sizeof(*srcs) * nsrcs);
 | 
						|
		} else
 | 
						|
			dptr->srcs = 0;
 | 
						|
		dptr++;
 | 
						|
	}
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
int snd_pcm_route_open(snd_pcm_t **pcmp, char *name,
 | 
						|
		       int sformat, unsigned int schannels,
 | 
						|
		       ttable_entry_t *ttable,
 | 
						|
		       unsigned int tt_ssize,
 | 
						|
		       unsigned int tt_cused, unsigned int tt_sused,
 | 
						|
		       snd_pcm_t *slave, int close_slave)
 | 
						|
{
 | 
						|
	snd_pcm_t *pcm;
 | 
						|
	snd_pcm_route_t *route;
 | 
						|
	int err;
 | 
						|
	assert(pcmp && slave && ttable);
 | 
						|
	if (sformat >= 0 && snd_pcm_format_linear(sformat) != 1)
 | 
						|
		return -EINVAL;
 | 
						|
	route = calloc(1, sizeof(snd_pcm_route_t));
 | 
						|
	if (!route) {
 | 
						|
		return -ENOMEM;
 | 
						|
	}
 | 
						|
	route->sformat = sformat;
 | 
						|
	route->schannels = schannels;
 | 
						|
	route->plug.read = snd_pcm_route_read_areas;
 | 
						|
	route->plug.write = snd_pcm_route_write_areas;
 | 
						|
	route->plug.slave = slave;
 | 
						|
	route->plug.close_slave = close_slave;
 | 
						|
 | 
						|
	pcm = calloc(1, sizeof(snd_pcm_t));
 | 
						|
	if (!pcm) {
 | 
						|
		free(route);
 | 
						|
		return -ENOMEM;
 | 
						|
	}
 | 
						|
	if (name)
 | 
						|
		pcm->name = strdup(name);
 | 
						|
	pcm->type = SND_PCM_TYPE_ROUTE;
 | 
						|
	pcm->stream = slave->stream;
 | 
						|
	pcm->mode = slave->mode;
 | 
						|
	pcm->ops = &snd_pcm_route_ops;
 | 
						|
	pcm->op_arg = pcm;
 | 
						|
	pcm->fast_ops = &snd_pcm_plugin_fast_ops;
 | 
						|
	pcm->fast_op_arg = pcm;
 | 
						|
	pcm->private = route;
 | 
						|
	pcm->poll_fd = slave->poll_fd;
 | 
						|
	pcm->hw_ptr = &route->plug.hw_ptr;
 | 
						|
	pcm->appl_ptr = &route->plug.appl_ptr;
 | 
						|
	err = route_load_ttable(&route->params, pcm->stream, tt_ssize, ttable, tt_cused, tt_sused);
 | 
						|
	if (err < 0) {
 | 
						|
		snd_pcm_close(pcm);
 | 
						|
		return err;
 | 
						|
	}
 | 
						|
	*pcmp = pcm;
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
int snd_pcm_route_load_ttable(snd_config_t *tt, ttable_entry_t *ttable,
 | 
						|
			      unsigned int tt_csize, unsigned int tt_ssize,
 | 
						|
			      unsigned int *tt_cused, unsigned int *tt_sused,
 | 
						|
			      int schannels)
 | 
						|
{
 | 
						|
	int cused = -1;
 | 
						|
	int sused = -1;
 | 
						|
	snd_config_iterator_t i;
 | 
						|
	unsigned int k;
 | 
						|
	for (k = 0; k < tt_csize * tt_ssize; ++k)
 | 
						|
		ttable[k] = 0.0;
 | 
						|
	snd_config_foreach(i, tt) {
 | 
						|
		snd_config_t *in = snd_config_entry(i);
 | 
						|
		snd_config_iterator_t j;
 | 
						|
		char *p;
 | 
						|
		long cchannel;
 | 
						|
		errno = 0;
 | 
						|
		cchannel = strtol(in->id, &p, 10);
 | 
						|
		if (errno || *p || 
 | 
						|
		    cchannel < 0 || (unsigned int) cchannel > tt_csize) {
 | 
						|
			ERR("Invalid client channel: %s", in->id);
 | 
						|
			return -EINVAL;
 | 
						|
		}
 | 
						|
		if (snd_config_type(in) != SND_CONFIG_TYPE_COMPOUND)
 | 
						|
			return -EINVAL;
 | 
						|
		snd_config_foreach(j, in) {
 | 
						|
			snd_config_t *jn = snd_config_entry(j);
 | 
						|
			double value;
 | 
						|
			long schannel;
 | 
						|
			int err;
 | 
						|
			errno = 0;
 | 
						|
			schannel = strtol(jn->id, &p, 10);
 | 
						|
			if (errno || *p || 
 | 
						|
			    schannel < 0 || (unsigned int) schannel > tt_ssize || 
 | 
						|
			    (schannels > 0 && schannel >= schannels)) {
 | 
						|
				ERR("Invalid slave channel: %s", jn->id);
 | 
						|
				return -EINVAL;
 | 
						|
			}
 | 
						|
			err = snd_config_real_get(jn, &value);
 | 
						|
			if (err < 0) {
 | 
						|
				long v;
 | 
						|
				err = snd_config_integer_get(jn, &v);
 | 
						|
				if (err < 0) {
 | 
						|
					ERR("Invalid type for %s", jn->id);
 | 
						|
					return -EINVAL;
 | 
						|
				}
 | 
						|
				value = v;
 | 
						|
			}
 | 
						|
			ttable[cchannel * tt_ssize + schannel] = value;
 | 
						|
			if (schannel > sused)
 | 
						|
				sused = schannel;
 | 
						|
		}
 | 
						|
		if (cchannel > cused)
 | 
						|
			cused = cchannel;
 | 
						|
	}
 | 
						|
	*tt_sused = sused + 1;
 | 
						|
	*tt_cused = cused + 1;
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
#define MAX_CHANNELS 32
 | 
						|
 | 
						|
int _snd_pcm_route_open(snd_pcm_t **pcmp, char *name,
 | 
						|
			snd_config_t *conf, 
 | 
						|
			int stream, int mode)
 | 
						|
{
 | 
						|
	snd_config_iterator_t i;
 | 
						|
	char *sname = NULL;
 | 
						|
	int err;
 | 
						|
	snd_pcm_t *spcm;
 | 
						|
	int sformat = -1;
 | 
						|
	long schannels = -1;
 | 
						|
	snd_config_t *tt = NULL;
 | 
						|
	ttable_entry_t ttable[MAX_CHANNELS*MAX_CHANNELS];
 | 
						|
	unsigned int cused, sused;
 | 
						|
	snd_config_foreach(i, conf) {
 | 
						|
		snd_config_t *n = snd_config_entry(i);
 | 
						|
		if (strcmp(n->id, "comment") == 0)
 | 
						|
			continue;
 | 
						|
		if (strcmp(n->id, "type") == 0)
 | 
						|
			continue;
 | 
						|
		if (strcmp(n->id, "stream") == 0)
 | 
						|
			continue;
 | 
						|
		if (strcmp(n->id, "sname") == 0) {
 | 
						|
			err = snd_config_string_get(n, &sname);
 | 
						|
			if (err < 0) {
 | 
						|
				ERR("Invalid type for %s", n->id);
 | 
						|
				return -EINVAL;
 | 
						|
			}
 | 
						|
			continue;
 | 
						|
		}
 | 
						|
		if (strcmp(n->id, "sformat") == 0) {
 | 
						|
			char *f;
 | 
						|
			err = snd_config_string_get(n, &f);
 | 
						|
			if (err < 0) {
 | 
						|
				ERR("Invalid type for %s", n->id);
 | 
						|
				return -EINVAL;
 | 
						|
			}
 | 
						|
			sformat = snd_pcm_format_value(f);
 | 
						|
			if (sformat < 0) {
 | 
						|
				ERR("Unknown sformat");
 | 
						|
				return -EINVAL;
 | 
						|
			}
 | 
						|
			if (snd_pcm_format_linear(sformat) != 1) {
 | 
						|
				ERR("sformat is not linear");
 | 
						|
				return -EINVAL;
 | 
						|
			}
 | 
						|
			continue;
 | 
						|
		}
 | 
						|
		if (strcmp(n->id, "schannels") == 0) {
 | 
						|
			err = snd_config_integer_get(n, &schannels);
 | 
						|
			if (err < 0) {
 | 
						|
				ERR("Invalid type for %s", n->id);
 | 
						|
				return -EINVAL;
 | 
						|
			}
 | 
						|
			continue;
 | 
						|
		}
 | 
						|
		if (strcmp(n->id, "ttable") == 0) {
 | 
						|
			if (snd_config_type(n) != SND_CONFIG_TYPE_COMPOUND) {
 | 
						|
				ERR("Invalid type for %s", n->id);
 | 
						|
				return -EINVAL;
 | 
						|
			}
 | 
						|
			tt = n;
 | 
						|
			continue;
 | 
						|
		}
 | 
						|
		ERR("Unknown field %s", n->id);
 | 
						|
		return -EINVAL;
 | 
						|
	}
 | 
						|
	if (!sname) {
 | 
						|
		ERR("sname is not defined");
 | 
						|
		return -EINVAL;
 | 
						|
	}
 | 
						|
	if (!tt) {
 | 
						|
		ERR("ttable is not defined");
 | 
						|
		return -EINVAL;
 | 
						|
	}
 | 
						|
 | 
						|
	err = snd_pcm_route_load_ttable(tt, ttable, MAX_CHANNELS, MAX_CHANNELS,
 | 
						|
					&cused, &sused, schannels);
 | 
						|
	if (err < 0)
 | 
						|
		return err;
 | 
						|
 | 
						|
	/* This is needed cause snd_config_update may destroy config */
 | 
						|
	sname = strdup(sname);
 | 
						|
	if (!sname)
 | 
						|
		return  -ENOMEM;
 | 
						|
	err = snd_pcm_open(&spcm, sname, stream, mode);
 | 
						|
	free(sname);
 | 
						|
	if (err < 0)
 | 
						|
		return err;
 | 
						|
	err = snd_pcm_route_open(pcmp, name, sformat, schannels,
 | 
						|
				 ttable, MAX_CHANNELS,
 | 
						|
				 cused, sused,
 | 
						|
				 spcm, 1);
 | 
						|
	if (err < 0)
 | 
						|
		snd_pcm_close(spcm);
 | 
						|
	return err;
 | 
						|
}
 | 
						|
				
 | 
						|
 |