/* Copyright 2020 Rockchip Electronics Co. LTD * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #undef DBG_MOD_ID #define DBG_MOD_ID RK_ID_SYS #include #include #include #include "rk_debug.h" #include "rk_mpi_sys.h" #include "rk_mpi_ao.h" #include "rk_mpi_adec.h" #include "argparse.h" typedef struct _rkTestSysCtx { RK_S32 s32LoopCount; RK_S32 s32DevId; RK_S32 s32SrcChnId; RK_S32 s32DstChnNum; } TEST_SYS_CTX_S; RK_S32 test_ao_dev_init(TEST_SYS_CTX_S *pstCtx) { AUDIO_DEV aoDevId = (AUDIO_DEV)pstCtx->s32DevId; AIO_ATTR_S aoAttr; memset(&aoAttr, 0, sizeof(AIO_ATTR_S)); aoAttr.enBitwidth = AUDIO_BIT_WIDTH_16; aoAttr.enSamplerate = AUDIO_SAMPLE_RATE_44100; aoAttr.enSoundmode = AUDIO_SOUND_MODE_MONO; aoAttr.u32FrmNum = 512; aoAttr.u32PtNumPerFrm = 1024; aoAttr.u32EXFlag = 0; aoAttr.u32ChnCnt = 2; RK_MPI_AO_SetPubAttr(aoDevId, &aoAttr); RK_MPI_AO_Enable(aoDevId); return RK_SUCCESS; } RK_S32 test_ao_dev_deinit(TEST_SYS_CTX_S *pstCtx) { RK_S32 s32Ret = RK_SUCCESS; s32Ret = RK_MPI_AO_Disable(pstCtx->s32DevId); if (s32Ret != RK_SUCCESS) { RK_LOGE("failed to disable ao dev, err: %d", s32Ret); return RK_FAILURE; } return s32Ret; } RK_S32 test_adec_create_channel(TEST_SYS_CTX_S *pstCtx, RK_S32 s32ChnId) { RK_S32 s32Ret = RK_SUCCESS; ADEC_CHN_ATTR_S stAdecAttr; ADEC_CHN AdChn = (ADEC_CHN)s32ChnId; stAdecAttr.enType = RK_AUDIO_ID_MP2; stAdecAttr.enMode = ADEC_MODE_PACK; stAdecAttr.stAdecCodec.u32Channels = 2; stAdecAttr.stAdecCodec.u32SampleRate = 44100; stAdecAttr.extraDataSize = 0; stAdecAttr.extraData = RK_NULL; s32Ret = RK_MPI_ADEC_CreateChn(AdChn, &stAdecAttr); if (s32Ret) { RK_LOGE("failed to create adec chn %d, err %d", AdChn, s32Ret); return RK_FAILURE; } return s32Ret; } RK_S32 test_adec_destroy_channel(TEST_SYS_CTX_S *pstCtx, RK_S32 s32ChnId) { RK_S32 s32Ret = RK_SUCCESS; RK_S32 s32DevId = pstCtx->s32DevId; ADEC_CHN AdChn = (ADEC_CHN)s32ChnId; s32Ret = RK_MPI_ADEC_DestroyChn(AdChn); if (s32Ret != RK_SUCCESS) { RK_LOGE("failed to destroy adec channel(%d), err: %d", AdChn, s32Ret); return RK_FAILURE; } return s32Ret; } RK_S32 test_ao_enable_channel(TEST_SYS_CTX_S *pstCtx, RK_S32 s32ChnId) { RK_S32 s32Ret = RK_SUCCESS; ADEC_CHN AdChn = (ADEC_CHN)s32ChnId; s32Ret = RK_MPI_AO_EnableChn(pstCtx->s32DevId, AdChn); if (s32Ret != 0) { RK_LOGE("failed to enable ao chn %d, err %d", AdChn, s32Ret); return RK_FAILURE; } return s32Ret; } RK_S32 test_ao_disable_channel(TEST_SYS_CTX_S *pstCtx, RK_S32 s32ChnId) { RK_S32 s32Ret = RK_SUCCESS; AO_CHN AoChn = (AO_CHN)s32ChnId; s32Ret = RK_MPI_AO_DisableChn(pstCtx->s32DevId, AoChn); if (s32Ret != RK_SUCCESS) { RK_LOGE("failed to disable ao channel(%d), err: %d", AoChn, s32Ret); return RK_FAILURE; } return RK_SUCCESS; } RK_S32 test_bind_adec_ao(TEST_SYS_CTX_S *pstCtx, RK_S32 s32SrcChnId, RK_S32 s32DstChnId) { RK_S32 s32Ret = RK_SUCCESS; RK_S32 s32DevId = pstCtx->s32DevId; MPP_CHN_S stSrcChn, stDstChn; stSrcChn.enModId = RK_ID_ADEC; stSrcChn.s32DevId = s32DevId; stSrcChn.s32ChnId = s32SrcChnId; stDstChn.enModId = RK_ID_AO; stDstChn.s32DevId = s32DevId; stDstChn.s32ChnId = s32DstChnId; s32Ret = RK_MPI_SYS_Bind(&stSrcChn, &stDstChn); return s32Ret; } RK_S32 test_unbind_adec_ao(TEST_SYS_CTX_S *pstCtx, RK_S32 s32SrcChnId, RK_S32 s32DstChnId) { RK_S32 s32Ret = RK_SUCCESS; RK_S32 s32DevId = pstCtx->s32DevId; MPP_CHN_S stSrcChn, stDstChn; stSrcChn.enModId = RK_ID_ADEC; stSrcChn.s32DevId = s32DevId; stSrcChn.s32ChnId = s32SrcChnId; stDstChn.enModId = RK_ID_AO; stDstChn.s32DevId = s32DevId; stDstChn.s32ChnId = s32DstChnId; s32Ret = RK_MPI_SYS_UnBind(&stSrcChn, &stDstChn); return s32Ret; } RK_S32 test_mpi_sys_get_bind_by_src(TEST_SYS_CTX_S *pstCtx, RK_S32 s32ChnId) { RK_S32 s32Ret = RK_SUCCESS; MPP_CHN_S stSrcChn; MPP_BIND_DEST_S pstBindDest; memset(&stSrcChn, 0, sizeof(MPP_CHN_S)); memset(&pstBindDest, 0, sizeof(MPP_BIND_DEST_S)); stSrcChn.enModId = RK_ID_ADEC; stSrcChn.s32DevId = pstCtx->s32DevId; stSrcChn.s32ChnId = s32ChnId; s32Ret = RK_MPI_SYS_GetBindbySrc(&stSrcChn, &pstBindDest); if (s32Ret == RK_SUCCESS) { for (RK_S32 i=0; i < pstBindDest.u32Num; i++) { MPP_CHN_S *pstDstChn = &pstBindDest.astMppChn[i]; RK_LOGD("get dst channel(modId=%d, devId=%d, chnId=%d)", pstDstChn->enModId, pstDstChn->s32DevId, pstDstChn->s32ChnId); } } else { RK_LOGE("failed to RK_MPI_SYS_GetBindbySrc"); } return s32Ret; } RK_S32 test_mpi_sys_get_bind_by_dest(TEST_SYS_CTX_S *pstCtx, RK_S32 s32ChnId) { RK_S32 s32Ret = RK_SUCCESS; MPP_CHN_S stSrcChn; MPP_CHN_S stDstChn; memset(&stSrcChn, 0, sizeof(MPP_CHN_S)); memset(&stDstChn, 0, sizeof(MPP_CHN_S)); stDstChn.enModId = RK_ID_AO; stDstChn.s32DevId = pstCtx->s32DevId; stDstChn.s32ChnId = s32ChnId; s32Ret = RK_MPI_SYS_GetBindbyDest(&stDstChn, &stSrcChn); if (s32Ret == RK_SUCCESS) { RK_LOGD("get src channel(modId=%d, devId=%d, chnId=%d)", stSrcChn.enModId, stSrcChn.s32DevId, stSrcChn.s32ChnId); } else { RK_LOGE("failed to RK_MPI_SYS_GetBindbyDest"); } return s32Ret; } RK_S32 unit_test_mpi_sys_bind(TEST_SYS_CTX_S *pstCtx) { RK_S32 s32Ret = RK_SUCCESS; RK_S32 s32SrcChnId = pstCtx->s32SrcChnId; RK_S32 s32DstNumChn = pstCtx->s32DstChnNum; // init Ao device s32Ret = test_ao_dev_init(pstCtx); // create adec channel s32Ret = test_adec_create_channel(pstCtx, s32SrcChnId); if (s32Ret != RK_SUCCESS) { goto __FAILED_ADEC; } // enable ao channel for (RK_S32 s32DstChnId = 0; s32DstChnId < s32DstNumChn; s32DstChnId++) { s32Ret = test_ao_enable_channel(pstCtx, s32DstChnId); if (s32Ret != RK_SUCCESS) { goto __FAILED_AO; } // bind adec->ao s32Ret = test_bind_adec_ao(pstCtx, s32SrcChnId, s32DstChnId); if (s32Ret == RK_SUCCESS) { RK_LOGD("succeed to bind ADEC(%d) AO(%d)", s32SrcChnId, s32DstChnId); } } // test RK_MPI_SYS_GetBindbyDest/GetBindbySrc s32Ret = test_mpi_sys_get_bind_by_src(pstCtx, s32SrcChnId); for (RK_S32 s32DstChnId = 0; s32DstChnId < s32DstNumChn; s32DstChnId++) { s32Ret = test_mpi_sys_get_bind_by_dest(pstCtx, s32DstChnId); } __FAILED_AO: for (RK_S32 s32DstChnId = 0; s32DstChnId < s32DstNumChn; s32DstChnId++) { // unbind adec->ao s32Ret = test_unbind_adec_ao(pstCtx, s32SrcChnId, s32DstChnId); if (s32Ret == RK_SUCCESS) { RK_LOGD("succeed to unbind ADEC(%d) AO(%d)", s32SrcChnId, s32DstChnId); } // disable ao channel test_ao_disable_channel(pstCtx, s32DstChnId); } test_adec_destroy_channel(pstCtx, s32SrcChnId); __FAILED_ADEC: // deinit Ao device s32Ret = test_ao_dev_deinit(pstCtx); return s32Ret; } RK_S32 unit_test_mpi_sys(TEST_SYS_CTX_S *pstCtx) { RK_S32 s32Ret = RK_SUCCESS; RK_S32 loopCount = pstCtx->s32LoopCount; do { s32Ret = unit_test_mpi_sys_bind(pstCtx); loopCount--; RK_LOGI("looping times %d", pstCtx->s32LoopCount - loopCount); } while (loopCount > 0); return s32Ret; } static const char *const usages[] = { "./rk_mpi_sys_test...", NULL, }; RK_S32 main(RK_S32 argc, const char **argv) { RK_S32 s32Ret = RK_SUCCESS; TEST_SYS_CTX_S stCtx; memset(&stCtx, 0, sizeof(TEST_SYS_CTX_S)); stCtx.s32LoopCount = 1; stCtx.s32DevId = 0; stCtx.s32SrcChnId = 0; stCtx.s32DstChnNum = 1; struct argparse_option options[] = { OPT_HELP(), OPT_GROUP("basic options:"), OPT_INTEGER('n', "loop_count", &(stCtx.s32LoopCount), "loop running count. default(1)", NULL, 0, 0), OPT_INTEGER('\0', "device_id", &(stCtx.s32DevId), "MODULE device id. default(0)", NULL, 0, 0), OPT_INTEGER('\0', "src_channel_id", &(stCtx.s32SrcChnId), "source MODULE channel id. default(0)", NULL, 0, 0), OPT_INTEGER('\0', "dst_channel_count", &(stCtx.s32DstChnNum), "the count of dst MODULE channel. default(1)", NULL, 0, 0), OPT_END(), }; struct argparse argparse; argparse_init(&argparse, options, usages, 0); argparse_describe(&argparse, "\nselect a test case to run.", "\nuse --help for details."); argc = argparse_parse(&argparse, argc, argv); s32Ret = RK_MPI_SYS_Init(); if (s32Ret != RK_SUCCESS) { return s32Ret; } s32Ret = unit_test_mpi_sys(&stCtx); if (s32Ret != RK_SUCCESS) { goto __FAILED; } s32Ret = RK_MPI_SYS_Exit(); if (s32Ret != RK_SUCCESS) { return s32Ret; } RK_LOGI("test running ok."); return RK_SUCCESS; __FAILED: RK_MPI_SYS_Exit(); RK_LOGE("test running failed!"); return s32Ret; }