KT24-1110_65E-HA-651B/apps/common/device/fm/fm_manage.c
2024-11-10 18:44:17 +08:00

318 lines
8.3 KiB
C
Raw Permalink Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

#include "app_config.h"
#include "system/includes.h"
#include "fm/fm_manage.h"
#include "rda5807/RDA5807.h"
#include "bk1080/BK1080.h"
#include "qn8035/QN8035.h"
#include "fm_inside/fm_inside.h"
#include "asm/audio_linein.h"
#include "audio_config.h"
#include "audio_dec/audio_dec_fm.h"
#if TCFG_APP_FM_EN
#define LOG_TAG_CONST APP_FM
#define LOG_TAG "[APP_FM]"
#define LOG_ERROR_ENABLE
#define LOG_DEBUG_ENABLE
#define LOG_INFO_ENABLE
/* #define LOG_DUMP_ENABLE */
#define LOG_CLI_ENABLE
#include "debug.h"
static FM_INTERFACE *t_fm_hdl = NULL;
static FM_INTERFACE *fm_hdl = NULL;
static struct _fm_dev_info *fm_dev_info;
static struct _fm_dev_platform_data *fm_dev_platform_data;
int linein_dec_open(u8 source, u32 sample_rate);
void linein_dec_close(void);
int fm_dev_init(void *_data)
{
fm_dev_platform_data = (struct _fm_dev_platform_data *)_data;
return 0;
fm_dev_info = (struct _fm_dev_info *)malloc(sizeof(struct _fm_dev_info));
if (fm_dev_info == NULL) {
printf("fm_dev_init info malloc err!");
return -1;
}
fm_dev_platform_data = (struct _fm_dev_platform_data *)_data;
memset(fm_dev_info, 0x00, sizeof(struct _fm_dev_info));
fm_dev_info->iic_hdl = fm_dev_platform_data->iic_hdl;
fm_dev_info->iic_delay = fm_dev_platform_data->iic_delay;
/* iic_init(fm_dev_info->iic_hdl);
extern void bk1080_test(void);
bk1080_test(); */
fm_manage_check_online();
if (!fm_hdl) {
/* printf("fm_manage could not find dev\n"); */
return -1;
} else {
extern void wdt_clear();
wdt_clear();
memcpy(fm_dev_info->logo, fm_hdl->logo, strlen(fm_hdl->logo));
/* printf("fm_manage found dev %s\n", fm_dev_info->logo); */
/* fm_manage_init(); */
/* fm_manage_set_fre(875); */
/* fm_manage_mute(0); */
/* u16 iii = 875; */
/* while(!fm_manage_set_fre(iii) && (iii < 1080)) { */
/* printf("fm fre no %d\n", iii); */
/* iii++; */
/* os_time_dly(1); */
/* } */
/* printf("fm fre %d\n", iii); */
/* while(1) { */
/* os_time_dly(1); */
/* } */
}
return 0;
}
u8 fm_dev_iic_write(u8 w_chip_id, u8 register_address, u8 *buf, u32 data_len)
{
u8 ret = 1;
iic_start(fm_dev_info->iic_hdl);
if (0 == iic_tx_byte(fm_dev_info->iic_hdl, w_chip_id)) {
ret = 0;
log_e("\n fm iic wr err 0\n");
goto __gcend;
}
delay(fm_dev_info->iic_delay);
if (0 == iic_tx_byte(fm_dev_info->iic_hdl, register_address)) {
ret = 0;
log_e("\n fm iic wr err 1\n");
goto __gcend;
}
u8 *pbuf = buf;
while (data_len--) {
delay(fm_dev_info->iic_delay);
if (0 == iic_tx_byte(fm_dev_info->iic_hdl, *pbuf++)) {
ret = 0;
log_e("\n fm iic wr err 2\n");
goto __gcend;
}
}
__gcend:
iic_stop(fm_dev_info->iic_hdl);
return ret;
}
u8 fm_dev_iic_readn(u8 r_chip_id, u8 register_address, u8 *buf, u8 data_len)
{
u8 read_len = 0;
iic_start(fm_dev_info->iic_hdl);
if (0 == iic_tx_byte(fm_dev_info->iic_hdl, (r_chip_id & 0x01) ? (r_chip_id - 1) : (r_chip_id))) {
log_e("\n fm iic rd err 0\n");
read_len = 0;
goto __gdend;
}
delay(fm_dev_info->iic_delay);
if (0 == iic_tx_byte(fm_dev_info->iic_hdl, register_address)) {
log_e("\n fm iic rd err 1\n");
read_len = 0;
goto __gdend;
}
delay(fm_dev_info->iic_delay);
iic_start(fm_dev_info->iic_hdl);
if (0 == iic_tx_byte(fm_dev_info->iic_hdl, r_chip_id)) {
log_e("\n fm iic rd err 2\n");
read_len = 0;
goto __gdend;
}
delay(fm_dev_info->iic_delay);
for (; data_len > 1; data_len--) {
*buf++ = iic_rx_byte(fm_dev_info->iic_hdl, 1);
read_len ++;
}
*buf = iic_rx_byte(fm_dev_info->iic_hdl, 0);
__gdend:
iic_stop(fm_dev_info->iic_hdl);
delay(fm_dev_info->iic_delay);
return read_len;
}
void fm_manage_check_online(void)
{
/* printf("fm check online dev \n"); */
/* printf("addr %x %x\n", fm_dev_begin, fm_dev_end); */
////先找外挂fm
list_for_each_fm(t_fm_hdl) {
/* printf("find %x %x %s\n", t_fm_hdl, t_fm_hdl->read_id((void *)fm_dev_info), t_fm_hdl->logo); */
if (t_fm_hdl->read_id((void *)fm_dev_info) &&
(memcmp(t_fm_hdl->logo, "fm_inside", strlen(t_fm_hdl->logo)))) {
fm_hdl = t_fm_hdl;
printf("fm find dev %s \n", t_fm_hdl->logo);
return;
}
}
/* printf("no ex fm dev !!!\n"); */
list_for_each_fm(t_fm_hdl) {
/* printf("# find %x %x %s\n", t_fm_hdl, t_fm_hdl->read_id((void *)fm_dev_info), t_fm_hdl->logo); */
if (t_fm_hdl->read_id((void *)fm_dev_info) &&
(!memcmp(t_fm_hdl->logo, "fm_inside", strlen(t_fm_hdl->logo)))) {
fm_hdl = t_fm_hdl;
printf("fm find dev %s \n", t_fm_hdl->logo);
return;
}
}
}
/* early_initcall(fm_manage_check_online); */
int fm_manage_init()
{
fm_dev_info = (struct _fm_dev_info *)malloc(sizeof(struct _fm_dev_info));
if (fm_dev_info == NULL) {
printf("fm_dev_init info malloc err!");
return -1;
}
memset(fm_dev_info, 0x00, sizeof(struct _fm_dev_info));
#if TCHFG_SOFT_I2C_ENABLE
fm_dev_info->iic_hdl = fm_dev_platform_data->iic_hdl;
fm_dev_info->iic_delay = fm_dev_platform_data->iic_delay;
iic_init(fm_dev_info->iic_hdl);
#endif
fm_manage_check_online();
if (!fm_hdl) {
printf("fm_manage could not find dev\n");
return -1;
} else {
memcpy(fm_dev_info->logo, fm_hdl->logo, strlen(fm_hdl->logo));
printf("fm_manage found dev %s\n", fm_dev_info->logo);
}
if (fm_hdl) {
fm_hdl->init((void *)fm_dev_info);
}
return 0;
}
int fm_manage_start()
{
if (!fm_hdl || !fm_dev_info) {
printf("fm_manage could not find dev\n");
return -1;
}
if (memcmp(fm_dev_info->logo, "fm_inside", strlen(fm_dev_info->logo))) {//不开启内部收音开启linein
#if (TCFG_FM_INPUT_WAY != LINEIN_INPUT_WAY_ADC)
if (!app_audio_get_volume(APP_AUDIO_STATE_MUSIC)) {
audio_linein_mute(1); //模拟输出时候dac为0也有数据
}
#endif
#if (TCFG_FM_INPUT_WAY == LINEIN_INPUT_WAY_ADC)
linein_dec_open(TCFG_FMIN_LR_CH, 44100);
#else
if (TCFG_FMIN_LR_CH & (BIT(0) | BIT(1))) {
audio_linein0_open(TCFG_FMIN_LR_CH, 1);
} else if (TCFG_FMIN_LR_CH & (BIT(2) | BIT(3))) {
audio_linein1_open(TCFG_FMIN_LR_CH, 1);
} else if (TCFG_FMIN_LR_CH & (BIT(4) | BIT(5))) {
audio_linein2_open(TCFG_FMIN_LR_CH, 1);
}
audio_linein_gain(1); // high gain
#endif
#if (TCFG_FM_INPUT_WAY != LINEIN_INPUT_WAY_ADC)
if (app_audio_get_volume(APP_AUDIO_STATE_MUSIC)) {
audio_linein_mute(0);
app_audio_set_volume(APP_AUDIO_STATE_MUSIC, app_audio_get_volume(APP_AUDIO_STATE_MUSIC), 1);//防止无法调整
}
//模拟输出时候dac为0也有数据
#endif
} else {
fm_dec_open(0xff, 37500);//打开内置收音解码通道
}
return 0;
}
static u16 cur_fmrx_freq = 0;
bool fm_manage_set_fre(u16 fre)
{
if (fm_hdl) {
cur_fmrx_freq = fre;
return fm_hdl->set_fre((void *)fm_dev_info, fre);
}
return 0;
}
u16 fm_manage_get_fre()
{
if (fm_hdl) {
return cur_fmrx_freq;
}
return 0;
}
void fm_manage_close(void)
{
if (fm_hdl) {
fm_hdl->close((void *)fm_dev_info);
fm_hdl = NULL;
}
if (fm_dev_info != NULL) {
free(fm_dev_info);
fm_dev_info = NULL;
}
if (t_fm_hdl && memcmp(t_fm_hdl->logo, "fm_inside", strlen(t_fm_hdl->logo))) { //不使用内部收音开启linein
#if (TCFG_FM_INPUT_WAY == LINEIN_INPUT_WAY_ADC)
linein_dec_close();
#else
if (TCFG_FMIN_LR_CH & (BIT(0) | BIT(1))) {
audio_linein0_close(TCFG_FMIN_LR_CH, 0);
} else if (TCFG_FMIN_LR_CH & (BIT(2) | BIT(3))) {
audio_linein1_close(TCFG_FMIN_LR_CH, 0);
} else if (TCFG_FMIN_LR_CH & (BIT(4) | BIT(5))) {
audio_linein2_close(TCFG_FMIN_LR_CH, 0);
}
#endif
}
}
void fm_manage_mute(u8 mute)
{
if (fm_hdl) {
fm_hdl->mute((void *)fm_dev_info, mute);
}
}
#endif