18c2ecf20Sopenharmony_ci/* 28c2ecf20Sopenharmony_ci * Linear conversion Plug-In 38c2ecf20Sopenharmony_ci * Copyright (c) 1999 by Jaroslav Kysela <perex@perex.cz>, 48c2ecf20Sopenharmony_ci * Abramo Bagnara <abramo@alsa-project.org> 58c2ecf20Sopenharmony_ci * 68c2ecf20Sopenharmony_ci * 78c2ecf20Sopenharmony_ci * This library is free software; you can redistribute it and/or modify 88c2ecf20Sopenharmony_ci * it under the terms of the GNU Library General Public License as 98c2ecf20Sopenharmony_ci * published by the Free Software Foundation; either version 2 of 108c2ecf20Sopenharmony_ci * the License, or (at your option) any later version. 118c2ecf20Sopenharmony_ci * 128c2ecf20Sopenharmony_ci * This program is distributed in the hope that it will be useful, 138c2ecf20Sopenharmony_ci * but WITHOUT ANY WARRANTY; without even the implied warranty of 148c2ecf20Sopenharmony_ci * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 158c2ecf20Sopenharmony_ci * GNU Library General Public License for more details. 168c2ecf20Sopenharmony_ci * 178c2ecf20Sopenharmony_ci * You should have received a copy of the GNU Library General Public 188c2ecf20Sopenharmony_ci * License along with this library; if not, write to the Free Software 198c2ecf20Sopenharmony_ci * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA 208c2ecf20Sopenharmony_ci * 218c2ecf20Sopenharmony_ci */ 228c2ecf20Sopenharmony_ci 238c2ecf20Sopenharmony_ci#include <linux/time.h> 248c2ecf20Sopenharmony_ci#include <sound/core.h> 258c2ecf20Sopenharmony_ci#include <sound/pcm.h> 268c2ecf20Sopenharmony_ci#include "pcm_plugin.h" 278c2ecf20Sopenharmony_ci 288c2ecf20Sopenharmony_ci/* 298c2ecf20Sopenharmony_ci * Basic linear conversion plugin 308c2ecf20Sopenharmony_ci */ 318c2ecf20Sopenharmony_ci 328c2ecf20Sopenharmony_cistruct linear_priv { 338c2ecf20Sopenharmony_ci int cvt_endian; /* need endian conversion? */ 348c2ecf20Sopenharmony_ci unsigned int src_ofs; /* byte offset in source format */ 358c2ecf20Sopenharmony_ci unsigned int dst_ofs; /* byte soffset in destination format */ 368c2ecf20Sopenharmony_ci unsigned int copy_ofs; /* byte offset in temporary u32 data */ 378c2ecf20Sopenharmony_ci unsigned int dst_bytes; /* byte size of destination format */ 388c2ecf20Sopenharmony_ci unsigned int copy_bytes; /* bytes to copy per conversion */ 398c2ecf20Sopenharmony_ci unsigned int flip; /* MSB flip for signeness, done after endian conv */ 408c2ecf20Sopenharmony_ci}; 418c2ecf20Sopenharmony_ci 428c2ecf20Sopenharmony_cistatic inline void do_convert(struct linear_priv *data, 438c2ecf20Sopenharmony_ci unsigned char *dst, unsigned char *src) 448c2ecf20Sopenharmony_ci{ 458c2ecf20Sopenharmony_ci unsigned int tmp = 0; 468c2ecf20Sopenharmony_ci unsigned char *p = (unsigned char *)&tmp; 478c2ecf20Sopenharmony_ci 488c2ecf20Sopenharmony_ci memcpy(p + data->copy_ofs, src + data->src_ofs, data->copy_bytes); 498c2ecf20Sopenharmony_ci if (data->cvt_endian) 508c2ecf20Sopenharmony_ci tmp = swab32(tmp); 518c2ecf20Sopenharmony_ci tmp ^= data->flip; 528c2ecf20Sopenharmony_ci memcpy(dst, p + data->dst_ofs, data->dst_bytes); 538c2ecf20Sopenharmony_ci} 548c2ecf20Sopenharmony_ci 558c2ecf20Sopenharmony_cistatic void convert(struct snd_pcm_plugin *plugin, 568c2ecf20Sopenharmony_ci const struct snd_pcm_plugin_channel *src_channels, 578c2ecf20Sopenharmony_ci struct snd_pcm_plugin_channel *dst_channels, 588c2ecf20Sopenharmony_ci snd_pcm_uframes_t frames) 598c2ecf20Sopenharmony_ci{ 608c2ecf20Sopenharmony_ci struct linear_priv *data = (struct linear_priv *)plugin->extra_data; 618c2ecf20Sopenharmony_ci int channel; 628c2ecf20Sopenharmony_ci int nchannels = plugin->src_format.channels; 638c2ecf20Sopenharmony_ci for (channel = 0; channel < nchannels; ++channel) { 648c2ecf20Sopenharmony_ci char *src; 658c2ecf20Sopenharmony_ci char *dst; 668c2ecf20Sopenharmony_ci int src_step, dst_step; 678c2ecf20Sopenharmony_ci snd_pcm_uframes_t frames1; 688c2ecf20Sopenharmony_ci if (!src_channels[channel].enabled) { 698c2ecf20Sopenharmony_ci if (dst_channels[channel].wanted) 708c2ecf20Sopenharmony_ci snd_pcm_area_silence(&dst_channels[channel].area, 0, frames, plugin->dst_format.format); 718c2ecf20Sopenharmony_ci dst_channels[channel].enabled = 0; 728c2ecf20Sopenharmony_ci continue; 738c2ecf20Sopenharmony_ci } 748c2ecf20Sopenharmony_ci dst_channels[channel].enabled = 1; 758c2ecf20Sopenharmony_ci src = src_channels[channel].area.addr + src_channels[channel].area.first / 8; 768c2ecf20Sopenharmony_ci dst = dst_channels[channel].area.addr + dst_channels[channel].area.first / 8; 778c2ecf20Sopenharmony_ci src_step = src_channels[channel].area.step / 8; 788c2ecf20Sopenharmony_ci dst_step = dst_channels[channel].area.step / 8; 798c2ecf20Sopenharmony_ci frames1 = frames; 808c2ecf20Sopenharmony_ci while (frames1-- > 0) { 818c2ecf20Sopenharmony_ci do_convert(data, dst, src); 828c2ecf20Sopenharmony_ci src += src_step; 838c2ecf20Sopenharmony_ci dst += dst_step; 848c2ecf20Sopenharmony_ci } 858c2ecf20Sopenharmony_ci } 868c2ecf20Sopenharmony_ci} 878c2ecf20Sopenharmony_ci 888c2ecf20Sopenharmony_cistatic snd_pcm_sframes_t linear_transfer(struct snd_pcm_plugin *plugin, 898c2ecf20Sopenharmony_ci const struct snd_pcm_plugin_channel *src_channels, 908c2ecf20Sopenharmony_ci struct snd_pcm_plugin_channel *dst_channels, 918c2ecf20Sopenharmony_ci snd_pcm_uframes_t frames) 928c2ecf20Sopenharmony_ci{ 938c2ecf20Sopenharmony_ci if (snd_BUG_ON(!plugin || !src_channels || !dst_channels)) 948c2ecf20Sopenharmony_ci return -ENXIO; 958c2ecf20Sopenharmony_ci if (frames == 0) 968c2ecf20Sopenharmony_ci return 0; 978c2ecf20Sopenharmony_ci#ifdef CONFIG_SND_DEBUG 988c2ecf20Sopenharmony_ci { 998c2ecf20Sopenharmony_ci unsigned int channel; 1008c2ecf20Sopenharmony_ci for (channel = 0; channel < plugin->src_format.channels; channel++) { 1018c2ecf20Sopenharmony_ci if (snd_BUG_ON(src_channels[channel].area.first % 8 || 1028c2ecf20Sopenharmony_ci src_channels[channel].area.step % 8)) 1038c2ecf20Sopenharmony_ci return -ENXIO; 1048c2ecf20Sopenharmony_ci if (snd_BUG_ON(dst_channels[channel].area.first % 8 || 1058c2ecf20Sopenharmony_ci dst_channels[channel].area.step % 8)) 1068c2ecf20Sopenharmony_ci return -ENXIO; 1078c2ecf20Sopenharmony_ci } 1088c2ecf20Sopenharmony_ci } 1098c2ecf20Sopenharmony_ci#endif 1108c2ecf20Sopenharmony_ci if (frames > dst_channels[0].frames) 1118c2ecf20Sopenharmony_ci frames = dst_channels[0].frames; 1128c2ecf20Sopenharmony_ci convert(plugin, src_channels, dst_channels, frames); 1138c2ecf20Sopenharmony_ci return frames; 1148c2ecf20Sopenharmony_ci} 1158c2ecf20Sopenharmony_ci 1168c2ecf20Sopenharmony_cistatic void init_data(struct linear_priv *data, 1178c2ecf20Sopenharmony_ci snd_pcm_format_t src_format, snd_pcm_format_t dst_format) 1188c2ecf20Sopenharmony_ci{ 1198c2ecf20Sopenharmony_ci int src_le, dst_le, src_bytes, dst_bytes; 1208c2ecf20Sopenharmony_ci 1218c2ecf20Sopenharmony_ci src_bytes = snd_pcm_format_width(src_format) / 8; 1228c2ecf20Sopenharmony_ci dst_bytes = snd_pcm_format_width(dst_format) / 8; 1238c2ecf20Sopenharmony_ci src_le = snd_pcm_format_little_endian(src_format) > 0; 1248c2ecf20Sopenharmony_ci dst_le = snd_pcm_format_little_endian(dst_format) > 0; 1258c2ecf20Sopenharmony_ci 1268c2ecf20Sopenharmony_ci data->dst_bytes = dst_bytes; 1278c2ecf20Sopenharmony_ci data->cvt_endian = src_le != dst_le; 1288c2ecf20Sopenharmony_ci data->copy_bytes = src_bytes < dst_bytes ? src_bytes : dst_bytes; 1298c2ecf20Sopenharmony_ci if (src_le) { 1308c2ecf20Sopenharmony_ci data->copy_ofs = 4 - data->copy_bytes; 1318c2ecf20Sopenharmony_ci data->src_ofs = src_bytes - data->copy_bytes; 1328c2ecf20Sopenharmony_ci } else 1338c2ecf20Sopenharmony_ci data->src_ofs = snd_pcm_format_physical_width(src_format) / 8 - 1348c2ecf20Sopenharmony_ci src_bytes; 1358c2ecf20Sopenharmony_ci if (dst_le) 1368c2ecf20Sopenharmony_ci data->dst_ofs = 4 - data->dst_bytes; 1378c2ecf20Sopenharmony_ci else 1388c2ecf20Sopenharmony_ci data->dst_ofs = snd_pcm_format_physical_width(dst_format) / 8 - 1398c2ecf20Sopenharmony_ci dst_bytes; 1408c2ecf20Sopenharmony_ci if (snd_pcm_format_signed(src_format) != 1418c2ecf20Sopenharmony_ci snd_pcm_format_signed(dst_format)) { 1428c2ecf20Sopenharmony_ci if (dst_le) 1438c2ecf20Sopenharmony_ci data->flip = (__force u32)cpu_to_le32(0x80000000); 1448c2ecf20Sopenharmony_ci else 1458c2ecf20Sopenharmony_ci data->flip = (__force u32)cpu_to_be32(0x80000000); 1468c2ecf20Sopenharmony_ci } 1478c2ecf20Sopenharmony_ci} 1488c2ecf20Sopenharmony_ci 1498c2ecf20Sopenharmony_ciint snd_pcm_plugin_build_linear(struct snd_pcm_substream *plug, 1508c2ecf20Sopenharmony_ci struct snd_pcm_plugin_format *src_format, 1518c2ecf20Sopenharmony_ci struct snd_pcm_plugin_format *dst_format, 1528c2ecf20Sopenharmony_ci struct snd_pcm_plugin **r_plugin) 1538c2ecf20Sopenharmony_ci{ 1548c2ecf20Sopenharmony_ci int err; 1558c2ecf20Sopenharmony_ci struct linear_priv *data; 1568c2ecf20Sopenharmony_ci struct snd_pcm_plugin *plugin; 1578c2ecf20Sopenharmony_ci 1588c2ecf20Sopenharmony_ci if (snd_BUG_ON(!r_plugin)) 1598c2ecf20Sopenharmony_ci return -ENXIO; 1608c2ecf20Sopenharmony_ci *r_plugin = NULL; 1618c2ecf20Sopenharmony_ci 1628c2ecf20Sopenharmony_ci if (snd_BUG_ON(src_format->rate != dst_format->rate)) 1638c2ecf20Sopenharmony_ci return -ENXIO; 1648c2ecf20Sopenharmony_ci if (snd_BUG_ON(src_format->channels != dst_format->channels)) 1658c2ecf20Sopenharmony_ci return -ENXIO; 1668c2ecf20Sopenharmony_ci if (snd_BUG_ON(!snd_pcm_format_linear(src_format->format) || 1678c2ecf20Sopenharmony_ci !snd_pcm_format_linear(dst_format->format))) 1688c2ecf20Sopenharmony_ci return -ENXIO; 1698c2ecf20Sopenharmony_ci 1708c2ecf20Sopenharmony_ci err = snd_pcm_plugin_build(plug, "linear format conversion", 1718c2ecf20Sopenharmony_ci src_format, dst_format, 1728c2ecf20Sopenharmony_ci sizeof(struct linear_priv), &plugin); 1738c2ecf20Sopenharmony_ci if (err < 0) 1748c2ecf20Sopenharmony_ci return err; 1758c2ecf20Sopenharmony_ci data = (struct linear_priv *)plugin->extra_data; 1768c2ecf20Sopenharmony_ci init_data(data, src_format->format, dst_format->format); 1778c2ecf20Sopenharmony_ci plugin->transfer = linear_transfer; 1788c2ecf20Sopenharmony_ci *r_plugin = plugin; 1798c2ecf20Sopenharmony_ci return 0; 1808c2ecf20Sopenharmony_ci} 181