project:cfg:BoardConfig_IPC: Added fastboot BoardConfig file and firmware post-scripts, distinguishing between the BoardConfigs for Luckfox Pico Pro and Luckfox Pico Max. project:app: Added fastboot_client and rk_smart_door for quick boot applications; updated rkipc app to adapt to the latest media library. media:samples: Added more usage examples. media:rockit: Fixed bugs; removed support for retrieving data frames from VPSS. media:isp: Updated rkaiq library and related tools to support connection to RKISP_Tuner. sysdrv:Makefile: Added support for compiling drv_ko on Luckfox Pico Ultra W using Ubuntu; added support for custom root filesystem. sysdrv:tools:board: Updated Buildroot optional mirror sources, updated some software versions, and stored device tree files and configuration files that undergo multiple modifications for U-Boot and kernel separately. sysdrv:source:mcu: Used RISC-V MCU SDK with RT-Thread system, mainly for initializing camera AE during quick boot. sysdrv:source:uboot: Added support for fastboot; added high baud rate DDR bin for serial firmware upgrades. sysdrv:source:kernel: Upgraded to version 5.10.160; increased NPU frequency for RV1106G3; added support for fastboot. Signed-off-by: luckfox-eng29 <eng29@luckfox.com>
280 lines
7.3 KiB
C
280 lines
7.3 KiB
C
// AO
|
||
|
||
#include <errno.h>
|
||
#include <pthread.h>
|
||
#include <signal.h>
|
||
#include <stdbool.h>
|
||
#include <stdio.h>
|
||
#include <stdlib.h>
|
||
#include <string.h>
|
||
#include <sys/poll.h>
|
||
#include <unistd.h>
|
||
|
||
#include "rk_debug.h"
|
||
#include "rk_defines.h"
|
||
#include "rk_mpi_amix.h"
|
||
#include "rk_mpi_ao.h"
|
||
#include "rk_mpi_mb.h"
|
||
#include "rk_mpi_sys.h"
|
||
|
||
static bool quit = false;
|
||
static void sigterm_handler(int sig) {
|
||
fprintf(stderr, "signal %d\n", sig);
|
||
quit = true;
|
||
}
|
||
|
||
static FILE *ao_file;
|
||
|
||
RK_S32 init_ao_vqe(RK_S32 vqeEnable) {
|
||
RK_S32 ret = RK_SUCCESS;
|
||
AO_VQE_CONFIG_S vqe_config;
|
||
char *pVqeCfgPath = NULL;
|
||
if (vqeEnable == 1)
|
||
pVqeCfgPath = "/oem/usr/share/vqefiles/config_aoagc.json";//agc
|
||
else
|
||
pVqeCfgPath = "/oem/usr/share/vqefiles/config_aovqe.json";//anr howl
|
||
|
||
memset(&vqe_config, 0, sizeof(AO_VQE_CONFIG_S));
|
||
if (pVqeCfgPath != RK_NULL) {
|
||
vqe_config.enCfgMode = AIO_VQE_CONFIG_LOAD_FILE;
|
||
memcpy(vqe_config.aCfgFile, pVqeCfgPath, strlen(pVqeCfgPath));
|
||
}
|
||
|
||
RK_LOGD("enCfgMode = %d", vqe_config.enCfgMode);
|
||
ret = RK_MPI_AO_SetVqeAttr(0, 0, &vqe_config);
|
||
if (ret) {
|
||
RK_LOGE("ao set vqe attr fail, aoChn = %d, reason = %X", 0, ret);
|
||
return ret;
|
||
}
|
||
|
||
ret = RK_MPI_AO_EnableVqe(0, 0);
|
||
if (ret) {
|
||
RK_LOGE("ao enable vqe fail, aoChn = %d, reason = %X", 0, ret);
|
||
return ret;
|
||
}
|
||
|
||
return RK_SUCCESS;
|
||
}
|
||
|
||
RK_S32 ao_set_other(RK_S32 s32SetVolume) {
|
||
printf("\n=======%s=======\n", __func__);
|
||
int s32DevId = 0;
|
||
RK_S32 volume = 0;
|
||
|
||
RK_MPI_AO_SetVolume(s32DevId, s32SetVolume);
|
||
|
||
RK_MPI_AO_GetVolume(s32DevId, &volume);
|
||
RK_LOGI("test info : get volume = %d", volume);
|
||
|
||
AUDIO_TRACK_MODE_E trackMode;
|
||
RK_MPI_AO_GetTrackMode(s32DevId, &trackMode);
|
||
RK_LOGI("test info : get track mode = %d", trackMode);
|
||
|
||
return 0;
|
||
}
|
||
|
||
RK_S32 open_device_ao(RK_S32 s32DeviceSampleRate, RK_S32 s32InputSampleRate,
|
||
RK_S32 channel, RK_S32 u32FrameCnt, RK_S32 vqeEnable) {
|
||
printf("\n=======%s=======\n", __func__);
|
||
RK_S32 result = 0;
|
||
AUDIO_DEV aoDevId = 0;
|
||
AO_CHN aoChn = 0;
|
||
AIO_ATTR_S aoAttr;
|
||
AO_CHN_PARAM_S pstParams;
|
||
|
||
memset(&pstParams, 0, sizeof(AO_CHN_PARAM_S));
|
||
memset(&aoAttr, 0, sizeof(AIO_ATTR_S));
|
||
#ifdef RV1126_RV1109
|
||
/*==============================================================================*/
|
||
//这是RV1126 声卡打开设置,RV1106设置无效,可以不设置
|
||
result = RK_MPI_AMIX_SetControl(aoDevId, "Playback Path", (char *)"SPK");
|
||
if (result != RK_SUCCESS) {
|
||
RK_LOGE("ao set Playback Path fail, reason = %x", result);
|
||
return RK_FAILURE;
|
||
}
|
||
#endif
|
||
sprintf((char *)aoAttr.u8CardName, "%s", "hw:0,0");
|
||
|
||
aoAttr.soundCard.channels = 2;
|
||
aoAttr.soundCard.sampleRate = s32DeviceSampleRate;
|
||
aoAttr.soundCard.bitWidth = AUDIO_BIT_WIDTH_16;
|
||
|
||
aoAttr.enBitwidth = AUDIO_BIT_WIDTH_16;
|
||
aoAttr.enSamplerate = (AUDIO_SAMPLE_RATE_E)s32InputSampleRate;
|
||
|
||
if (channel == 1)
|
||
aoAttr.enSoundmode = AUDIO_SOUND_MODE_MONO;
|
||
else if (channel == 2)
|
||
aoAttr.enSoundmode = AUDIO_SOUND_MODE_STEREO;
|
||
else {
|
||
RK_LOGE("unsupport = %d", channel);
|
||
return RK_FAILURE;
|
||
}
|
||
|
||
aoAttr.u32PtNumPerFrm = u32FrameCnt; // 1024
|
||
//以下参数没有特殊需要,无需修改
|
||
aoAttr.u32FrmNum = 4;
|
||
aoAttr.u32EXFlag = 0;
|
||
aoAttr.u32ChnCnt = 2;
|
||
|
||
RK_MPI_AO_SetPubAttr(aoDevId, &aoAttr);
|
||
RK_MPI_AO_Enable(aoDevId);
|
||
/*==============================================================================*/
|
||
pstParams.enLoopbackMode = AUDIO_LOOPBACK_NONE;
|
||
result = RK_MPI_AO_SetChnParams(aoDevId, aoChn, &pstParams);
|
||
if (result != RK_SUCCESS) {
|
||
RK_LOGE("ao set channel params, aoChn = %d", aoChn);
|
||
return RK_FAILURE;
|
||
}
|
||
|
||
if (channel == 1)
|
||
RK_MPI_AO_SetTrackMode(aoDevId, AUDIO_TRACK_OUT_STEREO);
|
||
else
|
||
RK_MPI_AO_SetTrackMode(aoDevId, AUDIO_TRACK_NORMAL);
|
||
|
||
if (vqeEnable)
|
||
init_ao_vqe(vqeEnable);
|
||
/*==============================================================================*/
|
||
result = RK_MPI_AO_EnableChn(aoDevId, aoChn);
|
||
if (result != 0) {
|
||
RK_LOGE("ao enable channel fail, aoChn = %d, reason = %x", aoChn, result);
|
||
return RK_FAILURE;
|
||
}
|
||
/*==============================================================================*/
|
||
// set sample rate of input data
|
||
printf("********RK_MPI_AO_EnableReSmp*********\n");
|
||
result =
|
||
RK_MPI_AO_EnableReSmp(aoDevId, aoChn, (AUDIO_SAMPLE_RATE_E)s32InputSampleRate);
|
||
if (result != 0) {
|
||
RK_LOGE("ao enable channel fail, reason = %x, aoChn = %d", result, aoChn);
|
||
return RK_FAILURE;
|
||
}
|
||
|
||
ao_set_other(100);
|
||
|
||
return RK_SUCCESS;
|
||
}
|
||
|
||
static RK_CHAR optstr[] = "?::d:r:i:t:R:c:v:";
|
||
static void print_usage(const RK_CHAR *name) {
|
||
printf("usage example:\n");
|
||
printf("\t%s -r 16000 -R 16000 -i /tmp/16000.pcm\n",
|
||
name); //正常情况-r -R 配置成一样的就好
|
||
printf("\t-r: device rate, Default:16000\n");
|
||
printf("\t-R: input rate, Default:16000\n");
|
||
printf("\t-i: input path, Default:\"/tmp/16000.pcm\"\n");
|
||
}
|
||
|
||
int main(int argc, char *argv[]) {
|
||
RK_S32 s32DeviceSampleRate = 16000; // device samplerate
|
||
RK_S32 s32InputSampleRate = 16000; // input pcm samplerate
|
||
RK_S32 u32Channel = 1;
|
||
RK_U32 u32FrameCnt = 1024;
|
||
RK_S32 vqeEnable = 1;
|
||
RK_CHAR *pInPath = "/tmp/16000.pcm";
|
||
int c;
|
||
|
||
while ((c = getopt(argc, argv, optstr)) != -1) {
|
||
switch (c) {
|
||
case 'r':
|
||
s32DeviceSampleRate = atoi(optarg);
|
||
break;
|
||
case 'R':
|
||
s32InputSampleRate = atoi(optarg);
|
||
break;
|
||
case 'c':
|
||
u32Channel = atoi(optarg);
|
||
break;
|
||
case 'i':
|
||
pInPath = optarg;
|
||
break;
|
||
case 'v':
|
||
vqeEnable = atoi(optarg);
|
||
break;
|
||
case '?':
|
||
default:
|
||
print_usage(argv[0]);
|
||
return -1;
|
||
}
|
||
}
|
||
|
||
printf("#Device SampleRate: %d\n", s32DeviceSampleRate);
|
||
printf("#Input SampleRate: %d\n", s32InputSampleRate);
|
||
printf("#Input Channel: %d\n", u32Channel);
|
||
printf("#Frame Count: %d\n", u32FrameCnt);
|
||
printf("#Input Path: %s\n", pInPath);
|
||
printf("#Vqe enable: %d\n", vqeEnable);
|
||
|
||
if (pInPath) {
|
||
ao_file = fopen(pInPath, "rb");
|
||
if (!ao_file) {
|
||
printf("ERROR: open file: %s fail, exit\n", pInPath);
|
||
return -1;
|
||
}
|
||
}
|
||
|
||
signal(SIGINT, sigterm_handler);
|
||
|
||
RK_MPI_SYS_Init();
|
||
open_device_ao(s32DeviceSampleRate, s32InputSampleRate, u32Channel, u32FrameCnt, vqeEnable);
|
||
|
||
printf("%s initial finish\n", __func__);
|
||
|
||
RK_U8 *srcData = RK_NULL;
|
||
RK_S32 srcSize = 0;
|
||
RK_U64 timeStamp = 0;
|
||
AUDIO_FRAME_S frame;
|
||
RK_S32 result = 0;
|
||
RK_S32 s32MilliSec = -1;
|
||
srcData = calloc(1024, sizeof(RK_U8));
|
||
memset(srcData, 0, 1024);
|
||
while (!quit) {
|
||
srcSize = fread(srcData, 1, 1024, ao_file);
|
||
frame.u32Len = srcSize;
|
||
frame.u64TimeStamp = timeStamp++;
|
||
frame.enBitWidth = AUDIO_BIT_WIDTH_16;
|
||
if (u32Channel == 1)
|
||
frame.enSoundMode = AUDIO_SOUND_MODE_MONO;
|
||
else if (u32Channel == 2)
|
||
frame.enSoundMode = AUDIO_SOUND_MODE_STEREO;
|
||
|
||
frame.bBypassMbBlk = RK_FALSE;
|
||
|
||
MB_EXT_CONFIG_S extConfig;
|
||
memset(&extConfig, 0, sizeof(extConfig));
|
||
extConfig.pOpaque = srcData;
|
||
extConfig.pu8VirAddr = srcData;
|
||
extConfig.u64Size = srcSize;
|
||
RK_MPI_SYS_CreateMB(&(frame.pMbBlk), &extConfig);
|
||
__RETRY:
|
||
result = RK_MPI_AO_SendFrame(0, 0, &frame, s32MilliSec);
|
||
if (result < 0) {
|
||
RK_LOGE("send frame fail, result = %d, TimeStamp = %lld, s32MilliSec = %d",
|
||
result, frame.u64TimeStamp, s32MilliSec);
|
||
goto __RETRY;
|
||
}
|
||
RK_MPI_MB_ReleaseMB(frame.pMbBlk);
|
||
|
||
if (srcSize <= 0) {
|
||
printf("eof");
|
||
RK_MPI_AO_WaitEos(0, 0, s32MilliSec);
|
||
break;
|
||
}
|
||
}
|
||
|
||
if (ao_file) {
|
||
fclose(ao_file);
|
||
ao_file = RK_NULL;
|
||
}
|
||
free(srcData);
|
||
|
||
if (vqeEnable)
|
||
RK_MPI_AO_DisableVqe(0, 0);
|
||
|
||
RK_MPI_AO_DisableReSmp(0, 0);
|
||
RK_MPI_AO_DisableChn(0, 0);
|
||
RK_MPI_AO_Disable(0);
|
||
RK_MPI_SYS_Exit();
|
||
return 0;
|
||
}
|