KT24-1110_65E-HA-651B/cpu/br25/audio_common/audio_link.c
2024-11-10 18:44:17 +08:00

570 lines
18 KiB
C

/*****************************************************************
>file name : lib/media/cpu/br22/audio_link.c
>author :
>create time : Fri 7 Dec 2018 14:59:12 PM CST
*****************************************************************/
#include "includes.h"
#include "asm/includes.h"
#include "media/includes.h"
#include "system/includes.h"
#include "app_config.h"
#include "asm/clock.h"
#include "asm/iis.h"
#include "audio_link.h"
#if TCFG_IIS_ENABLE
#define ALINK_TEST_ENABLE
#define ALINK_DEBUG_INFO
#ifdef ALINK_DEBUG_INFO
#define alink_printf printf
#else
#define alink_printf(...)
#endif
static u32 *ALNK_BUF_ADR[] = {
(u32 *)(&(JL_ALNK0->ADR0)),
(u32 *)(&(JL_ALNK0->ADR1)),
(u32 *)(&(JL_ALNK0->ADR2)),
(u32 *)(&(JL_ALNK0->ADR3)),
};
enum {
IIS_IO_MCLK = 0u,
IIS_IO_SCLK ,
IIS_IO_LRCK ,
IIS_IO_CH0 ,
IIS_IO_CH1 ,
IIS_IO_CH2 ,
IIS_IO_CH3 ,
};
enum {
MCLK_11M2896K = 0,
MCLK_12M288K
};
enum {
MCLK_EXTERNAL = 0u,
MCLK_SYS_CLK ,
MCLK_OSC_CLK ,
MCLK_PLL_CLK ,
};
enum {
MCLK_DIV_1 = 0u,
MCLK_DIV_2 ,
MCLK_DIV_4 ,
MCLK_DIV_8 ,
MCLK_DIV_16 ,
};
enum {
MCLK_LRDIV_EX = 0u,
MCLK_LRDIV_64FS ,
MCLK_LRDIV_128FS ,
MCLK_LRDIV_192FS ,
MCLK_LRDIV_256FS ,
MCLK_LRDIV_384FS ,
MCLK_LRDIV_512FS ,
MCLK_LRDIV_768FS ,
};
static const u8 alink0_IOS_portA[] = {
IO_PORTA_02, IO_PORTA_03, IO_PORTA_04, IO_PORTA_00, IO_PORTA_01, IO_PORTA_05, IO_PORTA_06,
};
static const u8 alink0_IOS_portB[] = {
IO_PORTC_02, IO_PORTA_05, IO_PORTA_06, IO_PORTA_00, IO_PORTA_02, IO_PORTA_03, IO_PORTA_04,
};
static u8 alink0_IOS_port[7] = {0};
ALINK_PARM *p_alink_parm;
//==================================================
enum alnk_clock {
ALINK_CLOCK_12M288K,
ALINK_CLOCK_11M2896K,
};
static void audio_link_clock_sel(enum alnk_clock clk)
{
switch (clk) {
case ALINK_CLOCK_12M288K:
SFR(JL_CLOCK->CLK_CON2, 6, 4, 5); //160/13
break;
case ALINK_CLOCK_11M2896K:
SFR(JL_CLOCK->CLK_CON2, 6, 4, 0); //192 / 17
break;
default:
break;
}
}
//==================================================
static void alink_clk_io_in_init(u8 gpio)
{
gpio_set_direction(gpio, 1);
gpio_set_pull_down(gpio, 0);
gpio_set_pull_up(gpio, 1);
gpio_set_die(gpio, 1);
}
static void *alink_addr(u8 ch)
{
u8 *buf_addr = NULL; //can be used
u32 buf_index = 0;
buf_addr = (u8 *)(p_alink_parm->ch_cfg[ch].buf);
u8 index_table[4] = {12, 13, 14, 15};
u8 bit_index = index_table[ch];
buf_index = (JL_ALNK0->CON0 & BIT(bit_index)) ? 0 : 1;
buf_addr = buf_addr + ((p_alink_parm->dma_len / 2) * buf_index);
return buf_addr;
}
___interrupt
static void alink_isr(void)
{
u16 reg;
s16 *buf_addr = NULL ;
u8 ch = 0;
reg = JL_ALNK0->CON2;
//channel 0
if (reg & BIT(4)) {
ch = 0;
ALINK_CLR_CH0_PND();
buf_addr = alink_addr(ALINK_CH0);
goto __isr_cb;
}
//channel 1
if (reg & BIT(5)) {
ch = 1;
ALINK_CLR_CH1_PND();
buf_addr = alink_addr(ALINK_CH1);
goto __isr_cb;
}
//channel 2
if (reg & BIT(6)) {
ch = 2;
ALINK_CLR_CH2_PND();
buf_addr = alink_addr(ALINK_CH2);
goto __isr_cb;
}
//channel 3
if (reg & BIT(7)) {
ch = 3;
ALINK_CLR_CH3_PND();
buf_addr = alink_addr(ALINK_CH3);
goto __isr_cb;
}
__isr_cb:
if (p_alink_parm->ch_cfg[ch].isr_cb) {
p_alink_parm->ch_cfg[ch].isr_cb(ch, buf_addr, p_alink_parm->dma_len / 2);
}
}
static void alink_sr(u32 rate)
{
alink_printf("ALINK_SR = %d\n", rate);
switch (rate) {
case ALINK_SR_48000:
/*12.288Mhz 256fs*/
audio_link_clock_sel(ALINK_CLOCK_12M288K);
ALINK_MDIV(MCLK_DIV_1);
ALINK_LRDIV(MCLK_LRDIV_256FS);
break ;
case ALINK_SR_44100:
/*11.289Mhz 256fs*/
audio_link_clock_sel(ALINK_CLOCK_11M2896K);
ALINK_MDIV(MCLK_DIV_1);
ALINK_LRDIV(MCLK_LRDIV_256FS);
break ;
case ALINK_SR_32000:
/*12.288Mhz 384fs*/
audio_link_clock_sel(ALINK_CLOCK_12M288K);
ALINK_MDIV(MCLK_DIV_1);
ALINK_LRDIV(MCLK_LRDIV_384FS);
break ;
case ALINK_SR_24000:
/*12.288Mhz 512fs*/
audio_link_clock_sel(ALINK_CLOCK_12M288K);
ALINK_MDIV(MCLK_DIV_2);
ALINK_LRDIV(MCLK_LRDIV_256FS);
break ;
case ALINK_SR_22050:
/*11.289Mhz 512fs*/
audio_link_clock_sel(ALINK_CLOCK_11M2896K);
ALINK_MDIV(MCLK_DIV_2);
ALINK_LRDIV(MCLK_LRDIV_256FS);
break ;
case ALINK_SR_16000:
/*12.288/2Mhz 384fs*/
audio_link_clock_sel(ALINK_CLOCK_12M288K);
ALINK_MDIV(MCLK_DIV_2);
ALINK_LRDIV(MCLK_LRDIV_384FS);
break ;
case ALINK_SR_12000:
/*12.288/2Mhz 512fs*/
audio_link_clock_sel(ALINK_CLOCK_12M288K);
ALINK_MDIV(MCLK_DIV_4);
ALINK_LRDIV(MCLK_LRDIV_256FS);
break ;
case ALINK_SR_11025:
/*11.289/2Mhz 512fs*/
audio_link_clock_sel(ALINK_CLOCK_11M2896K);
ALINK_MDIV(MCLK_DIV_4);
ALINK_LRDIV(MCLK_LRDIV_256FS);
break ;
case ALINK_SR_8000:
/*12.288/4Mhz 384fs*/
audio_link_clock_sel(ALINK_CLOCK_12M288K);
ALINK_MDIV(MCLK_DIV_4);
ALINK_LRDIV(MCLK_LRDIV_384FS);
break ;
default:
//44100
/*11.289Mhz 256fs*/
audio_link_clock_sel(ALINK_CLOCK_11M2896K);
ALINK_MDIV(MCLK_DIV_1);
ALINK_LRDIV(MCLK_LRDIV_256FS);
break;
}
if (p_alink_parm->role == ALINK_ROLE_SLAVE) {
ALINK_LRDIV(MCLK_LRDIV_EX);
}
}
void alink_channel_init(ALINK_PORT port, u8 ch_idx, u8 dir, void (*handle)(u8 ch, s16 *buf, u32 len))
{
if (!p_alink_parm->ch_cfg[ch_idx].enable) {
return;
}
p_alink_parm->ch_cfg[ch_idx].dir = dir;
p_alink_parm->ch_cfg[ch_idx].isr_cb = handle;
p_alink_parm->ch_cfg[ch_idx].buf = malloc(p_alink_parm->dma_len);
memset(p_alink_parm->ch_cfg[ch_idx].buf, 0xFF, p_alink_parm->dma_len);
u32 *buf_addr;
//===================================//
// ALNK工作模式 //
//===================================//
if (p_alink_parm->mode > ALINK_MD_IIS_RALIGN) {
ALINK_CHx_MODE_SEL((p_alink_parm->mode - ALINK_MD_IIS_RALIGN), ch_idx);
} else {
ALINK_CHx_MODE_SEL(p_alink_parm->mode, ch_idx);
}
//===================================//
// ALNK CH DMA BUF //
//===================================//
buf_addr = ALNK_BUF_ADR[ch_idx];
*buf_addr = (u32)(p_alink_parm->ch_cfg[ch_idx].buf);
//===================================//
// ALNK CH DAT IO INIT //
//===================================//
if (p_alink_parm->ch_cfg[ch_idx].dir == ALINK_DIR_RX) {
gpio_set_direction(alink0_IOS_port[IIS_IO_CH0 + ch_idx], 1);
gpio_set_pull_down(alink0_IOS_port[IIS_IO_CH0 + ch_idx], 0);
gpio_set_pull_up(alink0_IOS_port[IIS_IO_CH0 + ch_idx], 1);
gpio_set_die(alink0_IOS_port[IIS_IO_CH0 + ch_idx], 1);
ALINK_CHx_DIR_RX_MODE(ch_idx);
} else {
gpio_direction_output(alink0_IOS_port[IIS_IO_CH0 + ch_idx], 1);
ALINK_CHx_DIR_TX_MODE(ch_idx);
}
}
static void alink_info_dump()
{
alink_printf("JL_ALNK0->CON0 = 0x%x", JL_ALNK0->CON0);
alink_printf("JL_ALNK0->CON1 = 0x%x", JL_ALNK0->CON1);
alink_printf("JL_ALNK0->CON2 = 0x%x", JL_ALNK0->CON2);
alink_printf("JL_ALNK0->CON3 = 0x%x", JL_ALNK0->CON3);
alink_printf("JL_ALNK0->LEN = 0x%x", JL_ALNK0->LEN);
alink_printf("JL_ALNK0->ADR0 = 0x%x", JL_ALNK0->ADR0);
alink_printf("JL_ALNK0->ADR1 = 0x%x", JL_ALNK0->ADR1);
alink_printf("JL_ALNK0->ADR2 = 0x%x", JL_ALNK0->ADR2);
alink_printf("JL_ALNK0->ADR3 = 0x%x", JL_ALNK0->ADR3);
alink_printf("JL_CLOCK->CLK_CON2 = 0x%x", JL_CLOCK->CLK_CON2);
}
int alink_init(ALINK_PARM *parm)
{
if (parm == NULL) {
return -1;
}
ALNK_CON_RESET();
p_alink_parm = parm;
if (p_alink_parm->port_select == ALINK0_PORTA) {
for (int i = 0; i < 7; i++) {
alink0_IOS_port[i] = alink0_IOS_portA[i];
}
JL_IOMAP->CON1 &= ~BIT(23);
r_printf("iis_portA init\n");
} else {
for (int i = 0; i < 7; i++) {
alink0_IOS_port[i] = alink0_IOS_portB[i];
}
JL_IOMAP->CON1 |= BIT(23);
r_printf("iis_portB init\n");
}
request_irq(IRQ_ALINK0_IDX, 3, alink_isr, 0);
ALINK_MSRC(MCLK_PLL_CLK); /*MCLK source*/
//===================================//
// 输出时钟配置 //
//===================================//
if (parm->role == ALINK_ROLE_MASTER) {
//主机输出时钟
gpio_direction_output(alink0_IOS_port[IIS_IO_MCLK], 1);
ALINK_MOE(1); /*MCLK output to IO*/
gpio_direction_output(alink0_IOS_port[IIS_IO_LRCK], 1);
gpio_direction_output(alink0_IOS_port[IIS_IO_SCLK], 1);
ALINK_SOE(1); /*SCLK/LRCK output to IO*/
} else {
//从机输入时钟
alink_clk_io_in_init(alink0_IOS_port[IIS_IO_MCLK]);
ALINK_MOE(0); /*MCLK input to IO*/
alink_clk_io_in_init(alink0_IOS_port[IIS_IO_LRCK]);
alink_clk_io_in_init(alink0_IOS_port[IIS_IO_SCLK]);
ALINK_SOE(0); /*SCLK/LRCK output to IO*/
}
//===================================//
// 基本模式/扩展模式 //
//===================================//
ALINK_DSPE(p_alink_parm->mode >> 2);
//===================================//
// 数据位宽16/32bit //
//===================================//
//注意: 配置了24bit, 一定要选ALINK_FRAME_64SCLK, 因为sclk32 x 2才会有24bit;
if (p_alink_parm->bitwide == ALINK_LEN_24BIT) {
ASSERT(p_alink_parm->sclk_per_frame == ALINK_FRAME_64SCLK);
ALINK_24BIT_MODE();
//一个点需要4bytes, LR = 2, 双buf = 2
JL_ALNK0->LEN = p_alink_parm->dma_len / 8; //点数
} else {
ALINK_16BIT_MODE();
//一个点需要2bytes, LR = 2, 双buf = 2
JL_ALNK0->LEN = p_alink_parm->dma_len / 8; //点数
}
//===================================//
// 时钟边沿选择 //
//===================================//
if (p_alink_parm->clk_mode == ALINK_CLK_FALL_UPDATE_RAISE_SAMPLE) {
SCLKINV(0);
} else {
SCLKINV(1);
}
//===================================//
// 每帧数据sclk个数 //
//===================================//
if (p_alink_parm->sclk_per_frame == ALINK_FRAME_64SCLK) {
F32_EN(0);
} else {
F32_EN(1);
}
//===================================//
// 设置数据采样率 //
//===================================//
alink_sr(p_alink_parm->sample_rate);
return 0;
}
void alink_channel_close(ALINK_PORT port, u8 ch_idx)
{
gpio_set_pull_up(alink0_IOS_port[IIS_IO_CH0 + ch_idx], 0);
gpio_set_pull_down(alink0_IOS_port[IIS_IO_CH0 + ch_idx], 0);
gpio_set_direction(alink0_IOS_port[IIS_IO_CH0 + ch_idx], 1);
gpio_set_die(alink0_IOS_port[IIS_IO_CH0 + ch_idx], 0);
ALINK_CHx_CLOSE(1, ch_idx);
if (p_alink_parm->ch_cfg[ch_idx].buf) {
free(p_alink_parm->ch_cfg[ch_idx].buf);
p_alink_parm->ch_cfg[ch_idx].buf = NULL;
}
r_printf("alink_ch %d closed\n", ch_idx);
}
int alink_start(ALINK_PORT port)
{
if (p_alink_parm) {
ALINK_EN(1);
return 0;
}
return -1;
}
void alink_uninit(ALINK_PORT port)
{
ALINK_EN(0);
for (int i = 0; i < 4; i++) {
if (p_alink_parm->ch_cfg[i].buf) {
alink_channel_close(port, i);
}
}
p_alink_parm = NULL;
alink_printf("audio_link_exit OK\n");
}
int alink_sr_set(ALINK_PORT port, u16 sr)
{
if (p_alink_parm) {
ALINK_EN(0);
alink_sr(sr);
ALINK_EN(1);
return 0;
} else {
return -1;
}
}
extern ALINK_PARM alink_param;
static u8 alink_init_cnt = 0;
void audio_link_init(ALINK_PORT port)
{
if (alink_init_cnt == 0) {
alink_init(&alink_param);
alink_start(port);
}
alink_init_cnt++;
}
void audio_link_uninit(ALINK_PORT port)
{
alink_init_cnt--;
if (alink_init_cnt == 0) {
alink_uninit(port);
}
}
//===============================================//
// for APP use demo //
//===============================================//
#ifdef ALINK_TEST_ENABLE
short const tsin_441k[441] = {
0x0000, 0x122d, 0x23fb, 0x350f, 0x450f, 0x53aa, 0x6092, 0x6b85, 0x744b, 0x7ab5, 0x7ea2, 0x7fff, 0x7ec3, 0x7af6, 0x74ab, 0x6c03,
0x612a, 0x545a, 0x45d4, 0x35e3, 0x24db, 0x1314, 0x00e9, 0xeeba, 0xdce5, 0xcbc6, 0xbbb6, 0xad08, 0xa008, 0x94fa, 0x8c18, 0x858f,
0x8181, 0x8003, 0x811d, 0x84ca, 0x8af5, 0x9380, 0x9e3e, 0xaaf7, 0xb969, 0xc94a, 0xda46, 0xec06, 0xfe2d, 0x105e, 0x223a, 0x3365,
0x4385, 0x5246, 0x5f5d, 0x6a85, 0x7384, 0x7a2d, 0x7e5b, 0x7ffa, 0x7f01, 0x7b75, 0x7568, 0x6cfb, 0x6258, 0x55b7, 0x4759, 0x3789,
0x2699, 0x14e1, 0x02bc, 0xf089, 0xdea7, 0xcd71, 0xbd42, 0xae6d, 0xa13f, 0x95fd, 0x8ce1, 0x861a, 0x81cb, 0x800b, 0x80e3, 0x844e,
0x8a3c, 0x928c, 0x9d13, 0xa99c, 0xb7e6, 0xc7a5, 0xd889, 0xea39, 0xfc5a, 0x0e8f, 0x2077, 0x31b8, 0x41f6, 0x50de, 0x5e23, 0x697f,
0x72b8, 0x799e, 0x7e0d, 0x7fee, 0x7f37, 0x7bed, 0x761f, 0x6ded, 0x6380, 0x570f, 0x48db, 0x392c, 0x2855, 0x16ad, 0x048f, 0xf259,
0xe06b, 0xcf20, 0xbed2, 0xafd7, 0xa27c, 0x9705, 0x8db0, 0x86ab, 0x821c, 0x801a, 0x80b0, 0x83da, 0x8988, 0x919c, 0x9bee, 0xa846,
0xb666, 0xc603, 0xd6ce, 0xe86e, 0xfa88, 0x0cbf, 0x1eb3, 0x3008, 0x4064, 0x4f73, 0x5ce4, 0x6874, 0x71e6, 0x790a, 0x7db9, 0x7fdc,
0x7f68, 0x7c5e, 0x76d0, 0x6ed9, 0x64a3, 0x5863, 0x4a59, 0x3acc, 0x2a0f, 0x1878, 0x0661, 0xf42a, 0xe230, 0xd0d0, 0xc066, 0xb145,
0xa3bd, 0x9813, 0x8e85, 0x8743, 0x8274, 0x8030, 0x8083, 0x836b, 0x88da, 0x90b3, 0x9acd, 0xa6f5, 0xb4ea, 0xc465, 0xd515, 0xe6a3,
0xf8b6, 0x0aee, 0x1ced, 0x2e56, 0x3ecf, 0x4e02, 0x5ba1, 0x6764, 0x710e, 0x786f, 0x7d5e, 0x7fc3, 0x7f91, 0x7cc9, 0x777a, 0x6fc0,
0x65c1, 0x59b3, 0x4bd3, 0x3c6a, 0x2bc7, 0x1a41, 0x0833, 0xf5fb, 0xe3f6, 0xd283, 0xc1fc, 0xb2b7, 0xa503, 0x9926, 0x8f60, 0x87e1,
0x82d2, 0x804c, 0x805d, 0x8303, 0x8833, 0x8fcf, 0x99b2, 0xa5a8, 0xb372, 0xc2c9, 0xd35e, 0xe4da, 0xf6e4, 0x091c, 0x1b26, 0x2ca2,
0x3d37, 0x4c8e, 0x5a58, 0x664e, 0x7031, 0x77cd, 0x7cfd, 0x7fa3, 0x7fb4, 0x7d2e, 0x781f, 0x70a0, 0x66da, 0x5afd, 0x4d49, 0x3e04,
0x2d7d, 0x1c0a, 0x0a05, 0xf7cd, 0xe5bf, 0xd439, 0xc396, 0xb42d, 0xa64d, 0x9a3f, 0x9040, 0x8886, 0x8337, 0x806f, 0x803d, 0x82a2,
0x8791, 0x8ef2, 0x989c, 0xa45f, 0xb1fe, 0xc131, 0xd1aa, 0xe313, 0xf512, 0x074a, 0x195d, 0x2aeb, 0x3b9b, 0x4b16, 0x590b, 0x6533,
0x6f4d, 0x7726, 0x7c95, 0x7f7d, 0x7fd0, 0x7d8c, 0x78bd, 0x717b, 0x67ed, 0x5c43, 0x4ebb, 0x3f9a, 0x2f30, 0x1dd0, 0x0bd6, 0xf99f,
0xe788, 0xd5f1, 0xc534, 0xb5a7, 0xa79d, 0x9b5d, 0x9127, 0x8930, 0x83a2, 0x8098, 0x8024, 0x8247, 0x86f6, 0x8e1a, 0x978c, 0xa31c,
0xb08d, 0xbf9c, 0xcff8, 0xe14d, 0xf341, 0x0578, 0x1792, 0x2932, 0x39fd, 0x499a, 0x57ba, 0x6412, 0x6e64, 0x7678, 0x7c26, 0x7f50,
0x7fe6, 0x7de4, 0x7955, 0x7250, 0x68fb, 0x5d84, 0x5029, 0x412e, 0x30e0, 0x1f95, 0x0da7, 0xfb71, 0xe953, 0xd7ab, 0xc6d4, 0xb725,
0xa8f1, 0x9c80, 0x9213, 0x89e1, 0x8413, 0x80c9, 0x8012, 0x81f3, 0x8662, 0x8d48, 0x9681, 0xa1dd, 0xaf22, 0xbe0a, 0xce48, 0xdf89,
0xf171, 0x03a6, 0x15c7, 0x2777, 0x385b, 0x481a, 0x5664, 0x62ed, 0x6d74, 0x75c4, 0x7bb2, 0x7f1d, 0x7ff5, 0x7e35, 0x79e6, 0x731f,
0x6a03, 0x5ec1, 0x5193, 0x42be, 0x328f, 0x2159, 0x0f77, 0xfd44, 0xeb1f, 0xd967, 0xc877, 0xb8a7, 0xaa49, 0x9da8, 0x9305, 0x8a98,
0x848b, 0x80ff, 0x8006, 0x81a5, 0x85d3, 0x8c7c, 0x957b, 0xa0a3, 0xadba, 0xbc7b, 0xcc9b, 0xddc6, 0xefa2, 0x01d3, 0x13fa, 0x25ba,
0x36b6, 0x4697, 0x5509, 0x61c2, 0x6c80, 0x750b, 0x7b36, 0x7ee3, 0x7ffd, 0x7e7f, 0x7a71, 0x73e8, 0x6b06, 0x5ff8, 0x52f8, 0x444a,
0x343a, 0x231b, 0x1146, 0xff17, 0xecec, 0xdb25, 0xca1d, 0xba2c, 0xaba6, 0x9ed6, 0x93fd, 0x8b55, 0x850a, 0x813d, 0x8001, 0x815e,
0x854b, 0x8bb5, 0x947b, 0x9f6e, 0xac56, 0xbaf1, 0xcaf1, 0xdc05, 0xedd3
};
static u16 tx_s_cnt = 0;
int get_sine_data(u16 *s_cnt, s16 *data, u16 points, u8 ch)
{
while (points--) {
if (*s_cnt >= 441) {
*s_cnt = 0;
}
*data++ = tsin_441k[*s_cnt];
if (ch == 2) {
*data++ = tsin_441k[*s_cnt];
}
(*s_cnt)++;
}
return 0;
}
u32 data_buf[2][ALNK_BUF_POINTS_NUM * 2] __attribute__((aligned(4)));
void handle_tx(u8 ch, void *buf, u16 len)
{
get_sine_data(&tx_s_cnt, buf, len / 2 / 2, 2);
/* put_buf(buf, 32); */
#if 0
s16 *data_tx = (s16 *)buf;
s16 *data = (s16 *)data_buf;
for (int i = 0; i < len / sizeof(s16) ; i++) {
/* data_tx[i] = data[i]; */
data_tx[i] = 0x5a5a;
}
#endif
}
void handle_rx(u8 ch, void *buf, u16 len)
{
/* put_buf(buf, 32); */
#if 0
s16 *data_rx = (s16 *)buf;
s16 *data_b = (s16 *)data_buf;
for (int i = 0; i < len / sizeof(s16) ; i++) {
data_b[i] = data_rx[i];
}
#endif
}
ALINK_PARM test_alink = {
.port_select = ALINK0_PORTA, //MCLK:PA2 SCLK:PA3 LRCK:PA4 CH0:PA0 CH1:PA1 CH2:PA5 CH3:PA6
/* .port_select = ALINK0_PORTB, //MCLK:PC2 SCLK:PA5 LRCK:PA6 CH0:PA0 CH1:PA2 CH2:PA3 CH3:PA4 */
.ch_cfg[0].enable = 1,
.ch_cfg[1].enable = 1,
.mode = ALINK_MD_IIS,
/* .role = ALINK_ROLE_SLAVE, */
.role = ALINK_ROLE_MASTER,
.clk_mode = ALINK_CLK_FALL_UPDATE_RAISE_SAMPLE,
.bitwide = ALINK_LEN_16BIT,
.sclk_per_frame = ALINK_FRAME_64SCLK,
/* .sclk_per_frame = ALINK_FRAME_32SCLK, */
.dma_len = ALNK_BUF_POINTS_NUM * 2 * 2 * 2,
.sample_rate = 44100,
};
void audio_link_test(void)
{
alink_init(&test_alink);
alink_channel_init(0, 0, ALINK_DIR_RX, handle_rx);
alink_channel_init(0, 1, ALINK_DIR_TX, handle_tx);
alink_start(0);
alink_info_dump();
}
#endif
#endif