pinebuds/platform/hal/hal_i2s_tdm.c

316 lines
11 KiB
C

/***************************************************************************
*
* Copyright 2015-2019 BES.
* All rights reserved. All unpublished rights reserved.
*
* No part of this work may be used or reproduced in any form or by any
* means, or stored in a database or retrieval system, without prior written
* permission of BES.
*
* Use of this work is governed by a license granted by BES.
* This work contains confidential and proprietary information of
* BES. which is protected by copyright, trade secret,
* trademark and other intellectual property rights.
*
****************************************************************************/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "hal_i2s.h"
#include "hal_trace.h"
#include "hal_dma.h"
#include "hal_i2s_tdm.h"
#if 0
#define I2S_TDM_TRACE TRACE
#define I2S_TDM_DUMP8 DUMP8
#else
#define I2S_TDM_TRACE(n, str, ...)
#define I2S_TDM_DUMP8(str, ...)
#endif
#define I2S_TDM_FRAME_SIZE_MAX 512
#define I2S_TDM_TX_FRAME_NUM 2
#define I2S_TDM_TX_FRAME_SIZE I2S_TDM_FRAME_SIZE_MAX/16
static struct HAL_DMA_DESC_T i2s_tdm_tx_dma_desc[HAL_I2S_ID_QTY][I2S_TDM_TX_FRAME_NUM];
static uint16_t I2S_TDM_BUF_ALIGN i2s_tdm_tx_buf[HAL_I2S_ID_QTY][I2S_TDM_TX_FRAME_NUM][I2S_TDM_TX_FRAME_SIZE];
static struct HAL_DMA_CH_CFG_T tx_dma_cfg[HAL_I2S_ID_QTY];
static struct HAL_I2S_TDM_CONFIG_T i2s_tdm_cfg[HAL_I2S_ID_QTY];
static inline bool i2s_tdm_cycles_in_arrays(uint16_t cycles)
{
if(cycles == (uint16_t)HAL_I2S_TDM_CYCLES_16
|| cycles == (uint16_t)HAL_I2S_TDM_CYCLES_32
|| cycles == (uint16_t)HAL_I2S_TDM_CYCLES_64
|| cycles == (uint16_t)HAL_I2S_TDM_CYCLES_128
|| cycles == (uint16_t)HAL_I2S_TDM_CYCLES_256
|| cycles == (uint16_t)HAL_I2S_TDM_CYCLES_512
)
return true;
else
return false;
}
static inline bool i2s_tdm_fs_cycles_in_arrays(uint16_t fs_cycles)
{
if(fs_cycles == (uint16_t)HAL_I2S_TDM_FS_CYCLES_1
|| fs_cycles == (uint16_t)HAL_TDM_FS_CYCLES_8
|| fs_cycles == (uint16_t)HAL_I2S_TDM_FS_CYCLES_16
|| fs_cycles == (uint16_t)HAL_I2S_TDM_FS_CYCLES_32
|| fs_cycles == (uint16_t)HAL_I2S_TDM_FS_CYCLES_64
|| fs_cycles == (uint16_t)HAL_I2S_TDM_FS_CYCLES_128
|| fs_cycles == (uint16_t)HAL_I2S_TDM_FS_CYCLES_256
|| fs_cycles == (uint16_t)HAL_I2S_TDM_FS_CYCLES_ONE_LESS)
return true;
else
return false;
}
static inline bool i2s_tdm_slot_cycles_in_arrays(uint8_t slot_cycles)
{
if(slot_cycles == (uint8_t)HAL_I2S_TDM_SLOT_CYCLES_16
|| slot_cycles == (uint8_t)HAL_I2S_TDM_SLOT_CYCLES_32)
return true;
else
return false;
}
static void i2s_tdm0_tx_handler(uint8_t chan, uint32_t remains, uint32_t error, struct HAL_DMA_DESC_T *lli)
{
#if 0
static int cnt = 0;
cnt++;
if (cnt % 600 == 0)
{
I2S_TDM_TRACE(4,"I2S_TDM0-TX: remains=%ld, error=%ld, cnt=%d,tx_buff_len = %d.", remains, error, cnt,sizeof(i2s_tdm_tx_buf[HAL_I2S_ID_0]));
I2S_TDM_DUMP8("0x%x,",i2s_tdm_tx_buf[HAL_I2S_ID_0],sizeof(i2s_tdm_tx_buf[HAL_I2S_ID_0]) <= 32 ? sizeof(i2s_tdm_tx_buf[HAL_I2S_ID_0]) : 32 );
}
#endif
}
static void i2s_tdm1_tx_handler(uint8_t chan, uint32_t remains, uint32_t error, struct HAL_DMA_DESC_T *lli)
{
#if 0
static int cnt = 0;
cnt++;
if (cnt % 600 == 0) {
I2S_TDM_TRACE(4,"I2S_TDM1-TX: remains=%ld, error=%ld, cnt=%d,tx_buff_len = %d.", remains, error, cnt,sizeof(i2s_tdm_tx_buf[HAL_I2S_ID_1]));
I2S_TDM_DUMP8("0x%x,",i2s_tdm_tx_buf[HAL_I2S_ID_1],sizeof(i2s_tdm_tx_buf[HAL_I2S_ID_1]) <= 32 ? sizeof(i2s_tdm_tx_buf[HAL_I2S_ID_1]) : 32 );
}
#endif
}
int32_t hal_i2s_tdm_open(enum HAL_I2S_ID_T i2s_id,enum HAL_I2S_MODE_T mode)
{
int ret;
I2S_TDM_TRACE(3,"%s: i2s_id = %d,mode = %d.", __func__, i2s_id, mode);
ASSERT(i2s_id < HAL_I2S_ID_QTY,"%s: i2s_id = %d!", __func__, i2s_id);
// i2s open playback and capture.
ret = hal_i2s_open(i2s_id, AUD_STREAM_PLAYBACK, mode);
if(ret)
{
I2S_TDM_TRACE(2,"%s: hal_i2s_open playback failed.ret = %d.", __func__, ret);
goto __func_fail;
}
ret = hal_i2s_open(i2s_id, AUD_STREAM_CAPTURE, mode);
if(ret)
{
I2S_TDM_TRACE(2,"%s: hal_i2s_open capture failed.ret = %d.", __func__, ret);
goto __func_fail;
}
I2S_TDM_TRACE(1,"%s done.", __func__);
return 0;
__func_fail:
I2S_TDM_TRACE(2,"%s failed. ret = %d.", __func__, ret);
return ret;
}
int32_t hal_i2s_tdm_setup(enum HAL_I2S_ID_T i2s_id,
uint32_t sample_rate,
struct HAL_I2S_TDM_CONFIG_T *i2s_tdm_cfg)
{
int i,j,m,n;
int ret;
struct HAL_I2S_CONFIG_T i2s_cfg;
uint16_t *buf;
uint32_t cycles;
uint32_t fs_cycles;
uint32_t slot_cycles;
uint32_t fs_remain;
I2S_TDM_TRACE(3,"%s: i2s_id = %d,sample_rate = %d.", __func__, i2s_id, sample_rate);
ASSERT(i2s_id < HAL_I2S_ID_QTY,"%s: i2s_id = %d!", __func__, i2s_id);
ASSERT(i2s_tdm_cycles_in_arrays(i2s_tdm_cfg->cycles),
"%s: cycles(%d) error!", __func__, i2s_tdm_cfg->cycles);
ASSERT(i2s_tdm_fs_cycles_in_arrays(i2s_tdm_cfg->fs_cycles),
"%s: fs_cycles(%d) error!", __func__, i2s_tdm_cfg->fs_cycles);
ASSERT(i2s_tdm_slot_cycles_in_arrays(i2s_tdm_cfg->slot_cycles),
"%s: slot_cycles(%d) error!", __func__, i2s_tdm_cfg->slot_cycles);
i2s_tdm_cfg[i2s_id] = *i2s_tdm_cfg;
cycles = i2s_tdm_cfg->cycles;
fs_cycles = i2s_tdm_cfg->fs_cycles == HAL_I2S_TDM_FS_CYCLES_ONE_LESS ?\
cycles - 1: i2s_tdm_cfg->fs_cycles;
slot_cycles = i2s_tdm_cfg->slot_cycles;
memset(&i2s_cfg, 0, sizeof(i2s_cfg));
i2s_cfg.use_dma = true;
i2s_cfg.sync_start = true;
i2s_cfg.chan_sep_buf = false;
i2s_cfg.bits = slot_cycles;
i2s_cfg.channel_num = 2;
i2s_cfg.channel_map = AUD_CHANNEL_MAP_CH0 | AUD_CHANNEL_MAP_CH1;
i2s_cfg.sample_rate = (sample_rate*(cycles/i2s_cfg.bits))/2;
ret = hal_i2s_setup_stream(i2s_id, AUD_STREAM_PLAYBACK, &i2s_cfg);
if(ret)
{
I2S_TDM_TRACE(2,"%s: playback failed.ret = %d.", __func__, ret);
goto __func_fail;
}
i2s_cfg.sync_start = false;
ret = hal_i2s_setup_stream(i2s_id, AUD_STREAM_CAPTURE, &i2s_cfg);
if(ret)
{
I2S_TDM_TRACE(1,"hal_i2s_setup_stream capture failed.ret = %d.", ret);
goto __func_fail;
}
// Set tx buffer, output signal as slave device ws.
I2S_TDM_TRACE(3,"%s: cycles = %d, fs_cycles = %d", __func__, cycles, fs_cycles);
for (i = 0; i < I2S_TDM_TX_FRAME_NUM ; i++) {
buf = (uint16_t*)(&i2s_tdm_tx_buf[i2s_id][i][0]);
for(j = 0; j < I2S_TDM_TX_FRAME_SIZE/(cycles/16); j ++){
fs_remain = fs_cycles;
for(m = 0; m < (cycles/16); m++) {
buf[j*(cycles/16) + m] = 0;
for(n = 0; n < 16; n++) {
if (fs_remain > 0) {
buf[j*(cycles/16) + m] |= (1<< n);
fs_remain --;
} else {
break;
}
}
}
}
}
I2S_TDM_TRACE(1,"%s done.", __func__);
return 0;
__func_fail:
I2S_TDM_TRACE(2,"%s failed. ret = %d.", __func__, ret);
return ret;
}
int32_t hal_i2s_tdm_start_stream(enum HAL_I2S_ID_T i2s_id)
{
uint32_t i;
int ret;
I2S_TDM_TRACE(2,"%s: i2s_id = %d.", __func__, i2s_id);
ASSERT(i2s_id < HAL_I2S_ID_QTY,"%s: i2s_id = %d!", __func__, i2s_id);
memset(&tx_dma_cfg[i2s_id], 0, sizeof(tx_dma_cfg[i2s_id]));
tx_dma_cfg[i2s_id].dst = 0; //useless
tx_dma_cfg[i2s_id].dst_bsize = HAL_DMA_BSIZE_4;
tx_dma_cfg[i2s_id].dst_periph = i2s_id == HAL_I2S_ID_0 ? HAL_AUDMA_I2S0_TX : HAL_AUDMA_I2S1_TX;
tx_dma_cfg[i2s_id].dst_width = HAL_DMA_WIDTH_HALFWORD;
tx_dma_cfg[i2s_id].handler = i2s_id == HAL_I2S_ID_0 ? i2s_tdm0_tx_handler : i2s_tdm1_tx_handler;
tx_dma_cfg[i2s_id].src_bsize = HAL_DMA_BSIZE_4;
tx_dma_cfg[i2s_id].src_tsize = (sizeof(i2s_tdm_tx_buf[i2s_id][0])/2);
tx_dma_cfg[i2s_id].src_width = HAL_DMA_WIDTH_HALFWORD;
tx_dma_cfg[i2s_id].try_burst = 1;
tx_dma_cfg[i2s_id].type = HAL_DMA_FLOW_M2P_DMA;
tx_dma_cfg[i2s_id].ch = hal_audma_get_chan(tx_dma_cfg[i2s_id].dst_periph, HAL_DMA_HIGH_PRIO);
for (i = 0; i < I2S_TDM_TX_FRAME_NUM; i++) {
tx_dma_cfg[i2s_id].src = (uint32_t)&i2s_tdm_tx_buf[i2s_id][i][0];
ret = hal_audma_init_desc(&i2s_tdm_tx_dma_desc[i2s_id][i],
&tx_dma_cfg[i2s_id],
&i2s_tdm_tx_dma_desc[i2s_id][(i + 1) % I2S_TDM_TX_FRAME_NUM],
1);
if(ret)
{
I2S_TDM_TRACE(2,"%s: hal_audma_init_desc tx failed.ret = %d.", __func__, ret);
goto __func_fail;
}
}
ret = hal_audma_sg_start(&i2s_tdm_tx_dma_desc[i2s_id][0], &tx_dma_cfg[i2s_id]);
if(ret)
{
I2S_TDM_TRACE(2,"%s: hal_audma_sg_start tx failed.ret = %d.", __func__, ret);
goto __func_fail;
}
// i2s start stream playback and capture.
ret = hal_i2s_start_stream(i2s_id, AUD_STREAM_PLAYBACK);
if(ret)
{
I2S_TDM_TRACE(2,"%s: hal_i2s_start_stream failed.ret = %d.", __func__, ret);
}
ret = hal_i2s_start_stream(i2s_id, AUD_STREAM_CAPTURE);
if(ret)
{
I2S_TDM_TRACE(1,"hal_i2s_start_stream tx failed.ret = %d.", ret);
goto __func_fail;
}
I2S_TDM_TRACE(1,"%s done.", __func__);
return ret;
__func_fail:
I2S_TDM_TRACE(2,"%s failed.ret = %d.", __func__, ret);
return ret;
}
int32_t hal_i2s_tdm_stop_stream(enum HAL_I2S_ID_T i2s_id)
{
I2S_TDM_TRACE(2,"%s: i2s_id = %d.", __func__, i2s_id);
ASSERT(i2s_id < HAL_I2S_ID_QTY,"%s: i2s_id = %d!", __func__, i2s_id);
hal_dma_stop(tx_dma_cfg[i2s_id].ch);
hal_i2s_stop_stream(i2s_id,AUD_STREAM_PLAYBACK);
hal_i2s_stop_stream(i2s_id,AUD_STREAM_CAPTURE);
I2S_TDM_TRACE(1,"%s done.", __func__);
return 0;
}
int32_t hal_i2s_tdm_close(enum HAL_I2S_ID_T i2s_id)
{
I2S_TDM_TRACE(2,"%s: i2s_id = %d.", __func__, i2s_id);
ASSERT(i2s_id < HAL_I2S_ID_QTY,"%s: i2s_id = %d!", __func__, i2s_id);
hal_i2s_close(i2s_id,AUD_STREAM_PLAYBACK);
hal_i2s_close(i2s_id,AUD_STREAM_CAPTURE);
I2S_TDM_TRACE(1,"%s done.", __func__);
return 0;
}
void hal_i2s_tdm_get_config(enum HAL_I2S_ID_T i2s_id,struct HAL_I2S_TDM_CONFIG_T *tdm_cfg)
{
I2S_TDM_TRACE(2,"%s: i2s_id = %d.", __func__, i2s_id);
ASSERT(i2s_id < HAL_I2S_ID_QTY,"%s: i2s_id = %d!", __func__, i2s_id);
*tdm_cfg = i2s_tdm_cfg[i2s_id];
}
void hal_i2s_tdm_set_config(enum HAL_I2S_ID_T i2s_id,struct HAL_I2S_TDM_CONFIG_T *tdm_cfg)
{
I2S_TDM_TRACE(2,"%s: i2s_id = %d.", __func__, i2s_id);
ASSERT(i2s_id < HAL_I2S_ID_QTY,"%s: i2s_id = %d!", __func__, i2s_id);
i2s_tdm_cfg[i2s_id] = *tdm_cfg;
}