pinebuds/platform/hal/hal_tdm.c

528 lines
18 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 "hal_tdm.h"
#include "hal_dma.h"
#include "hal_i2s.h"
#include "hal_trace.h"
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#ifndef CHIP_HAS_TDM
#include "hal_i2s_tdm.h"
#endif
#if 0
#define TDM_TRACE TRACE
#define TDM_DUMP8 DUMP8
#else
#define TDM_TRACE(n, str, ...)
#define TDM_DUMP8(str, ...)
#endif
#ifdef CHIP_HAS_TDM
static const char *const invalid_id = "Invalid I2S ID: %d\n";
static inline uint32_t _tdm_get_reg_base(enum HAL_I2S_ID_T id) {
ASSERT(id < HAL_I2S_ID_QTY, invalid_id, id);
switch (id) {
case HAL_I2S_ID_0:
return (I2S0_BASE | I2SIP_TDM_CTRL_REG_OFFSET);
break;
#if defined(CHIP_HAS_I2S) && (CHIP_HAS_I2S > 1)
case HAL_I2S_ID_1:
return (I2S1_BASE | I2SIP_TDM_CTRL_REG_OFFSET);
break;
#endif
default:
break;
}
return 0;
}
static inline bool tdm_slot_cycles_in_arrays(uint8_t slot_cycles) {
if (slot_cycles == (uint8_t)HAL_TDM_SLOT_CYCLES_16 ||
slot_cycles == (uint8_t)HAL_TDM_SLOT_CYCLES_32)
return true;
else
return false;
}
static inline bool tdm_cycles_in_arrays(uint16_t cycles) {
if (cycles == (uint16_t)HAL_TDM_CYCLES_16 ||
cycles == (uint16_t)HAL_TDM_CYCLES_32 ||
cycles == (uint16_t)HAL_TDM_CYCLES_64 ||
cycles == (uint16_t)HAL_TDM_CYCLES_128 ||
cycles == (uint16_t)HAL_TDM_CYCLES_256)
return true;
else
return false;
}
static inline bool tdm_fs_cycles_in_arrays(uint16_t fs_cycles) {
if (fs_cycles == (uint16_t)HAL_TDM_FS_CYCLES_1 ||
fs_cycles == (uint16_t)HAL_TDM_FS_CYCLES_8 ||
fs_cycles == (uint16_t)HAL_TDM_FS_CYCLES_16 ||
fs_cycles == (uint16_t)HAL_TDM_FS_CYCLES_32 ||
fs_cycles == (uint16_t)HAL_TDM_FS_CYCLES_64 ||
fs_cycles == (uint16_t)HAL_TDM_FS_CYCLES_128 ||
fs_cycles == (uint16_t)HAL_TDM_FS_CYCLES_ONE_LESS)
return true;
else
return false;
}
static void tdm_get_config(enum HAL_I2S_ID_T i2s_id,
struct HAL_TDM_CONFIG_T *tdm_config) {
volatile uint32_t *base_addr;
uint32_t val;
base_addr = (uint32_t *)_tdm_get_reg_base(i2s_id);
val = *base_addr;
if ((val & (TDM_MODE_FS_ASSERTED_AT_LAST << TDM_MODE_FS_ASSERTED_SHIFT)) ==
(TDM_MODE_FS_ASSERTED_AT_LAST << TDM_MODE_FS_ASSERTED_SHIFT)) {
tdm_config->mode = HAL_TDM_MODE_FS_ASSERTED_AT_LAST;
} else {
tdm_config->mode = HAL_TDM_MODE_FS_ASSERTED_AT_FIRST;
}
if ((val & (TDM_FS_EDGE_NEGEDGE << TDM_FS_EDGE_SHIFT)) ==
(TDM_FS_EDGE_NEGEDGE << TDM_FS_EDGE_SHIFT)) {
tdm_config->edge = HAL_TDM_FS_EDGE_NEGEDGE;
} else {
tdm_config->edge = HAL_TDM_FS_EDGE_POSEDGE;
}
if ((val & (TDM_FRAME_WIDTH_16_CYCLES << TDM_FRAME_WIDTH_SHIFT)) ==
(TDM_FRAME_WIDTH_16_CYCLES << TDM_FRAME_WIDTH_SHIFT)) {
tdm_config->cycles = HAL_TDM_CYCLES_16;
} else if ((val & (TDM_FRAME_WIDTH_32_CYCLES << TDM_FRAME_WIDTH_SHIFT)) ==
(TDM_FRAME_WIDTH_32_CYCLES << TDM_FRAME_WIDTH_SHIFT)) {
tdm_config->cycles = HAL_TDM_CYCLES_32;
} else if ((val & (TDM_FRAME_WIDTH_64_CYCLES << TDM_FRAME_WIDTH_SHIFT)) ==
(TDM_FRAME_WIDTH_64_CYCLES << TDM_FRAME_WIDTH_SHIFT)) {
tdm_config->cycles = HAL_TDM_CYCLES_64;
} else if ((val & (TDM_FRAME_WIDTH_128_CYCLES << TDM_FRAME_WIDTH_SHIFT)) ==
(TDM_FRAME_WIDTH_128_CYCLES << TDM_FRAME_WIDTH_SHIFT)) {
tdm_config->cycles = HAL_TDM_CYCLES_128;
} else if ((val & (TDM_FRAME_WIDTH_256_CYCLES << TDM_FRAME_WIDTH_SHIFT)) ==
(TDM_FRAME_WIDTH_256_CYCLES << TDM_FRAME_WIDTH_SHIFT)) {
tdm_config->cycles = HAL_TDM_CYCLES_256;
} else {
tdm_config->cycles = HAL_TDM_CYCLES_256;
}
if ((val & (TDM_FS_WIDTH_8_CYCLES << TDM_FS_WIDTH_SHIFT)) ==
(TDM_FS_WIDTH_8_CYCLES << TDM_FS_WIDTH_SHIFT)) {
tdm_config->fs_cycles = HAL_TDM_FS_CYCLES_8;
} else if ((val & (TDM_FS_WIDTH_16_CYCLES << TDM_FS_WIDTH_SHIFT)) ==
(TDM_FS_WIDTH_16_CYCLES << TDM_FS_WIDTH_SHIFT)) {
tdm_config->fs_cycles = HAL_TDM_FS_CYCLES_16;
} else if ((val & (TDM_FS_WIDTH_32_CYCLES << TDM_FS_WIDTH_SHIFT)) ==
(TDM_FS_WIDTH_32_CYCLES << TDM_FS_WIDTH_SHIFT)) {
tdm_config->fs_cycles = HAL_TDM_FS_CYCLES_32;
} else if ((val & (TDM_FS_WIDTH_64_CYCLES << TDM_FS_WIDTH_SHIFT)) ==
(TDM_FS_WIDTH_64_CYCLES << TDM_FS_WIDTH_SHIFT)) {
tdm_config->fs_cycles = HAL_TDM_FS_CYCLES_64;
} else if ((val & (TDM_FS_WIDTH_128_CYCLES << TDM_FS_WIDTH_SHIFT)) ==
(TDM_FS_WIDTH_128_CYCLES << TDM_FS_WIDTH_SHIFT)) {
tdm_config->fs_cycles = HAL_TDM_FS_CYCLES_128;
} else if ((val & (TDM_FS_WIDTH_1_CYCLE << TDM_FS_WIDTH_SHIFT)) ==
(TDM_FS_WIDTH_1_CYCLE << TDM_FS_WIDTH_SHIFT)) {
tdm_config->fs_cycles = HAL_TDM_FS_CYCLES_ONE_LESS;
} else {
tdm_config->fs_cycles = HAL_TDM_FS_CYCLES_ONE_LESS;
}
if ((val & (TDM_SLOT_WIDTH_16_BIT << TDM_SLOT_WIDTH_SHIFT)) ==
(TDM_SLOT_WIDTH_16_BIT << TDM_SLOT_WIDTH_SHIFT)) {
tdm_config->slot_cycles = HAL_TDM_SLOT_CYCLES_16;
} else {
tdm_config->slot_cycles = HAL_TDM_SLOT_CYCLES_32;
}
tdm_config->data_offset = ((val >> TDM_DATA_OFFSET_SHIT) & 0x7);
}
static void tdm_set_config(enum HAL_I2S_ID_T i2s_id,
struct HAL_TDM_CONFIG_T *tdm_config) {
volatile uint32_t *base_addr;
uint32_t val = 0;
base_addr = (uint32_t *)_tdm_get_reg_base(i2s_id);
ASSERT(tdm_config->mode < HAL_TDM_MODE_NUM, "%s: mode = %d error!", __func__,
tdm_config->mode);
ASSERT(tdm_config->edge < HAL_TDM_FS_EDGE_NUM, "%s: edge = %d error!",
__func__, tdm_config->edge);
ASSERT(tdm_cycles_in_arrays((uint16_t)tdm_config->cycles),
"%s: cycles = %d error!", __func__, tdm_config->cycles);
ASSERT(tdm_fs_cycles_in_arrays((uint16_t)tdm_config->fs_cycles),
"%s: fs_cycles = %d cycles = %d error!", __func__,
tdm_config->fs_cycles, tdm_config->cycles);
ASSERT(tdm_slot_cycles_in_arrays((uint8_t)tdm_config->slot_cycles),
"%s: slot_cycles = %d error!", __func__, tdm_config->slot_cycles);
if (tdm_config->mode == HAL_TDM_MODE_FS_ASSERTED_AT_LAST) {
val |= (TDM_MODE_FS_ASSERTED_AT_LAST << TDM_MODE_FS_ASSERTED_SHIFT);
}
if (tdm_config->edge == HAL_TDM_FS_EDGE_NEGEDGE) {
val |= (TDM_FS_EDGE_NEGEDGE << TDM_FS_EDGE_SHIFT);
}
if (tdm_config->cycles == HAL_TDM_CYCLES_16) {
val |= (TDM_FRAME_WIDTH_16_CYCLES << TDM_FRAME_WIDTH_SHIFT);
} else if (tdm_config->cycles == HAL_TDM_CYCLES_32) {
val |= (TDM_FRAME_WIDTH_32_CYCLES << TDM_FRAME_WIDTH_SHIFT);
} else if (tdm_config->cycles == HAL_TDM_CYCLES_64) {
val |= (TDM_FRAME_WIDTH_64_CYCLES << TDM_FRAME_WIDTH_SHIFT);
} else if (tdm_config->cycles == HAL_TDM_CYCLES_128) {
val |= (TDM_FRAME_WIDTH_128_CYCLES << TDM_FRAME_WIDTH_SHIFT);
} else {
val |= (TDM_FRAME_WIDTH_256_CYCLES << TDM_FRAME_WIDTH_SHIFT);
}
if (tdm_config->fs_cycles == HAL_TDM_FS_CYCLES_8) {
val |= (TDM_FS_WIDTH_8_CYCLES << TDM_FS_WIDTH_SHIFT);
} else if (tdm_config->fs_cycles == HAL_TDM_FS_CYCLES_16) {
val |= (TDM_FS_WIDTH_16_CYCLES << TDM_FS_WIDTH_SHIFT);
} else if (tdm_config->fs_cycles == HAL_TDM_FS_CYCLES_32) {
val |= (TDM_FS_WIDTH_32_CYCLES << TDM_FS_WIDTH_SHIFT);
} else if (tdm_config->fs_cycles == HAL_TDM_FS_CYCLES_64) {
val |= (TDM_FS_WIDTH_64_CYCLES << TDM_FS_WIDTH_SHIFT);
} else if (tdm_config->fs_cycles == HAL_TDM_FS_CYCLES_128) {
val |= (TDM_FS_WIDTH_128_CYCLES << TDM_FS_WIDTH_SHIFT);
} else {
val |= (TDM_FS_WIDTH_FRAME_LENGTH_1_CYCLES << TDM_FS_WIDTH_SHIFT);
}
if (tdm_config->slot_cycles == HAL_TDM_SLOT_CYCLES_16) {
val |= (TDM_SLOT_WIDTH_16_BIT << TDM_SLOT_WIDTH_SHIFT);
}
if (tdm_config->data_offset >= TDM_DATA_OFFSET_MIN &&
tdm_config->data_offset <= TDM_DATA_OFFSET_MAX) {
val |= (tdm_config->data_offset << TDM_DATA_OFFSET_SHIT);
}
*base_addr = val;
}
static void tdm_get_default_config(struct HAL_TDM_CONFIG_T *tdm_config) {
tdm_config->mode = HAL_TDM_MODE_FS_ASSERTED_AT_FIRST;
tdm_config->edge = HAL_TDM_FS_EDGE_POSEDGE;
tdm_config->cycles = HAL_TDM_CYCLES_32;
tdm_config->fs_cycles = HAL_TDM_FS_CYCLES_1;
tdm_config->slot_cycles = HAL_TDM_SLOT_CYCLES_32;
tdm_config->data_offset = 0;
}
static void tdm_get_i2s_config(struct HAL_TDM_CONFIG_T *tdm_config) {
tdm_config->mode = HAL_TDM_MODE_FS_ASSERTED_AT_LAST;
tdm_config->edge = HAL_TDM_FS_EDGE_NEGEDGE;
tdm_config->cycles = HAL_TDM_CYCLES_32;
tdm_config->fs_cycles = HAL_TDM_FS_CYCLES_16;
tdm_config->slot_cycles = HAL_TDM_SLOT_CYCLES_16;
tdm_config->data_offset = 0;
}
static void tdm_enable(enum HAL_I2S_ID_T i2s_id, bool enable) {
volatile uint32_t *base_addr;
uint32_t val = 0;
base_addr = (uint32_t *)_tdm_get_reg_base(i2s_id);
val = *base_addr;
if (enable) {
val |= (TDM_ENABLE << TDM_ENABLE_SHIFT);
} else {
val &= ~(TDM_ENABLE << TDM_ENABLE_SHIFT);
}
*base_addr = val;
}
static int32_t tdm_open(enum HAL_I2S_ID_T i2s_id, enum AUD_STREAM_T stream,
enum HAL_I2S_MODE_T mode) {
int ret;
struct HAL_TDM_CONFIG_T tdm_config;
TDM_TRACE(4, "%s: i2s_id = %d, stream = %d, mode = %d.", __func__, i2s_id,
stream, mode);
ret = hal_i2s_open(i2s_id, stream, mode);
if (ret) {
TDM_TRACE(2, "%s: hal_i2s_open failed.ret = %d.", __func__, ret);
}
tdm_enable(i2s_id, false);
tdm_get_default_config(&tdm_config);
tdm_set_config(i2s_id, &tdm_config);
TDM_TRACE(1, "%s: done.", __func__);
return ret;
}
static int32_t tdm_setup_stream(enum HAL_I2S_ID_T i2s_id,
enum AUD_STREAM_T stream, uint32_t sample_rate,
struct HAL_TDM_CONFIG_T *tdm_cfg) {
struct HAL_I2S_CONFIG_T i2s_cfg;
uint32_t cycles;
const uint8_t bits = 16;
int ret;
TDM_TRACE(4, "%s: i2s_id = %d, stream = %d, sample_rate = %d", __func__,
i2s_id, stream, sample_rate);
TDM_TRACE(5,
"%s: tdm_cfg: mode = %d,edge = %d,cycles = %d,fs_cycles = %d,"
"slot_cycles= %d, offs = %d.",
__func__, tdm_cfg->mode, tdm_cfg->edge, tdm_cfg->cycles,
tdm_cfg->fs_cycles, tdm_cfg->slot_cycles, tdm_cfg->data_offset);
cycles = tdm_cfg->cycles;
ASSERT(tdm_cycles_in_arrays(cycles), "%s: cycles = %d error!", __func__,
tdm_cfg->cycles);
memset(&i2s_cfg, 0, sizeof(i2s_cfg));
i2s_cfg.use_dma = true;
i2s_cfg.sync_start = tdm_cfg->sync_start;
i2s_cfg.chan_sep_buf = false;
i2s_cfg.bits = bits;
i2s_cfg.channel_num = 2;
i2s_cfg.channel_map = AUD_CHANNEL_MAP_CH0 | AUD_CHANNEL_MAP_CH1;
i2s_cfg.sample_rate = sample_rate * (cycles / bits) / 2;
ret = hal_i2s_setup_stream(i2s_id, stream, &i2s_cfg);
if (ret) {
TDM_TRACE(2, "%s: playback failed.ret = %d.", __func__, ret);
} else {
tdm_enable(i2s_id, false);
tdm_set_config(i2s_id, tdm_cfg);
}
TDM_TRACE(1, "%s done.", __func__);
return ret;
}
int32_t tdm_as_i2s_setup_stream(enum HAL_I2S_ID_T i2s_id,
enum AUD_STREAM_T stream,
uint32_t sample_rate) {
struct HAL_I2S_CONFIG_T i2s_cfg;
struct HAL_TDM_CONFIG_T tdm_cfg;
int ret;
TDM_TRACE(4, "%s: i2s_id = %d, stream = %d, sample_rate = 0x%x.", __func__,
i2s_id, stream, sample_rate);
memset(&i2s_cfg, 0, sizeof(i2s_cfg));
i2s_cfg.use_dma = true;
i2s_cfg.sync_start = stream == AUD_STREAM_PLAYBACK ? true : false;
i2s_cfg.chan_sep_buf = false;
i2s_cfg.bits = 16;
i2s_cfg.channel_num = 2;
i2s_cfg.channel_map = AUD_CHANNEL_MAP_CH0 | AUD_CHANNEL_MAP_CH1;
i2s_cfg.sample_rate = sample_rate;
ret = hal_i2s_setup_stream(i2s_id, stream, &i2s_cfg);
if (ret) {
TDM_TRACE(2, "%s: playback failed.ret = %d.", __func__, ret);
} else {
tdm_enable(i2s_id, false);
tdm_get_i2s_config(&tdm_cfg);
tdm_set_config(i2s_id, &tdm_cfg);
}
TDM_TRACE(1, "%s: done.", __func__);
return ret;
}
static int32_t tdm_start_stream(enum HAL_I2S_ID_T i2s_id,
enum AUD_STREAM_T stream) {
int ret;
TDM_TRACE(3, "%s: i2s_id = %d, stream = %d.", __func__, i2s_id, stream);
tdm_enable(i2s_id, true);
ret = hal_i2s_start_stream(i2s_id, stream);
TDM_TRACE(2, "%s done. ret = %d.", __func__, ret);
return ret;
}
static int32_t tdm_stop_stream(enum HAL_I2S_ID_T i2s_id,
enum AUD_STREAM_T stream) {
int ret;
TDM_TRACE(3, "%s: i2s_id = %d, stream = %d.", __func__, i2s_id, stream);
tdm_enable(i2s_id, false);
ret = hal_i2s_stop_stream(i2s_id, stream);
TDM_TRACE(2, "%s done. ret = %d.", __func__, ret);
return ret;
}
static int32_t tdm_close(enum HAL_I2S_ID_T i2s_id, enum AUD_STREAM_T stream) {
int ret;
TDM_TRACE(3, "%s: i2s_id = %d, stream = %d.", __func__, i2s_id, stream);
tdm_enable(i2s_id, false);
ret = hal_i2s_close(i2s_id, stream);
TDM_TRACE(2, "%s done. ret = %d.", __func__, ret);
return ret;
}
#endif // CHIP_HAS_TDM
int32_t hal_tdm_open(enum HAL_I2S_ID_T i2s_id, enum AUD_STREAM_T stream,
enum HAL_I2S_MODE_T mode) {
int ret;
TDM_TRACE(3, "hal_tdm_open:i2s_id = %d, stream = %d, mode = %d.", i2s_id,
stream, mode);
ASSERT(i2s_id < HAL_I2S_ID_QTY, "%s: i2s_id = %d!", __func__, i2s_id);
#ifdef CHIP_HAS_TDM
ret = tdm_open(i2s_id, stream, mode);
#else
ASSERT(stream == AUD_STREAM_CAPTURE, "stream = AUD_STREAM_PLAYBACK!");
ret = hal_i2s_tdm_open(i2s_id, mode);
#endif
TDM_TRACE(2, "%s done. ret = %d.", __func__, ret);
return ret;
}
int32_t hal_tdm_setup_stream(enum HAL_I2S_ID_T i2s_id, enum AUD_STREAM_T stream,
uint32_t sample_rate,
struct HAL_TDM_CONFIG_T *tdm_cfg) {
int ret;
TDM_TRACE(4, "%s:i2s_id = %d, stream = %d, sample_rate = %d.", __func__,
i2s_id, stream, sample_rate);
ASSERT(i2s_id < HAL_I2S_ID_QTY, "%s: i2s_id = %d!", __func__, i2s_id);
TDM_TRACE(4, "%s: cycles = %d, fs_cycles = %d,slot_cycles = %d.", __func__,
tdm_cfg->cycles, tdm_cfg->fs_cycles, tdm_cfg->slot_cycles);
#ifdef CHIP_HAS_TDM
ret = tdm_setup_stream(i2s_id, stream, sample_rate, tdm_cfg);
#else
struct HAL_I2S_TDM_CONFIG_T i2s_tdm_cfg;
i2s_tdm_cfg.cycles = (enum HAL_I2S_TDM_CYCLES_T)tdm_cfg->cycles;
i2s_tdm_cfg.fs_cycles = (enum HAL_I2S_TDM_FS_CYCLES_T)tdm_cfg->fs_cycles;
i2s_tdm_cfg.slot_cycles =
(enum HAL_I2S_TDM_SLOT_CYCLES_T)tdm_cfg->slot_cycles;
i2s_tdm_cfg.data_offset = tdm_cfg->data_offset;
ret = hal_i2s_tdm_setup(i2s_id, sample_rate, &i2s_tdm_cfg);
#endif
TDM_TRACE(2, "%s done. ret = %d.", __func__, ret);
return ret;
}
int32_t hal_tdm_as_i2s_setup_stream(enum HAL_I2S_ID_T i2s_id,
enum AUD_STREAM_T stream,
uint32_t sample_rate) {
int ret;
TDM_TRACE(4, "%s: i2s_id = %d, stream = %d, sample_rate = %d.", __func__,
i2s_id, stream, sample_rate);
ASSERT(i2s_id < HAL_I2S_ID_QTY, "%s: i2s_id = %d!", __func__, i2s_id);
// i2s setup stream playback and capture.
#ifdef CHIP_HAS_TDM
ret = tdm_as_i2s_setup_stream(i2s_id, stream, sample_rate);
#else
struct HAL_I2S_TDM_CONFIG_T i2s_tdm_cfg = {
HAL_I2S_TDM_CYCLES_32,
HAL_I2S_TDM_FS_CYCLES_16,
HAL_I2S_TDM_SLOT_CYCLES_16,
};
ret = hal_i2s_tdm_setup(i2s_id, sample_rate, &i2s_tdm_cfg);
#endif
TDM_TRACE(2, "%s done. ret = %d.", __func__, ret);
return ret;
}
int32_t hal_tdm_start_stream(enum HAL_I2S_ID_T i2s_id,
enum AUD_STREAM_T stream) {
int ret;
TDM_TRACE(3, "%s: i2s_id = %d stream = %d.", __func__, i2s_id, stream);
ASSERT(i2s_id < HAL_I2S_ID_QTY, "%s: i2s_id = %d!", __func__, i2s_id);
#ifdef CHIP_HAS_TDM
ret = tdm_start_stream(i2s_id, stream);
#else
ret = hal_i2s_tdm_start_stream(i2s_id);
#endif
TDM_TRACE(2, "%s done. ret = %d.", __func__, ret);
return ret;
}
int32_t hal_tdm_stop_stream(enum HAL_I2S_ID_T i2s_id,
enum AUD_STREAM_T stream) {
int32_t ret;
TDM_TRACE(3, "%s: i2s_id = %d stream = %d.", __func__, i2s_id, stream);
ASSERT(i2s_id < HAL_I2S_ID_QTY, "%s: i2s_id = %d!", __func__, i2s_id);
#ifdef CHIP_HAS_TDM
ret = tdm_stop_stream(i2s_id, stream);
#else
ret = hal_i2s_tdm_stop_stream(i2s_id);
#endif
TDM_TRACE(2, "%s done. ret = %d.", __func__, ret);
return ret;
}
int32_t hal_tdm_close(enum HAL_I2S_ID_T i2s_id, enum AUD_STREAM_T stream) {
int32_t ret;
TDM_TRACE(3, "%s: i2s_id = %d stream = %d.", __func__, i2s_id, stream);
ASSERT(i2s_id < HAL_I2S_ID_QTY, "%s: i2s_id = %d!", __func__, i2s_id);
#ifdef CHIP_HAS_TDM
ret = tdm_close(i2s_id, stream);
#else
ret = hal_i2s_tdm_close(i2s_id);
#endif
TDM_TRACE(2, "%s done. ret = %d.", __func__, ret);
return ret;
}
void hal_tdm_get_config(enum HAL_I2S_ID_T i2s_id,
struct HAL_TDM_CONFIG_T *tdm_cfg) {
ASSERT(i2s_id < HAL_I2S_ID_QTY, "%s: i2s_id = %d!", __func__, i2s_id);
#ifdef CHIP_HAS_TDM
tdm_get_config(i2s_id, tdm_cfg);
#else
struct HAL_I2S_TDM_CONFIG_T i2s_tdm_cfg;
hal_i2s_tdm_get_config(i2s_id, &i2s_tdm_cfg);
tdm_cfg->mode = HAL_TDM_MODE_FS_ASSERTED_AT_FIRST;
tdm_cfg->edge = HAL_TDM_FS_EDGE_POSEDGE;
tdm_cfg->cycles = (enum HAL_TDM_CYCLES_T)i2s_tdm_cfg.cycles;
tdm_cfg->fs_cycles = (enum HAL_TDM_FS_CYCLES)i2s_tdm_cfg.fs_cycles;
tdm_cfg->slot_cycles = (enum HAL_TDM_SLOT_CYCLES_T)i2s_tdm_cfg.slot_cycles;
tdm_cfg->data_offset = i2s_tdm_cfg.data_offset;
#endif
}
void hal_tdm_set_config(enum HAL_I2S_ID_T i2s_id,
struct HAL_TDM_CONFIG_T *tdm_cfg) {
ASSERT(i2s_id < HAL_I2S_ID_QTY, "%s: i2s_id = %d!", __func__, i2s_id);
#ifdef CHIP_HAS_TDM
tdm_set_config(i2s_id, tdm_cfg);
#else
struct HAL_I2S_TDM_CONFIG_T i2s_tdm_cfg;
i2s_tdm_cfg.cycles = HAL_I2S_TDM_CYCLES_32;
i2s_tdm_cfg.fs_cycles = HAL_I2S_TDM_FS_CYCLES_16;
i2s_tdm_cfg.slot_cycles = HAL_I2S_TDM_SLOT_CYCLES_16;
i2s_tdm_cfg.data_offset = 0;
hal_i2s_tdm_set_config(i2s_id, &i2s_tdm_cfg);
#endif
}