/*++
* linux/sound/soc/wmt/wmt_swmixer.c
* WonderMedia I2S audio driver for ALSA
*
* Copyright c 2010 WonderMedia Technologies, Inc.
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*
* WonderMedia Technologies, Inc.
* 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
--*/
#include "wmt_swmixer.h"
#include
void wmt_sw_u2s(int fmt, char *buffer, unsigned int chunksize)
{
unsigned int index;
if (fmt == SNDRV_PCM_FORMAT_U8) {
for (index = 0; index < chunksize; ++index)
*(buffer + index) ^= 0x80;
}
}
void wmt_pcm_fmt_trans(int fmt, int channel, char *src_buf, char *dst_buf, unsigned int chunksize)
{
unsigned int index = 0;
float_data_t f_data;
unsigned short data;
/* always convert to 2ch, s16le (4 bytes) */
if ((fmt == SNDRV_PCM_FORMAT_S16_LE) && (channel == 1)) {
/* transfer from 1ch s16le(2 bytes) to 2ch s16le(4 bytes) */
for (index = 0; index < (chunksize / 2); ++index) {
*((unsigned int *)dst_buf + index) = (*((unsigned short *)src_buf + index) << 16 |
*((unsigned short *)src_buf + index));
}
}
else if ((fmt == SNDRV_PCM_FORMAT_U8) && (channel == 1)) {
/* transfer from 1ch U8(1 bytes) to 2ch s16le(4 bytes) */
for (index = 0; index < chunksize; ++index) {
/* padding zero to byte 0 & byte 2 */
*((unsigned int *)dst_buf + index) = (*((unsigned char *)src_buf + index) << 24 |
*((unsigned char *)src_buf + index) << 8);
}
}
else if ((fmt == SNDRV_PCM_FORMAT_U8) && (channel == 2)) {
/* transfer from 2ch U8(2 bytes) to 2ch s16le(4 bytes) */
for (index = 0; index < chunksize; ++index) {
/* padding zero to byte 0 */
*((unsigned short *)dst_buf + index) = *((unsigned char *)src_buf + index) << 8;
}
}
else if ((fmt == SNDRV_PCM_FORMAT_FLOAT) && (channel == 2)) {
/* transfer from 2ch float(8 bytes) to 2ch s16le(4 bytes) */
for (index = 0; index < (chunksize / 4); ++index) {
f_data = *((float_data_t *)src_buf + index);
if (!f_data.sign) {
data = (f_data.frac + 0x800000) >> (8 - (f_data.exp - 127));
}
else {
data = ~((f_data.frac + 0x800000) >> (8 - (f_data.exp - 127))) + 1;
}
*((unsigned short *)dst_buf + index) = data;
}
}
else if ((fmt == SNDRV_PCM_FORMAT_FLOAT) && (channel == 1)) {
/* transfer from 1ch float(4 bytes) to 2ch s16le(4 bytes) */
for (index = 0; index < (chunksize / 4); ++index) {
f_data = *((float_data_t *)src_buf + index);
if (!f_data.sign) {
data = (unsigned short)((f_data.frac + 0x800000) >> (8 - (f_data.exp - 127)));
}
else {
data = (unsigned short)(~((f_data.frac + 0x800000) >> (8 - (f_data.exp - 127))) + 1);
}
*((unsigned int *)dst_buf + index) = (data << 16) | data;
}
}
}