xref: /third_party/alsa-lib/src/pcm/pcm_alaw.c (revision d5ac70f0)
1d5ac70f0Sopenharmony_ci/**
2d5ac70f0Sopenharmony_ci * \file pcm/pcm_alaw.c
3d5ac70f0Sopenharmony_ci * \ingroup PCM_Plugins
4d5ac70f0Sopenharmony_ci * \brief PCM A-Law Conversion Plugin Interface
5d5ac70f0Sopenharmony_ci * \author Abramo Bagnara <abramo@alsa-project.org>
6d5ac70f0Sopenharmony_ci * \date 2000-2001
7d5ac70f0Sopenharmony_ci */
8d5ac70f0Sopenharmony_ci/*
9d5ac70f0Sopenharmony_ci *  PCM - A-Law conversion
10d5ac70f0Sopenharmony_ci *  Copyright (c) 2000 by Abramo Bagnara <abramo@alsa-project.org>
11d5ac70f0Sopenharmony_ci *
12d5ac70f0Sopenharmony_ci *
13d5ac70f0Sopenharmony_ci *   This library is free software; you can redistribute it and/or modify
14d5ac70f0Sopenharmony_ci *   it under the terms of the GNU Lesser General Public License as
15d5ac70f0Sopenharmony_ci *   published by the Free Software Foundation; either version 2.1 of
16d5ac70f0Sopenharmony_ci *   the License, or (at your option) any later version.
17d5ac70f0Sopenharmony_ci *
18d5ac70f0Sopenharmony_ci *   This program is distributed in the hope that it will be useful,
19d5ac70f0Sopenharmony_ci *   but WITHOUT ANY WARRANTY; without even the implied warranty of
20d5ac70f0Sopenharmony_ci *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
21d5ac70f0Sopenharmony_ci *   GNU Lesser General Public License for more details.
22d5ac70f0Sopenharmony_ci *
23d5ac70f0Sopenharmony_ci *   You should have received a copy of the GNU Lesser General Public
24d5ac70f0Sopenharmony_ci *   License along with this library; if not, write to the Free Software
25d5ac70f0Sopenharmony_ci *   Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
26d5ac70f0Sopenharmony_ci *
27d5ac70f0Sopenharmony_ci */
28d5ac70f0Sopenharmony_ci
29d5ac70f0Sopenharmony_ci#include "pcm_local.h"
30d5ac70f0Sopenharmony_ci#include "bswap.h"
31d5ac70f0Sopenharmony_ci#include "pcm_plugin.h"
32d5ac70f0Sopenharmony_ci
33d5ac70f0Sopenharmony_ci#include "plugin_ops.h"
34d5ac70f0Sopenharmony_ci
35d5ac70f0Sopenharmony_ci#ifndef PIC
36d5ac70f0Sopenharmony_ci/* entry for static linking */
37d5ac70f0Sopenharmony_ciconst char *_snd_module_pcm_alaw = "";
38d5ac70f0Sopenharmony_ci#endif
39d5ac70f0Sopenharmony_ci
40d5ac70f0Sopenharmony_ci#ifndef DOC_HIDDEN
41d5ac70f0Sopenharmony_ci
42d5ac70f0Sopenharmony_citypedef void (*alaw_f)(const snd_pcm_channel_area_t *dst_areas,
43d5ac70f0Sopenharmony_ci		       snd_pcm_uframes_t dst_offset,
44d5ac70f0Sopenharmony_ci		       const snd_pcm_channel_area_t *src_areas,
45d5ac70f0Sopenharmony_ci		       snd_pcm_uframes_t src_offset,
46d5ac70f0Sopenharmony_ci		       unsigned int channels, snd_pcm_uframes_t frames,
47d5ac70f0Sopenharmony_ci		       unsigned int getputidx);
48d5ac70f0Sopenharmony_ci
49d5ac70f0Sopenharmony_citypedef struct {
50d5ac70f0Sopenharmony_ci	/* This field need to be the first */
51d5ac70f0Sopenharmony_ci	snd_pcm_plugin_t plug;
52d5ac70f0Sopenharmony_ci	unsigned int getput_idx;
53d5ac70f0Sopenharmony_ci	alaw_f func;
54d5ac70f0Sopenharmony_ci	snd_pcm_format_t sformat;
55d5ac70f0Sopenharmony_ci} snd_pcm_alaw_t;
56d5ac70f0Sopenharmony_ci
57d5ac70f0Sopenharmony_ci#endif
58d5ac70f0Sopenharmony_ci
59d5ac70f0Sopenharmony_cistatic inline int val_seg(int val)
60d5ac70f0Sopenharmony_ci{
61d5ac70f0Sopenharmony_ci	int r = 1;
62d5ac70f0Sopenharmony_ci	val >>= 8;
63d5ac70f0Sopenharmony_ci	if (val & 0xf0) {
64d5ac70f0Sopenharmony_ci		val >>= 4;
65d5ac70f0Sopenharmony_ci		r += 4;
66d5ac70f0Sopenharmony_ci	}
67d5ac70f0Sopenharmony_ci	if (val & 0x0c) {
68d5ac70f0Sopenharmony_ci		val >>= 2;
69d5ac70f0Sopenharmony_ci		r += 2;
70d5ac70f0Sopenharmony_ci	}
71d5ac70f0Sopenharmony_ci	if (val & 0x02)
72d5ac70f0Sopenharmony_ci		r += 1;
73d5ac70f0Sopenharmony_ci	return r;
74d5ac70f0Sopenharmony_ci}
75d5ac70f0Sopenharmony_ci
76d5ac70f0Sopenharmony_ci/*
77d5ac70f0Sopenharmony_ci * s16_to_alaw() - Convert a 16-bit linear PCM value to 8-bit A-law
78d5ac70f0Sopenharmony_ci *
79d5ac70f0Sopenharmony_ci * s16_to_alaw() accepts an 16-bit integer and encodes it as A-law data.
80d5ac70f0Sopenharmony_ci *
81d5ac70f0Sopenharmony_ci *		Linear Input Code	Compressed Code
82d5ac70f0Sopenharmony_ci *	------------------------	---------------
83d5ac70f0Sopenharmony_ci *	0000000wxyza			000wxyz
84d5ac70f0Sopenharmony_ci *	0000001wxyza			001wxyz
85d5ac70f0Sopenharmony_ci *	000001wxyzab			010wxyz
86d5ac70f0Sopenharmony_ci *	00001wxyzabc			011wxyz
87d5ac70f0Sopenharmony_ci *	0001wxyzabcd			100wxyz
88d5ac70f0Sopenharmony_ci *	001wxyzabcde			101wxyz
89d5ac70f0Sopenharmony_ci *	01wxyzabcdef			110wxyz
90d5ac70f0Sopenharmony_ci *	1wxyzabcdefg			111wxyz
91d5ac70f0Sopenharmony_ci *
92d5ac70f0Sopenharmony_ci * For further information see John C. Bellamy's Digital Telephony, 1982,
93d5ac70f0Sopenharmony_ci * John Wiley & Sons, pps 98-111 and 472-476.
94d5ac70f0Sopenharmony_ci */
95d5ac70f0Sopenharmony_ci
96d5ac70f0Sopenharmony_cistatic unsigned char s16_to_alaw(int pcm_val)
97d5ac70f0Sopenharmony_ci{
98d5ac70f0Sopenharmony_ci	int		mask;
99d5ac70f0Sopenharmony_ci	int		seg;
100d5ac70f0Sopenharmony_ci	unsigned char	aval;
101d5ac70f0Sopenharmony_ci
102d5ac70f0Sopenharmony_ci	if (pcm_val >= 0) {
103d5ac70f0Sopenharmony_ci		mask = 0xD5;
104d5ac70f0Sopenharmony_ci	} else {
105d5ac70f0Sopenharmony_ci		mask = 0x55;
106d5ac70f0Sopenharmony_ci		pcm_val = -pcm_val;
107d5ac70f0Sopenharmony_ci		if (pcm_val > 0x7fff)
108d5ac70f0Sopenharmony_ci			pcm_val = 0x7fff;
109d5ac70f0Sopenharmony_ci	}
110d5ac70f0Sopenharmony_ci
111d5ac70f0Sopenharmony_ci	if (pcm_val < 256)
112d5ac70f0Sopenharmony_ci		aval = pcm_val >> 4;
113d5ac70f0Sopenharmony_ci	else {
114d5ac70f0Sopenharmony_ci		/* Convert the scaled magnitude to segment number. */
115d5ac70f0Sopenharmony_ci		seg = val_seg(pcm_val);
116d5ac70f0Sopenharmony_ci		aval = (seg << 4) | ((pcm_val >> (seg + 3)) & 0x0f);
117d5ac70f0Sopenharmony_ci	}
118d5ac70f0Sopenharmony_ci	return aval ^ mask;
119d5ac70f0Sopenharmony_ci}
120d5ac70f0Sopenharmony_ci
121d5ac70f0Sopenharmony_ci/*
122d5ac70f0Sopenharmony_ci * alaw_to_s16() - Convert an A-law value to 16-bit linear PCM
123d5ac70f0Sopenharmony_ci *
124d5ac70f0Sopenharmony_ci */
125d5ac70f0Sopenharmony_cistatic int alaw_to_s16(unsigned char a_val)
126d5ac70f0Sopenharmony_ci{
127d5ac70f0Sopenharmony_ci	int		t;
128d5ac70f0Sopenharmony_ci	int		seg;
129d5ac70f0Sopenharmony_ci
130d5ac70f0Sopenharmony_ci	a_val ^= 0x55;
131d5ac70f0Sopenharmony_ci	t = a_val & 0x7f;
132d5ac70f0Sopenharmony_ci	if (t < 16)
133d5ac70f0Sopenharmony_ci		t = (t << 4) + 8;
134d5ac70f0Sopenharmony_ci	else {
135d5ac70f0Sopenharmony_ci		seg = (t >> 4) & 0x07;
136d5ac70f0Sopenharmony_ci		t = ((t & 0x0f) << 4) + 0x108;
137d5ac70f0Sopenharmony_ci		t <<= seg -1;
138d5ac70f0Sopenharmony_ci	}
139d5ac70f0Sopenharmony_ci	return ((a_val & 0x80) ? t : -t);
140d5ac70f0Sopenharmony_ci}
141d5ac70f0Sopenharmony_ci
142d5ac70f0Sopenharmony_ci#ifndef DOC_HIDDEN
143d5ac70f0Sopenharmony_ci
144d5ac70f0Sopenharmony_civoid snd_pcm_alaw_decode(const snd_pcm_channel_area_t *dst_areas,
145d5ac70f0Sopenharmony_ci			 snd_pcm_uframes_t dst_offset,
146d5ac70f0Sopenharmony_ci			 const snd_pcm_channel_area_t *src_areas,
147d5ac70f0Sopenharmony_ci			 snd_pcm_uframes_t src_offset,
148d5ac70f0Sopenharmony_ci			 unsigned int channels, snd_pcm_uframes_t frames,
149d5ac70f0Sopenharmony_ci			 unsigned int putidx)
150d5ac70f0Sopenharmony_ci{
151d5ac70f0Sopenharmony_ci#define PUT16_LABELS
152d5ac70f0Sopenharmony_ci#include "plugin_ops.h"
153d5ac70f0Sopenharmony_ci#undef PUT16_LABELS
154d5ac70f0Sopenharmony_ci	void *put = put16_labels[putidx];
155d5ac70f0Sopenharmony_ci	unsigned int channel;
156d5ac70f0Sopenharmony_ci	for (channel = 0; channel < channels; ++channel) {
157d5ac70f0Sopenharmony_ci		const unsigned char *src;
158d5ac70f0Sopenharmony_ci		char *dst;
159d5ac70f0Sopenharmony_ci		int src_step, dst_step;
160d5ac70f0Sopenharmony_ci		snd_pcm_uframes_t frames1;
161d5ac70f0Sopenharmony_ci		const snd_pcm_channel_area_t *src_area = &src_areas[channel];
162d5ac70f0Sopenharmony_ci		const snd_pcm_channel_area_t *dst_area = &dst_areas[channel];
163d5ac70f0Sopenharmony_ci		src = snd_pcm_channel_area_addr(src_area, src_offset);
164d5ac70f0Sopenharmony_ci		dst = snd_pcm_channel_area_addr(dst_area, dst_offset);
165d5ac70f0Sopenharmony_ci		src_step = snd_pcm_channel_area_step(src_area);
166d5ac70f0Sopenharmony_ci		dst_step = snd_pcm_channel_area_step(dst_area);
167d5ac70f0Sopenharmony_ci		frames1 = frames;
168d5ac70f0Sopenharmony_ci		while (frames1-- > 0) {
169d5ac70f0Sopenharmony_ci			int16_t sample = alaw_to_s16(*src);
170d5ac70f0Sopenharmony_ci			goto *put;
171d5ac70f0Sopenharmony_ci#define PUT16_END after
172d5ac70f0Sopenharmony_ci#include "plugin_ops.h"
173d5ac70f0Sopenharmony_ci#undef PUT16_END
174d5ac70f0Sopenharmony_ci		after:
175d5ac70f0Sopenharmony_ci			src += src_step;
176d5ac70f0Sopenharmony_ci			dst += dst_step;
177d5ac70f0Sopenharmony_ci		}
178d5ac70f0Sopenharmony_ci	}
179d5ac70f0Sopenharmony_ci}
180d5ac70f0Sopenharmony_ci
181d5ac70f0Sopenharmony_civoid snd_pcm_alaw_encode(const snd_pcm_channel_area_t *dst_areas,
182d5ac70f0Sopenharmony_ci			 snd_pcm_uframes_t dst_offset,
183d5ac70f0Sopenharmony_ci			 const snd_pcm_channel_area_t *src_areas,
184d5ac70f0Sopenharmony_ci			 snd_pcm_uframes_t src_offset,
185d5ac70f0Sopenharmony_ci			 unsigned int channels, snd_pcm_uframes_t frames,
186d5ac70f0Sopenharmony_ci			 unsigned int getidx)
187d5ac70f0Sopenharmony_ci{
188d5ac70f0Sopenharmony_ci#define GET16_LABELS
189d5ac70f0Sopenharmony_ci#include "plugin_ops.h"
190d5ac70f0Sopenharmony_ci#undef GET16_LABELS
191d5ac70f0Sopenharmony_ci	void *get = get16_labels[getidx];
192d5ac70f0Sopenharmony_ci	unsigned int channel;
193d5ac70f0Sopenharmony_ci	int16_t sample = 0;
194d5ac70f0Sopenharmony_ci	for (channel = 0; channel < channels; ++channel) {
195d5ac70f0Sopenharmony_ci		const char *src;
196d5ac70f0Sopenharmony_ci		char *dst;
197d5ac70f0Sopenharmony_ci		int src_step, dst_step;
198d5ac70f0Sopenharmony_ci		snd_pcm_uframes_t frames1;
199d5ac70f0Sopenharmony_ci		const snd_pcm_channel_area_t *src_area = &src_areas[channel];
200d5ac70f0Sopenharmony_ci		const snd_pcm_channel_area_t *dst_area = &dst_areas[channel];
201d5ac70f0Sopenharmony_ci		src = snd_pcm_channel_area_addr(src_area, src_offset);
202d5ac70f0Sopenharmony_ci		dst = snd_pcm_channel_area_addr(dst_area, dst_offset);
203d5ac70f0Sopenharmony_ci		src_step = snd_pcm_channel_area_step(src_area);
204d5ac70f0Sopenharmony_ci		dst_step = snd_pcm_channel_area_step(dst_area);
205d5ac70f0Sopenharmony_ci		frames1 = frames;
206d5ac70f0Sopenharmony_ci		while (frames1-- > 0) {
207d5ac70f0Sopenharmony_ci			goto *get;
208d5ac70f0Sopenharmony_ci#define GET16_END after
209d5ac70f0Sopenharmony_ci#include "plugin_ops.h"
210d5ac70f0Sopenharmony_ci#undef GET16_END
211d5ac70f0Sopenharmony_ci		after:
212d5ac70f0Sopenharmony_ci			*dst = s16_to_alaw(sample);
213d5ac70f0Sopenharmony_ci			src += src_step;
214d5ac70f0Sopenharmony_ci			dst += dst_step;
215d5ac70f0Sopenharmony_ci		}
216d5ac70f0Sopenharmony_ci	}
217d5ac70f0Sopenharmony_ci}
218d5ac70f0Sopenharmony_ci
219d5ac70f0Sopenharmony_ci#endif /* DOC_HIDDEN */
220d5ac70f0Sopenharmony_ci
221d5ac70f0Sopenharmony_cistatic int snd_pcm_alaw_hw_refine_cprepare(snd_pcm_t *pcm, snd_pcm_hw_params_t *params)
222d5ac70f0Sopenharmony_ci{
223d5ac70f0Sopenharmony_ci	snd_pcm_alaw_t *alaw = pcm->private_data;
224d5ac70f0Sopenharmony_ci	int err;
225d5ac70f0Sopenharmony_ci	snd_pcm_access_mask_t access_mask = { SND_PCM_ACCBIT_SHM };
226d5ac70f0Sopenharmony_ci	err = _snd_pcm_hw_param_set_mask(params, SND_PCM_HW_PARAM_ACCESS,
227d5ac70f0Sopenharmony_ci					 &access_mask);
228d5ac70f0Sopenharmony_ci	if (err < 0)
229d5ac70f0Sopenharmony_ci		return err;
230d5ac70f0Sopenharmony_ci	if (alaw->sformat == SND_PCM_FORMAT_A_LAW) {
231d5ac70f0Sopenharmony_ci		snd_pcm_format_mask_t format_mask = { SND_PCM_FMTBIT_LINEAR };
232d5ac70f0Sopenharmony_ci		err = _snd_pcm_hw_param_set_mask(params, SND_PCM_HW_PARAM_FORMAT,
233d5ac70f0Sopenharmony_ci						 &format_mask);
234d5ac70f0Sopenharmony_ci	} else {
235d5ac70f0Sopenharmony_ci		err = _snd_pcm_hw_params_set_format(params,
236d5ac70f0Sopenharmony_ci						   SND_PCM_FORMAT_A_LAW);
237d5ac70f0Sopenharmony_ci	}
238d5ac70f0Sopenharmony_ci	if (err < 0)
239d5ac70f0Sopenharmony_ci		return err;
240d5ac70f0Sopenharmony_ci	err = _snd_pcm_hw_params_set_subformat(params, SND_PCM_SUBFORMAT_STD);
241d5ac70f0Sopenharmony_ci	if (err < 0)
242d5ac70f0Sopenharmony_ci		return err;
243d5ac70f0Sopenharmony_ci	params->info &= ~(SND_PCM_INFO_MMAP | SND_PCM_INFO_MMAP_VALID);
244d5ac70f0Sopenharmony_ci	return 0;
245d5ac70f0Sopenharmony_ci}
246d5ac70f0Sopenharmony_ci
247d5ac70f0Sopenharmony_cistatic int snd_pcm_alaw_hw_refine_sprepare(snd_pcm_t *pcm, snd_pcm_hw_params_t *sparams)
248d5ac70f0Sopenharmony_ci{
249d5ac70f0Sopenharmony_ci	snd_pcm_alaw_t *alaw = pcm->private_data;
250d5ac70f0Sopenharmony_ci	snd_pcm_access_mask_t saccess_mask = { SND_PCM_ACCBIT_MMAP };
251d5ac70f0Sopenharmony_ci	_snd_pcm_hw_params_any(sparams);
252d5ac70f0Sopenharmony_ci	_snd_pcm_hw_param_set_mask(sparams, SND_PCM_HW_PARAM_ACCESS,
253d5ac70f0Sopenharmony_ci				   &saccess_mask);
254d5ac70f0Sopenharmony_ci	_snd_pcm_hw_params_set_format(sparams, alaw->sformat);
255d5ac70f0Sopenharmony_ci	_snd_pcm_hw_params_set_subformat(sparams, SND_PCM_SUBFORMAT_STD);
256d5ac70f0Sopenharmony_ci	return 0;
257d5ac70f0Sopenharmony_ci}
258d5ac70f0Sopenharmony_ci
259d5ac70f0Sopenharmony_cistatic int snd_pcm_alaw_hw_refine_schange(snd_pcm_t *pcm ATTRIBUTE_UNUSED, snd_pcm_hw_params_t *params,
260d5ac70f0Sopenharmony_ci					    snd_pcm_hw_params_t *sparams)
261d5ac70f0Sopenharmony_ci{
262d5ac70f0Sopenharmony_ci	int err;
263d5ac70f0Sopenharmony_ci	unsigned int links = (SND_PCM_HW_PARBIT_CHANNELS |
264d5ac70f0Sopenharmony_ci			      SND_PCM_HW_PARBIT_RATE |
265d5ac70f0Sopenharmony_ci			      SND_PCM_HW_PARBIT_PERIOD_SIZE |
266d5ac70f0Sopenharmony_ci			      SND_PCM_HW_PARBIT_BUFFER_SIZE |
267d5ac70f0Sopenharmony_ci			      SND_PCM_HW_PARBIT_PERIODS |
268d5ac70f0Sopenharmony_ci			      SND_PCM_HW_PARBIT_PERIOD_TIME |
269d5ac70f0Sopenharmony_ci			      SND_PCM_HW_PARBIT_BUFFER_TIME |
270d5ac70f0Sopenharmony_ci			      SND_PCM_HW_PARBIT_TICK_TIME);
271d5ac70f0Sopenharmony_ci	err = _snd_pcm_hw_params_refine(sparams, links, params);
272d5ac70f0Sopenharmony_ci	if (err < 0)
273d5ac70f0Sopenharmony_ci		return err;
274d5ac70f0Sopenharmony_ci	return 0;
275d5ac70f0Sopenharmony_ci}
276d5ac70f0Sopenharmony_ci
277d5ac70f0Sopenharmony_cistatic int snd_pcm_alaw_hw_refine_cchange(snd_pcm_t *pcm ATTRIBUTE_UNUSED, snd_pcm_hw_params_t *params,
278d5ac70f0Sopenharmony_ci					    snd_pcm_hw_params_t *sparams)
279d5ac70f0Sopenharmony_ci{
280d5ac70f0Sopenharmony_ci	int err;
281d5ac70f0Sopenharmony_ci	unsigned int links = (SND_PCM_HW_PARBIT_CHANNELS |
282d5ac70f0Sopenharmony_ci			      SND_PCM_HW_PARBIT_RATE |
283d5ac70f0Sopenharmony_ci			      SND_PCM_HW_PARBIT_PERIOD_SIZE |
284d5ac70f0Sopenharmony_ci			      SND_PCM_HW_PARBIT_BUFFER_SIZE |
285d5ac70f0Sopenharmony_ci			      SND_PCM_HW_PARBIT_PERIODS |
286d5ac70f0Sopenharmony_ci			      SND_PCM_HW_PARBIT_PERIOD_TIME |
287d5ac70f0Sopenharmony_ci			      SND_PCM_HW_PARBIT_BUFFER_TIME |
288d5ac70f0Sopenharmony_ci			      SND_PCM_HW_PARBIT_TICK_TIME);
289d5ac70f0Sopenharmony_ci	err = _snd_pcm_hw_params_refine(params, links, sparams);
290d5ac70f0Sopenharmony_ci	if (err < 0)
291d5ac70f0Sopenharmony_ci		return err;
292d5ac70f0Sopenharmony_ci	return 0;
293d5ac70f0Sopenharmony_ci}
294d5ac70f0Sopenharmony_ci
295d5ac70f0Sopenharmony_cistatic int snd_pcm_alaw_hw_refine(snd_pcm_t *pcm, snd_pcm_hw_params_t *params)
296d5ac70f0Sopenharmony_ci{
297d5ac70f0Sopenharmony_ci	return snd_pcm_hw_refine_slave(pcm, params,
298d5ac70f0Sopenharmony_ci				       snd_pcm_alaw_hw_refine_cprepare,
299d5ac70f0Sopenharmony_ci				       snd_pcm_alaw_hw_refine_cchange,
300d5ac70f0Sopenharmony_ci				       snd_pcm_alaw_hw_refine_sprepare,
301d5ac70f0Sopenharmony_ci				       snd_pcm_alaw_hw_refine_schange,
302d5ac70f0Sopenharmony_ci				       snd_pcm_generic_hw_refine);
303d5ac70f0Sopenharmony_ci}
304d5ac70f0Sopenharmony_ci
305d5ac70f0Sopenharmony_cistatic int snd_pcm_alaw_hw_params(snd_pcm_t *pcm, snd_pcm_hw_params_t * params)
306d5ac70f0Sopenharmony_ci{
307d5ac70f0Sopenharmony_ci	snd_pcm_alaw_t *alaw = pcm->private_data;
308d5ac70f0Sopenharmony_ci	snd_pcm_format_t format;
309d5ac70f0Sopenharmony_ci	int err = snd_pcm_hw_params_slave(pcm, params,
310d5ac70f0Sopenharmony_ci					  snd_pcm_alaw_hw_refine_cchange,
311d5ac70f0Sopenharmony_ci					  snd_pcm_alaw_hw_refine_sprepare,
312d5ac70f0Sopenharmony_ci					  snd_pcm_alaw_hw_refine_schange,
313d5ac70f0Sopenharmony_ci					  snd_pcm_generic_hw_params);
314d5ac70f0Sopenharmony_ci	if (err < 0)
315d5ac70f0Sopenharmony_ci		return err;
316d5ac70f0Sopenharmony_ci
317d5ac70f0Sopenharmony_ci	err = INTERNAL(snd_pcm_hw_params_get_format)(params, &format);
318d5ac70f0Sopenharmony_ci	if (err < 0)
319d5ac70f0Sopenharmony_ci		return err;
320d5ac70f0Sopenharmony_ci
321d5ac70f0Sopenharmony_ci	if (pcm->stream == SND_PCM_STREAM_PLAYBACK) {
322d5ac70f0Sopenharmony_ci		if (alaw->sformat == SND_PCM_FORMAT_A_LAW) {
323d5ac70f0Sopenharmony_ci			alaw->getput_idx = snd_pcm_linear_get_index(format, SND_PCM_FORMAT_S16);
324d5ac70f0Sopenharmony_ci			alaw->func = snd_pcm_alaw_encode;
325d5ac70f0Sopenharmony_ci		} else {
326d5ac70f0Sopenharmony_ci			alaw->getput_idx = snd_pcm_linear_put_index(SND_PCM_FORMAT_S16, alaw->sformat);
327d5ac70f0Sopenharmony_ci			alaw->func = snd_pcm_alaw_decode;
328d5ac70f0Sopenharmony_ci		}
329d5ac70f0Sopenharmony_ci	} else {
330d5ac70f0Sopenharmony_ci		if (alaw->sformat == SND_PCM_FORMAT_A_LAW) {
331d5ac70f0Sopenharmony_ci			alaw->getput_idx = snd_pcm_linear_put_index(SND_PCM_FORMAT_S16, format);
332d5ac70f0Sopenharmony_ci			alaw->func = snd_pcm_alaw_decode;
333d5ac70f0Sopenharmony_ci		} else {
334d5ac70f0Sopenharmony_ci			alaw->getput_idx = snd_pcm_linear_get_index(alaw->sformat, SND_PCM_FORMAT_S16);
335d5ac70f0Sopenharmony_ci			alaw->func = snd_pcm_alaw_encode;
336d5ac70f0Sopenharmony_ci		}
337d5ac70f0Sopenharmony_ci	}
338d5ac70f0Sopenharmony_ci	return 0;
339d5ac70f0Sopenharmony_ci}
340d5ac70f0Sopenharmony_ci
341d5ac70f0Sopenharmony_cistatic snd_pcm_uframes_t
342d5ac70f0Sopenharmony_cisnd_pcm_alaw_write_areas(snd_pcm_t *pcm,
343d5ac70f0Sopenharmony_ci			 const snd_pcm_channel_area_t *areas,
344d5ac70f0Sopenharmony_ci			 snd_pcm_uframes_t offset,
345d5ac70f0Sopenharmony_ci			 snd_pcm_uframes_t size,
346d5ac70f0Sopenharmony_ci			 const snd_pcm_channel_area_t *slave_areas,
347d5ac70f0Sopenharmony_ci			 snd_pcm_uframes_t slave_offset,
348d5ac70f0Sopenharmony_ci			 snd_pcm_uframes_t *slave_sizep)
349d5ac70f0Sopenharmony_ci{
350d5ac70f0Sopenharmony_ci	snd_pcm_alaw_t *alaw = pcm->private_data;
351d5ac70f0Sopenharmony_ci	if (size > *slave_sizep)
352d5ac70f0Sopenharmony_ci		size = *slave_sizep;
353d5ac70f0Sopenharmony_ci	alaw->func(slave_areas, slave_offset,
354d5ac70f0Sopenharmony_ci		   areas, offset,
355d5ac70f0Sopenharmony_ci		   pcm->channels, size,
356d5ac70f0Sopenharmony_ci		   alaw->getput_idx);
357d5ac70f0Sopenharmony_ci	*slave_sizep = size;
358d5ac70f0Sopenharmony_ci	return size;
359d5ac70f0Sopenharmony_ci}
360d5ac70f0Sopenharmony_ci
361d5ac70f0Sopenharmony_cistatic snd_pcm_uframes_t
362d5ac70f0Sopenharmony_cisnd_pcm_alaw_read_areas(snd_pcm_t *pcm,
363d5ac70f0Sopenharmony_ci			const snd_pcm_channel_area_t *areas,
364d5ac70f0Sopenharmony_ci			snd_pcm_uframes_t offset,
365d5ac70f0Sopenharmony_ci			snd_pcm_uframes_t size,
366d5ac70f0Sopenharmony_ci			const snd_pcm_channel_area_t *slave_areas,
367d5ac70f0Sopenharmony_ci			snd_pcm_uframes_t slave_offset,
368d5ac70f0Sopenharmony_ci			snd_pcm_uframes_t *slave_sizep)
369d5ac70f0Sopenharmony_ci{
370d5ac70f0Sopenharmony_ci	snd_pcm_alaw_t *alaw = pcm->private_data;
371d5ac70f0Sopenharmony_ci	if (size > *slave_sizep)
372d5ac70f0Sopenharmony_ci		size = *slave_sizep;
373d5ac70f0Sopenharmony_ci	alaw->func(areas, offset,
374d5ac70f0Sopenharmony_ci		   slave_areas, slave_offset,
375d5ac70f0Sopenharmony_ci		   pcm->channels, size,
376d5ac70f0Sopenharmony_ci		   alaw->getput_idx);
377d5ac70f0Sopenharmony_ci	*slave_sizep = size;
378d5ac70f0Sopenharmony_ci	return size;
379d5ac70f0Sopenharmony_ci}
380d5ac70f0Sopenharmony_ci
381d5ac70f0Sopenharmony_cistatic void snd_pcm_alaw_dump(snd_pcm_t *pcm, snd_output_t *out)
382d5ac70f0Sopenharmony_ci{
383d5ac70f0Sopenharmony_ci	snd_pcm_alaw_t *alaw = pcm->private_data;
384d5ac70f0Sopenharmony_ci	snd_output_printf(out, "A-Law conversion PCM (%s)\n",
385d5ac70f0Sopenharmony_ci		snd_pcm_format_name(alaw->sformat));
386d5ac70f0Sopenharmony_ci	if (pcm->setup) {
387d5ac70f0Sopenharmony_ci		snd_output_printf(out, "Its setup is:\n");
388d5ac70f0Sopenharmony_ci		snd_pcm_dump_setup(pcm, out);
389d5ac70f0Sopenharmony_ci	}
390d5ac70f0Sopenharmony_ci	snd_output_printf(out, "Slave: ");
391d5ac70f0Sopenharmony_ci	snd_pcm_dump(alaw->plug.gen.slave, out);
392d5ac70f0Sopenharmony_ci}
393d5ac70f0Sopenharmony_ci
394d5ac70f0Sopenharmony_cistatic const snd_pcm_ops_t snd_pcm_alaw_ops = {
395d5ac70f0Sopenharmony_ci	.close = snd_pcm_generic_close,
396d5ac70f0Sopenharmony_ci	.info = snd_pcm_generic_info,
397d5ac70f0Sopenharmony_ci	.hw_refine = snd_pcm_alaw_hw_refine,
398d5ac70f0Sopenharmony_ci	.hw_params = snd_pcm_alaw_hw_params,
399d5ac70f0Sopenharmony_ci	.hw_free = snd_pcm_generic_hw_free,
400d5ac70f0Sopenharmony_ci	.sw_params = snd_pcm_generic_sw_params,
401d5ac70f0Sopenharmony_ci	.channel_info = snd_pcm_generic_channel_info,
402d5ac70f0Sopenharmony_ci	.dump = snd_pcm_alaw_dump,
403d5ac70f0Sopenharmony_ci	.nonblock = snd_pcm_generic_nonblock,
404d5ac70f0Sopenharmony_ci	.async = snd_pcm_generic_async,
405d5ac70f0Sopenharmony_ci	.mmap = snd_pcm_generic_mmap,
406d5ac70f0Sopenharmony_ci	.munmap = snd_pcm_generic_munmap,
407d5ac70f0Sopenharmony_ci	.query_chmaps = snd_pcm_generic_query_chmaps,
408d5ac70f0Sopenharmony_ci	.get_chmap = snd_pcm_generic_get_chmap,
409d5ac70f0Sopenharmony_ci	.set_chmap = snd_pcm_generic_set_chmap,
410d5ac70f0Sopenharmony_ci};
411d5ac70f0Sopenharmony_ci
412d5ac70f0Sopenharmony_ci/**
413d5ac70f0Sopenharmony_ci * \brief Creates a new A-Law conversion PCM
414d5ac70f0Sopenharmony_ci * \param pcmp Returns created PCM handle
415d5ac70f0Sopenharmony_ci * \param name Name of PCM
416d5ac70f0Sopenharmony_ci * \param sformat Slave (destination) format
417d5ac70f0Sopenharmony_ci * \param slave Slave PCM handle
418d5ac70f0Sopenharmony_ci * \param close_slave When set, the slave PCM handle is closed with copy PCM
419d5ac70f0Sopenharmony_ci * \retval zero on success otherwise a negative error code
420d5ac70f0Sopenharmony_ci * \warning Using of this function might be dangerous in the sense
421d5ac70f0Sopenharmony_ci *          of compatibility reasons. The prototype might be freely
422d5ac70f0Sopenharmony_ci *          changed in future.
423d5ac70f0Sopenharmony_ci */
424d5ac70f0Sopenharmony_ciint snd_pcm_alaw_open(snd_pcm_t **pcmp, const char *name, snd_pcm_format_t sformat, snd_pcm_t *slave, int close_slave)
425d5ac70f0Sopenharmony_ci{
426d5ac70f0Sopenharmony_ci	snd_pcm_t *pcm;
427d5ac70f0Sopenharmony_ci	snd_pcm_alaw_t *alaw;
428d5ac70f0Sopenharmony_ci	int err;
429d5ac70f0Sopenharmony_ci	assert(pcmp && slave);
430d5ac70f0Sopenharmony_ci	if (snd_pcm_format_linear(sformat) != 1 &&
431d5ac70f0Sopenharmony_ci	    sformat != SND_PCM_FORMAT_A_LAW)
432d5ac70f0Sopenharmony_ci		return -EINVAL;
433d5ac70f0Sopenharmony_ci	alaw = calloc(1, sizeof(snd_pcm_alaw_t));
434d5ac70f0Sopenharmony_ci	if (!alaw) {
435d5ac70f0Sopenharmony_ci		return -ENOMEM;
436d5ac70f0Sopenharmony_ci	}
437d5ac70f0Sopenharmony_ci	snd_pcm_plugin_init(&alaw->plug);
438d5ac70f0Sopenharmony_ci	alaw->sformat = sformat;
439d5ac70f0Sopenharmony_ci	alaw->plug.read = snd_pcm_alaw_read_areas;
440d5ac70f0Sopenharmony_ci	alaw->plug.write = snd_pcm_alaw_write_areas;
441d5ac70f0Sopenharmony_ci	alaw->plug.undo_read = snd_pcm_plugin_undo_read_generic;
442d5ac70f0Sopenharmony_ci	alaw->plug.undo_write = snd_pcm_plugin_undo_write_generic;
443d5ac70f0Sopenharmony_ci	alaw->plug.gen.slave = slave;
444d5ac70f0Sopenharmony_ci	alaw->plug.gen.close_slave = close_slave;
445d5ac70f0Sopenharmony_ci
446d5ac70f0Sopenharmony_ci	err = snd_pcm_new(&pcm, SND_PCM_TYPE_ALAW, name, slave->stream, slave->mode);
447d5ac70f0Sopenharmony_ci	if (err < 0) {
448d5ac70f0Sopenharmony_ci		free(alaw);
449d5ac70f0Sopenharmony_ci		return err;
450d5ac70f0Sopenharmony_ci	}
451d5ac70f0Sopenharmony_ci	pcm->ops = &snd_pcm_alaw_ops;
452d5ac70f0Sopenharmony_ci	pcm->fast_ops = &snd_pcm_plugin_fast_ops;
453d5ac70f0Sopenharmony_ci	pcm->private_data = alaw;
454d5ac70f0Sopenharmony_ci	pcm->poll_fd = slave->poll_fd;
455d5ac70f0Sopenharmony_ci	pcm->poll_events = slave->poll_events;
456d5ac70f0Sopenharmony_ci	pcm->tstamp_type = slave->tstamp_type;
457d5ac70f0Sopenharmony_ci	snd_pcm_set_hw_ptr(pcm, &alaw->plug.hw_ptr, -1, 0);
458d5ac70f0Sopenharmony_ci	snd_pcm_set_appl_ptr(pcm, &alaw->plug.appl_ptr, -1, 0);
459d5ac70f0Sopenharmony_ci	*pcmp = pcm;
460d5ac70f0Sopenharmony_ci
461d5ac70f0Sopenharmony_ci	return 0;
462d5ac70f0Sopenharmony_ci}
463d5ac70f0Sopenharmony_ci
464d5ac70f0Sopenharmony_ci/*! \page pcm_plugins
465d5ac70f0Sopenharmony_ci
466d5ac70f0Sopenharmony_ci\section pcm_plugins_alaw Plugin: A-Law
467d5ac70f0Sopenharmony_ci
468d5ac70f0Sopenharmony_ciThis plugin converts A-Law samples to linear or linear to A-Law samples
469d5ac70f0Sopenharmony_cifrom master A-Law conversion PCM to given slave PCM. The channel count,
470d5ac70f0Sopenharmony_ciformat and rate must match for both of them.
471d5ac70f0Sopenharmony_ci
472d5ac70f0Sopenharmony_ci\code
473d5ac70f0Sopenharmony_cipcm.name {
474d5ac70f0Sopenharmony_ci        type alaw               # A-Law conversion PCM
475d5ac70f0Sopenharmony_ci        slave STR               # Slave name
476d5ac70f0Sopenharmony_ci        # or
477d5ac70f0Sopenharmony_ci        slave {                 # Slave definition
478d5ac70f0Sopenharmony_ci                pcm STR         # Slave PCM name
479d5ac70f0Sopenharmony_ci                # or
480d5ac70f0Sopenharmony_ci                pcm { }         # Slave PCM definition
481d5ac70f0Sopenharmony_ci                format STR      # Slave format
482d5ac70f0Sopenharmony_ci        }
483d5ac70f0Sopenharmony_ci}
484d5ac70f0Sopenharmony_ci\endcode
485d5ac70f0Sopenharmony_ci
486d5ac70f0Sopenharmony_ci\subsection pcm_plugins_alaw_funcref Function reference
487d5ac70f0Sopenharmony_ci
488d5ac70f0Sopenharmony_ci<UL>
489d5ac70f0Sopenharmony_ci  <LI>snd_pcm_alaw_open()
490d5ac70f0Sopenharmony_ci  <LI>_snd_pcm_alaw_open()
491d5ac70f0Sopenharmony_ci</UL>
492d5ac70f0Sopenharmony_ci
493d5ac70f0Sopenharmony_ci*/
494d5ac70f0Sopenharmony_ci
495d5ac70f0Sopenharmony_ci/**
496d5ac70f0Sopenharmony_ci * \brief Creates a new A-Law conversion PCM
497d5ac70f0Sopenharmony_ci * \param pcmp Returns created PCM handle
498d5ac70f0Sopenharmony_ci * \param name Name of PCM
499d5ac70f0Sopenharmony_ci * \param root Root configuration node
500d5ac70f0Sopenharmony_ci * \param conf Configuration node with copy PCM description
501d5ac70f0Sopenharmony_ci * \param stream Stream type
502d5ac70f0Sopenharmony_ci * \param mode Stream mode
503d5ac70f0Sopenharmony_ci * \retval zero on success otherwise a negative error code
504d5ac70f0Sopenharmony_ci * \warning Using of this function might be dangerous in the sense
505d5ac70f0Sopenharmony_ci *          of compatibility reasons. The prototype might be freely
506d5ac70f0Sopenharmony_ci *          changed in future.
507d5ac70f0Sopenharmony_ci */
508d5ac70f0Sopenharmony_ciint _snd_pcm_alaw_open(snd_pcm_t **pcmp, const char *name,
509d5ac70f0Sopenharmony_ci		       snd_config_t *root, snd_config_t *conf,
510d5ac70f0Sopenharmony_ci		       snd_pcm_stream_t stream, int mode)
511d5ac70f0Sopenharmony_ci{
512d5ac70f0Sopenharmony_ci	snd_config_iterator_t i, next;
513d5ac70f0Sopenharmony_ci	int err;
514d5ac70f0Sopenharmony_ci	snd_pcm_t *spcm;
515d5ac70f0Sopenharmony_ci	snd_config_t *slave = NULL, *sconf;
516d5ac70f0Sopenharmony_ci	snd_pcm_format_t sformat;
517d5ac70f0Sopenharmony_ci	snd_config_for_each(i, next, conf) {
518d5ac70f0Sopenharmony_ci		snd_config_t *n = snd_config_iterator_entry(i);
519d5ac70f0Sopenharmony_ci		const char *id;
520d5ac70f0Sopenharmony_ci		if (snd_config_get_id(n, &id) < 0)
521d5ac70f0Sopenharmony_ci			continue;
522d5ac70f0Sopenharmony_ci		if (snd_pcm_conf_generic_id(id))
523d5ac70f0Sopenharmony_ci			continue;
524d5ac70f0Sopenharmony_ci		if (strcmp(id, "slave") == 0) {
525d5ac70f0Sopenharmony_ci			slave = n;
526d5ac70f0Sopenharmony_ci			continue;
527d5ac70f0Sopenharmony_ci		}
528d5ac70f0Sopenharmony_ci		SNDERR("Unknown field %s", id);
529d5ac70f0Sopenharmony_ci		return -EINVAL;
530d5ac70f0Sopenharmony_ci	}
531d5ac70f0Sopenharmony_ci	if (!slave) {
532d5ac70f0Sopenharmony_ci		SNDERR("slave is not defined");
533d5ac70f0Sopenharmony_ci		return -EINVAL;
534d5ac70f0Sopenharmony_ci	}
535d5ac70f0Sopenharmony_ci	err = snd_pcm_slave_conf(root, slave, &sconf, 1,
536d5ac70f0Sopenharmony_ci				 SND_PCM_HW_PARAM_FORMAT, SCONF_MANDATORY, &sformat);
537d5ac70f0Sopenharmony_ci	if (err < 0)
538d5ac70f0Sopenharmony_ci		return err;
539d5ac70f0Sopenharmony_ci	if (snd_pcm_format_linear(sformat) != 1 &&
540d5ac70f0Sopenharmony_ci	    sformat != SND_PCM_FORMAT_A_LAW) {
541d5ac70f0Sopenharmony_ci	    	snd_config_delete(sconf);
542d5ac70f0Sopenharmony_ci		SNDERR("invalid slave format");
543d5ac70f0Sopenharmony_ci		return -EINVAL;
544d5ac70f0Sopenharmony_ci	}
545d5ac70f0Sopenharmony_ci	err = snd_pcm_open_slave(&spcm, root, sconf, stream, mode, conf);
546d5ac70f0Sopenharmony_ci	snd_config_delete(sconf);
547d5ac70f0Sopenharmony_ci	if (err < 0)
548d5ac70f0Sopenharmony_ci		return err;
549d5ac70f0Sopenharmony_ci	err = snd_pcm_alaw_open(pcmp, name, sformat, spcm, 1);
550d5ac70f0Sopenharmony_ci	if (err < 0)
551d5ac70f0Sopenharmony_ci		snd_pcm_close(spcm);
552d5ac70f0Sopenharmony_ci	return err;
553d5ac70f0Sopenharmony_ci}
554d5ac70f0Sopenharmony_ci#ifndef DOC_HIDDEN
555d5ac70f0Sopenharmony_ciSND_DLSYM_BUILD_VERSION(_snd_pcm_alaw_open, SND_PCM_DLSYM_VERSION);
556d5ac70f0Sopenharmony_ci#endif
557