SA8450 camera 源码分析

目录

代码路径:/apps/qnx_ap/AMSS/multimedia/camera_safe/qcd

Initialize the IFE and CSI Phy devices

 初始化IFE 

初始化 IFE CSID VFE 的结构体

获取IFE CSID base addr 的vaddr

VFEDriver_Init

RegisterISTCallback

HwMgr_CSIPHYOpen


代码路径:/apps/qnx_ap/AMSS/multimedia/camera_safe/qcd

Initialize the IFE and CSI Phy devices

 Initialization function for the Camera HW. During initialization, the HW Topology Manager will* query information about the current configuration through the Camera Config Handler to be used* for initializing its internal context. It will also open each of the CSI Phy and IFE devices* used the system through the HW Manager block.** @return* Error code defined in QCDStatus_e**************************************************************************************************/
QCDStatus_e HTM_Init (void)
{QCDStatus_e status = QCD_ERROR_GENERAL_FAILURE;CameraConfigGetInfo_t configInfo = {0};CameraConfigGetInfo_t mappingInfo = {0};QCD_LOG(LOG_HTM,LOG_LEVEL_HIGH,"HTM Init Starting");configInfo.infoType = CAMERA_CONFIG_INFO_ALL;mappingInfo.infoType = CAMERA_CONFIG_INFO_INPUT_MAPPINGS;if (TRUE == g_htmInitialized){QCD_LOG(LOG_HTM,LOG_LEVEL_ERROR,"HTM already initialized");status = QCD_ERROR_INVALID_STATE;}else if((QCD_SUCCESS != CC_GetInfo(&configInfo))|| (NULL == configInfo.info.pConfigInfo)){QCD_LOG(LOG_HTM,LOG_LEVEL_FATAL,"Unable to get config from CC");status = QCD_ERROR_NOT_FOUND;}else if((QCD_SUCCESS != CC_GetInfo(&mappingInfo))|| (NULL == mappingInfo.info.pInpuMappings)){QCD_LOG(LOG_HTM,LOG_LEVEL_FATAL,"Unable to get mappings from CC");status = QCD_ERROR_NOT_FOUND;}/** 1. Initialize the IFE and CSI Phy devices*/else{uint32_t csiIdx = 0;QCDStatus_e hwInitStatus = QCD_SUCCESS;g_pBoardInfo = &configInfo.info.pConfigInfo->boardConfig;g_pCsiDevicesList = &configInfo.info.pConfigInfo->csiDevices;g_pInputMappingList = mappingInfo.info.pInpuMappings;for (csiIdx = 0; csiIdx < g_pCsiDevicesList->numCsiDevs; csiIdx++){uint64_t ifeBitmap = g_pCsiDevicesList->csiDevConfig[csiIdx].ifeMap;uint32_t ifeIdx = 0;for (ifeIdx = 0; ifeIdx < g_pCsiDevicesList->csiDevConfig[csiIdx].numIfeMap; ifeIdx++){uint32_t nextIfeId = GET_IFE_ID_FROM_BITMAP(ifeBitmap);QCD_LOG(LOG_HTM,LOG_LEVEL_MEDIUM,"Opening IFE ID %u",nextIfeId);hwInitStatus = HwMgr_IFEOpen(nextIfeId);if (QCD_SUCCESS != hwInitStatus){QCD_LOG(LOG_HTM,LOG_LEVEL_FATAL,"IFE ID %u open failed",nextIfeId);break;}else{QCD_LOG(LOG_HTM,LOG_LEVEL_MEDIUM,"IFE ID %u open success",nextIfeId);// Set a global context to check whether IFE device was opened successfully.g_ifeDeviceOpened[nextIfeId] = TRUE;}ifeBitmap = ifeBitmap >> IFE_ID_SIZE_IN_BITS;}if (QCD_SUCCESS != hwInitStatus){// No use intializing CSI PHY if IFE init has failedbreak;}QCD_LOG(LOG_HTM,LOG_LEVEL_MEDIUM,"Opening CSI Phy Device %u",g_pCsiDevicesList->csiDevConfig[csiIdx].csiId);hwInitStatus = HwMgr_CSIPHYOpen(g_pCsiDevicesList->csiDevConfig[csiIdx].csiId);if (QCD_SUCCESS != hwInitStatus){QCD_LOG(LOG_HTM,LOG_LEVEL_FATAL,"CSIPhy ID %u open failed",g_pCsiDevicesList->csiDevConfig[csiIdx].csiId);// Do not conitnue with the loop if a CSI open failsbreak;}else{// Set a global context to check whether CSI device was opened successfully.g_csiDeviceOpened[g_pCsiDevicesList->csiDevConfig[csiIdx].csiId] = TRUE;}}/** 2. Get input mappings**/if (QCD_SUCCESS == hwInitStatus){QCD_LOG(LOG_HTM,LOG_LEVEL_MEDIUM,"Initializing input mappings");uint32_t inputIdx = 0; //Index into a list of input IDsuint32_t pathIdx = 0;  //Index into a list of paths for that input IDuint32_t chanIndex = 0; //Index into the list of channels for each input ID/* Input mappings are structured in a way where one input ID can reserve a number* of RDIs based on the number of channels that input has. When a client opens* an input, the next available RDI defined in the mapping list is reserved for* it.*/for (inputIdx = 0; inputIdx < g_pInputMappingList->numInputs; inputIdx++){for (pathIdx = 0; pathIdx < g_pInputMappingList->inputs[inputIdx].numReservedPaths;pathIdx++){for(chanIndex = 0;chanIndex <g_pInputMappingList->inputs[inputIdx].reservedPaths[pathIdx].numPaths;chanIndex++){CameraInputPath_t const *pInputPath =&g_pInputMappingList->inputs[inputIdx].reservedPaths[pathIdx].inputPath[chanIndex];IfeLiteMappingInfo_t *pMappingInfo =&g_ifeMappingInfo[pInputPath->ifeId];/*Not checking these pointers for NULL here, trusting the ASIL B* block Camera Config on this*/QCD_LOG(LOG_HTM,LOG_LEVEL_MEDIUM,"Initializing inputIdx %u DevId %u PortId %u Path Idx %u QCarCam"\"id %u chanIndex %u ife id %u rdi %u csiId %u channel id %u",inputIdx,g_pInputMappingList->inputs[inputIdx].inputDevId,g_pInputMappingList->inputs[inputIdx].inputDevPortId,pathIdx,g_pInputMappingList->inputs[inputIdx].qCarCamId,chanIndex,pInputPath->ifeId,pInputPath->rdiId,pInputPath->csiId,pInputPath->channelId);if (NULL != pMappingInfo->rdiInfo[pInputPath->rdiId].inputPath){/** If we are here, someone entered wrong configurations because* we are assuming only one path per RDI defined*/QCD_LOG(LOG_HTM,LOG_LEVEL_CRIT,"Duplicate RDI %u input path found check camera configs",pInputPath->rdiId);hwInitStatus = QCD_ERROR_NOT_PERMITTED;break;}QCD_LOG(LOG_HTM,LOG_LEVEL_HIGH,"rdiInfo[%u].qCarCamId = %u",pInputPath->rdiId,g_pInputMappingList->inputs[inputIdx].qCarCamId);pMappingInfo->rdiInfo[pInputPath->rdiId].inputPath = pInputPath;pMappingInfo->rdiInfo[pInputPath->rdiId].qCarCamId =g_pInputMappingList->inputs[inputIdx].qCarCamId;pMappingInfo->rdiInfo[pInputPath->rdiId].state = RDI_STATUS_READY;}}if (QCD_SUCCESS != hwInitStatus){//Don't continue if a mapping failsbreak;}}}if (QCD_SUCCESS == hwInitStatus){if (QCD_SUCCESS != OSAL_CriticalSectionCreate(&g_mutIfeMap)){QCD_LOG(LOG_HTM,LOG_LEVEL_FATAL,"Failed to initialize mutex for mappings");hwInitStatus = QCD_ERROR_GENERAL_FAILURE;}}status = hwInitStatus;}if(QCD_SUCCESS != status){(void)HTM_DeInit();}else{g_htmInitialized = TRUE;}QCD_LOG(LOG_HTM,LOG_LEVEL_MEDIUM,"HTM Init Done");return status;
}

 //获取boardconfig信息

/**********************************************************************************************//**
@briefReturns a pointer to the camera configuration structure, based on the requested info type.@param pInfo[In/Out] Contains the type of info requested and the pointer to the associated structure.@returnStatus as defined in QCDStatus_e
**************************************************************************************************/
QCDStatus_e CC_GetInfo(CameraConfigGetInfo_t* pInfo)
{QCDStatus_e result = QCD_ERROR_GENERAL_FAILURE;uint32_t socIdx = 0U;CameraConfigGlobalInfo_t const *pSetupConfig = NULL;if (NULL == pInfo){result = QCD_ERROR_BAD_PARAM;QCD_LOG(LOG_CONFIG_MGR, LOG_LEVEL_ERROR, "NULL Input param");}else if (NULL != (pSetupConfig = CC_GetGlobalConfig())){    //获取boardconfig信息// Retrieve Configurationswitch(pInfo->infoType){  //初始化时configInfo.infoType = CAMERA_CONFIG_INFO_ALL;case CAMERA_CONFIG_INFO_ALL:{for (socIdx = 0U; socIdx < pSetupConfig->numSocConfig; socIdx++){if (pSetupConfig->socId == pSetupConfig->pConfigInfo[socIdx]->socId){pInfo->info.pConfigInfo =pSetupConfig->pConfigInfo[socIdx];result = QCD_SUCCESS;break;}}break;}case CAMERA_CONFIG_INFO_BOARD:{for (socIdx = 0U; socIdx < pSetupConfig->numSocConfig; socIdx++){if (pSetupConfig->socId == pSetupConfig->pConfigInfo[socIdx]->socId){pInfo->info.pBoardInfo =&pSetupConfig->pConfigInfo[socIdx]->boardConfig;result = QCD_SUCCESS;break;}}break;}case CAMERA_CONFIG_INFO_CSI_DEVICES:{for (socIdx = 0U; socIdx < pSetupConfig->numSocConfig; socIdx++){if (pSetupConfig->socId == pSetupConfig->pConfigInfo[socIdx]->socId){pInfo->info.pCsiDevices =&pSetupConfig->pConfigInfo[socIdx]->csiDevices;result = QCD_SUCCESS;break;}}break;}case CAMERA_CONFIG_INFO_INPUT_MAPPINGS:{for (socIdx = 0U; socIdx < pSetupConfig->numSocConfig; socIdx++){if (pSetupConfig->socId == pSetupConfig->pConfigInfo[socIdx]->socId){pInfo->info.pInpuMappings =&pSetupConfig->pConfigInfo[socIdx]->inputMappings;result = QCD_SUCCESS;break;}}break;}default:{result = QCD_ERROR_NOT_SUPPORTED;QCD_LOG(LOG_CONFIG_MGR, LOG_LEVEL_ERROR, "Invalid Info type %d",pInfo->infoType);break;}}}else{//If we are here there is a memory corruption for sureQCD_LOG(LOG_CONFIG_MGR, LOG_LEVEL_ERROR, "Null pointer to CC global configs");}return result;
}

/**********************************************************************************************//**
@briefGetter function for camera configs@returnPointer to the camera config function
**************************************************************************************************/
CameraConfigGlobalInfo_t const *CC_GetGlobalConfig(void)
{return pSetupConfig;
}

 解析xml配置获取boardconfig信息

/**********************************************************************************************//**
@briefInitialize Camera Config reader.If FEAT_ENABLE_XML_CONFIGS is enabled:The function will first try to call the xml configparser to see if valid qcd_config xml file is presentIf the file is present:If the file is invalid, an error will be returnedOtherwise, the xml configs will be used for camera configurationsIf the file is not present:The function will continue reading the hard-coded camera configs for this chip ID@returnError code defined in QCDStatus_e
**************************************************************************************************/
QCDStatus_e CCR_Init(void)
{QCDStatus_e result = QCD_ERROR_GENERAL_FAILURE;PlatformChipId_e chipId = CHIP_ID_INVALID;pSetupConfig->pConfigInfo[0] = &gSoc0Config;pSetupConfig->pConfigInfo[1] = &gSoc1Config;/* If the XML feature is enabled, the reader will  first try to read camera configs from* the xml:*  If the XML config file is not present*      Continue with reading the hard-coded camera configs*  If the XML config file is present*      If the XML is invalid or there is an error reading the xml*          Report and error and return failure to initializer*  If the feature is disabled, continue reading the hard-coded configs*/
#ifdef FEAT_ENABLE_XML_CONFIGSresult = CC_ParseXMLFile(&gSetupConfig);if ((QCD_SUCCESS != result) && (QCD_ERROR_FILE_NOT_FOUND != result)){CC_ERR("ParseXMLFile failed with result 0x%x",result);}else if (QCD_ERROR_FILE_NOT_FOUND == result)
#endif{// Read chip id for this chipresult = PLM_GetInfo(GET_CHIP_ID, (void*)&chipId, sizeof(chipId));if (result != QCD_SUCCESS){QCD_LOG(LOG_CONFIG_MGR, LOG_LEVEL_ERROR,"PLM_GetInfo failed with result %d",result);}if (QCD_SUCCESS == result){switch(chipId){case CHIP_ID_SA8540P:{result = CCR_GetSA8540ConfigTable(&pSetupConfig);break;}case CHIP_ID_SA8295P:{result = CCR_GetSA8295ConfigTable(&pSetupConfig);break;}default:{QCD_LOG(LOG_CONFIG_MGR, LOG_LEVEL_ERROR,"Wrong chip ID %d for Get configuration",chipId);break;}}}}
#ifdef FEAT_ENABLE_XML_CONFIGSelse{// Mandatory misra else. Its a success path so no need to put a log here.}
#endifif (QCD_SUCCESS == result){// Read SOC id on which this image is runningresult = PLM_GetInfo(GET_SOC_ID,&pSetupConfig->socId,sizeof(pSetupConfig->socId));if (result != QCD_SUCCESS){QCD_LOG(LOG_CONFIG_MGR, LOG_LEVEL_ERROR,"PLM_GetInfo failed with result 0x%08x",result);}}return result;
}

@param pInfoPointer to the global Config Info struct to be populated@returnError code defined in QCDStatus_eQCD_ERROR_FILE_NOT found if the xml does not exist.
**************************************************************************************************/
QCDStatus_e CC_ParseXMLFile(CameraConfigGlobalInfo_t *pInfo)
{QCDStatus_e result = QCD_ERROR_LAST;xmlDocPtr pXmlDoc = NULL;xmlNodePtr pCur = NULL;const char* filename = CAMERA_CONFIG_XML_FILE;QCD_LOG(LOG_CONFIG_MGR, LOG_LEVEL_HIGH,"Start Parsing Camera Config XML File %s",filename);if (0 != access(filename, R_OK)){/* XML based config is an optional feature for development. Do not need* to log this as an error as this is expected. */QCD_LOG(LOG_CONFIG_MGR, LOG_LEVEL_HIGH,"QCD Config XML file %s not available, ""falling back to default configuration",filename);result = QCD_ERROR_FILE_NOT_FOUND;}else if (NULL == (pXmlDoc = xmlParseFile(filename))){CC_ERR("CameraConfig file %s not parsed successfully",filename);result = QCD_ERROR_NOT_SUPPORTED;}else if (QCD_SUCCESS != (result = ValidateConfigXml(pXmlDoc))){CC_ERR("CameraConfig xml validation failed. Err 0x%08x",result);}else if (NULL == (pCur = xmlDocGetRootElement(pXmlDoc))){CC_ERR("Empty CameraConfig file %s",filename);result = QCD_ERROR_NOT_SUPPORTED;}else if (QCD_SUCCESS != (result = CC_XmlGetUint32(&pInfo->version,"version",pCur,FALSE,0U))){CC_ERR("Unable to get version. Err: 0x%08x", result);}else if (CAM_CFG_XML_VERSION != pInfo->version){CC_ERR("Unsupported CameraConfig version 0x%x (require 0x%x)",pInfo->version,CAM_CFG_XML_VERSION);result = QCD_ERROR_INVALID_CONFIG;}else{pCur = pCur->xmlChildrenNode;uint32_t idx = 0U;while ((NULL != pCur) && (idx < CAM_CFG_MAX_NUM_SOC)){if (0 == (xmlStrcmp(pCur->name, (const xmlChar *) "soc"))){if (QCD_SUCCESS == (result = CC_XmlGetUint32(&pInfo->pConfigInfo[idx]->socId,"id",pCur,FALSE,0U))){if (QCD_SUCCESS != (result = ParseSOCConfig(pCur,pInfo->pConfigInfo[idx],pInfo->pConfigInfo[idx]->socId))){CC_ERR("SoC ID %u xml config parsing failed",pInfo->pConfigInfo[idx]->socId);break;}idx++;}else{CC_ERR("Unable to get SoC ID. Err 0x%08x", result);break;}}pCur = pCur->next;}if (QCD_SUCCESS == result){pInfo->numSocConfig = idx;}}if (NULL != pXmlDoc){xmlFreeDoc(pXmlDoc);}return result;
}

 初始化IFE 

@param deviceIdDevice id for the IFE device.It should be in between 0 to 1 less than max number of IFE devices supported by platform.@returnError code defined in QCDStatus_e if open call fails else QCD_SUCCESSif all above steps is success
**************************************************************************************************/
QCDStatus_e HwMgr_IFEOpen(uint32_t deviceId)
{QCDStatus_e result          = QCD_ERROR_GENERAL_FAILURE;QCDStatus_e closeIFEResult  = QCD_ERROR_GENERAL_FAILURE;if (deviceId >= MAX_IFE_DEVICES){result = QCD_ERROR_BAD_PARAM;QCD_LOG(LOG_IFE, LOG_LEVEL_ERROR, "Invalid device id %u", deviceId);}else if (IFE_STATE_OPEN == g_ifeCtxt[deviceId].state){result = QCD_ERROR_INVALID_STATE;QCD_LOG(LOG_IFE, LOG_LEVEL_ERROR,"IFE %u : Invalid IFE state %d",deviceId, g_ifeCtxt[deviceId].state);}else{// Initialize device context//初始化 IFE CSID VFE的结构体 result = InitializeDeviceContext(deviceId);}if (QCD_SUCCESS == result){//获取IFE 寄存器base addr 的虚拟地址// Get base address from platfrom managerresult = GetBaseAddress(deviceId);}if (QCD_SUCCESS == result){// Initialize low level driverresult = InitializeDriver(deviceId);}if (QCD_SUCCESS == result){// Create mutex for IFE deviceresult = OSAL_CriticalSectionCreate(&g_ifeCtxt[deviceId].hMutex);if (QCD_SUCCESS != result){QCD_LOG(LOG_IFE, LOG_LEVEL_ERROR,"IFE %u : OSAL_CriticalSectionCreate failed with result = 0x%08x",deviceId, result);}}if (QCD_SUCCESS == result){QCD_LOG(LOG_IFE, LOG_LEVEL_HIGH,"IFE %u : OSAL_CriticalSectionCreate success Handle 0x%p",deviceId, g_ifeCtxt[deviceId].hMutex);// Create non periodic timer for IFE device for enabling error propagationresult = OSAL_TimerCreate(FALSE,0,ProcessErrorPropagationRequest,&g_csidCtxt[deviceId],"IFE_ErrorPropagator",g_OSALThreadDefaultPriority,&g_ifeCtxt[deviceId].hErrorPropagationTimer);if (QCD_SUCCESS != result){QCD_LOG(LOG_IFE, LOG_LEVEL_ERROR,"IFE %u : OSAL_TimerCreate failed with result = 0x%08x",deviceId, result);}}if (QCD_SUCCESS == result){QCD_LOG(LOG_IFE, LOG_LEVEL_HIGH,"IFE %u : OSAL_TimerCreate success Handle 0x%p",deviceId, g_ifeCtxt[deviceId].hErrorPropagationTimer);// Register IST callbacks.result = RegisterISTCallback(deviceId);}if (QCD_SUCCESS == result){if (FALSE == bIsSMMURegistrationDone){// Register SMMU callbackresult = PLM_SMMUfaultRegister(&SMMUCallback);if (QCD_SUCCESS != result){QCD_LOG(LOG_IFE, LOG_LEVEL_ERROR,"PLM_SMMUfaultRegister failed with result = 0x%08x",result);}else{bIsSMMURegistrationDone = TRUE;QCD_LOG(LOG_IFE, LOG_LEVEL_HIGH, "PLM_SMMUfaultRegister success");}}}if (QCD_SUCCESS == result){g_ifeCtxt[deviceId].state = IFE_STATE_OPEN;// Reset IFE as per power on sequenceresult = HwMgr_IFECtrl(deviceId,IFE_CMD_ID_RESET,NULL,0,NULL,0);if (QCD_SUCCESS != result){QCD_LOG(LOG_IFE, LOG_LEVEL_ERROR,"IFE %u : IFE Reset failed with result = 0x%08x",deviceId, result);}}if (QCD_SUCCESS == result){QCD_LOG(LOG_IFE, LOG_LEVEL_HIGH,"IFE %u : IFE Reset & Open success",deviceId);}else{// Close IFE devicecloseIFEResult = HwMgr_IFEClose(deviceId);if (QCD_SUCCESS != closeIFEResult){QCD_LOG(LOG_IFE, LOG_LEVEL_ERROR,"IFE %u : HwMgr_IFEClose failed with result = 0x%08x",deviceId, closeIFEResult);}}
#ifdef QCD_AIS_MIX_USECASE// Register LPM callbacks.PLM_RegisterLpmCallback(LPMcallback);
#endifreturn result;
}

初始化 IFE CSID VFE 的结构体

    Initialize global contexts for IFE, VFE and CSID blocks with default open values.Context handling for the CSID and VFE happens at the manager level only.It should be called only once as part of IFE open call. So there is no mutex protection.No param checks are done as its already done in caller function.@param deviceIdDevice id for the IFE device.It should be in between 0 to 1 less than max number of IFE devices supported by platform.@returnError code defined in QCDStatus_e if memset fails else QCD_SUCCESS
**************************************************************************************************/
static QCDStatus_e InitializeDeviceContext(uint32_t deviceId)
{QCDStatus_e  result         = QCD_ERROR_GENERAL_FAILURE;uint32_t     rdiIdx         = 0U;void         *pMemsetReturn = NULL;// Initialize IFE Deviceg_ifeCtxt[deviceId].state                   = IFE_STATE_INVALID;g_ifeCtxt[deviceId].hwCaps.hwType           = IFE_DEVICE_TYPE_LITE;g_ifeCtxt[deviceId].hwCaps.numRdi           = MAX_IFE_OUTPUT_PATH;g_ifeCtxt[deviceId].hwCaps.ifeId            = deviceId;g_ifeCtxt[deviceId].hMutex                  = NULL;g_ifeCtxt[deviceId].hErrorPropagationTimer  = NULL;// Initialize CSID Deviceg_csidCtxt[deviceId].csidId         = deviceId;g_csidCtxt[deviceId].rdiPathMax     = MAX_IFE_OUTPUT_PATH;g_csidCtxt[deviceId].baseAddr       = 0U;g_csidCtxt[deviceId].state          = CSID_STATE_INVALID;g_csidCtxt[deviceId].hIST           = NULL;g_csidCtxt[deviceId].bEnableGlobalErrorPropagation= FALSE;// Initialize all register with 0pMemsetReturn = OSAL_Memset(&g_csidCtxt[deviceId].diagRegister,0x0,sizeof(g_csidCtxt[deviceId].diagRegister));if (NULL == pMemsetReturn){result = QCD_ERROR_GENERAL_FAILURE;QCD_LOG(LOG_IFE, LOG_LEVEL_ERROR,"IFE %u : OSAL_Memset failed",deviceId);}else{// Update result as memset was successfulresult = QCD_SUCCESS;// Initialize RDI contextfor (rdiIdx = 0U; rdiIdx < MAX_IFE_OUTPUT_PATH; rdiIdx++){     //初始化g_csidCtxt g_csidCtxt[deviceId].outputRDICtxt[rdiIdx].state                = CSID_STATE_INVALID;g_csidCtxt[deviceId].outputRDICtxt[rdiIdx].bIsTPGEnable         = FALSE;g_csidCtxt[deviceId].outputRDICtxt[rdiIdx].bIsFirstSofReceived  = FALSE;g_csidCtxt[deviceId].eventMsg[rdiIdx].bNotifyUpperLayer         = FALSE;g_csidCtxt[deviceId].eventMsg[rdiIdx].type                      = IFE_EVENT_TYPE_MAX;}// Initialize VFE Device//初始化 g_vfeCtxtg_vfeCtxt[deviceId].vfeId           = deviceId;g_vfeCtxt[deviceId].rdiPathMax      = MAX_IFE_OUTPUT_PATH;g_vfeCtxt[deviceId].baseAddr        = 0U;g_vfeCtxt[deviceId].baseAddrBusWR   = 0U;g_vfeCtxt[deviceId].hAsyncEvent     = NULL;g_vfeCtxt[deviceId].hIST            = NULL;g_vfeCtxt[deviceId].hEventQueue     = NULL;g_vfeCtxt[deviceId].hISTQueue       = NULL;// Initialize all register with 0pMemsetReturn = OSAL_Memset(&g_vfeCtxt[deviceId].diagRegister,0x0,sizeof(g_vfeCtxt[deviceId].diagRegister));if (NULL == pMemsetReturn){result = QCD_ERROR_GENERAL_FAILURE;QCD_LOG(LOG_IFE, LOG_LEVEL_ERROR,"IFE %u : OSAL_Memset failed",deviceId);}}if (QCD_SUCCESS == result){// Initialize RDI context//初始化RDI context for (rdiIdx = 0U; rdiIdx < MAX_IFE_OUTPUT_PATH; rdiIdx++){g_vfeCtxt[deviceId].outputRDICtxt[rdiIdx].mode          = VFE_OUTPUT_MODE_FRAME_BASED;g_vfeCtxt[deviceId].outputRDICtxt[rdiIdx].state         = VFE_STATE_INVALID;g_vfeCtxt[deviceId].outputRDICtxt[rdiIdx].hHwFIFO       = NULL;g_vfeCtxt[deviceId].outputRDICtxt[rdiIdx].hMutex        = NULL;g_vfeCtxt[deviceId].outputRDICtxt[rdiIdx].frameCnt      = 0UL;g_vfeCtxt[deviceId].outputRDICtxt[rdiIdx].sofCnt        = 0UL;g_vfeCtxt[deviceId].outputRDICtxt[rdiIdx].frameHeaderVA = 0UL;g_vfeCtxt[deviceId].outputRDICtxt[rdiIdx].frameHeaderDA = 0UL;g_vfeCtxt[deviceId].outputRDICtxt[rdiIdx].hwTimestamp   = 0U;g_vfeCtxt[deviceId].outputRDICtxt[rdiIdx].hOutputBufferQueue    = NULL;g_vfeCtxt[deviceId].outputRDICtxt[rdiIdx].lastHeaderFrameCnt    = 0;g_vfeCtxt[deviceId].outputRDICtxt[rdiIdx].frameHeaderWriteIdx   = 0U;}}return result;
}

获取IFE CSID base addr 的vaddr 进行虚拟地址与物理地址的映射

@param deviceIdDevice id for the IFE device.It should be in between 0 to 1 less than max number of IFE devices supported by platform.@returnError code defined in QCDStatus_e if PLM call fails else QCD_SUCCESS
**************************************************************************************************/
static QCDStatus_e GetBaseAddress(uint32_t deviceId)
{QCDStatus_e         result  = QCD_ERROR_GENERAL_FAILURE;CameraHWBlockDef_t  hwDef   = {0};/** Get VFE device base address from platform manager.* Same CAMERA_HWBLOCK_IFE enum is exposed by PLM for IFE & IFE_Lite devices.*/hwDef.hwBlock   = CAMERA_HWBLOCK_IFE;hwDef.id        = deviceId;result = PLM_GetInfo(GET_HWBLOCK_VADDR, &hwDef, (uint32_t)sizeof(hwDef));if (QCD_SUCCESS != result){QCD_LOG(LOG_IFE, LOG_LEVEL_ERROR,"IFE %u : PLM_GetInfo failed with result = 0x%08x",deviceId, result);}else{// Update base addressg_vfeCtxt[deviceId].baseAddr = hwDef.offset;// Base address for BUS WR block is having fixed offset from VFE base addressg_vfeCtxt[deviceId].baseAddrBusWR = hwDef.offset + IFE_BUS_WR_ADDR_OFFSET;QCD_LOG(LOG_IFE, LOG_LEVEL_HIGH,"IFE %u : Virtual address for VFE 0x%lx",deviceId, g_vfeCtxt[deviceId].baseAddr);/** Get CSID device base address from platform manager.* Same CAMERA_HWBLOCK_IFE_CSID enum is exposed by PLM for IFE & IFE_Lite devices.*/hwDef.hwBlock   = CAMERA_HWBLOCK_IFE_CSID;hwDef.id        = deviceId;result = PLM_GetInfo(GET_HWBLOCK_VADDR, &hwDef, (uint32_t)sizeof(hwDef));if (QCD_SUCCESS != result){QCD_LOG(LOG_IFE, LOG_LEVEL_ERROR,"IFE %u : PLM_GetInfo failed with result = 0x%08x",deviceId, result);}}if (QCD_SUCCESS == result){// Update base addressg_csidCtxt[deviceId].baseAddr = hwDef.offset;QCD_LOG(LOG_IFE, LOG_LEVEL_HIGH,"IFE %u : Virtual address for CSID 0x%lx",deviceId, g_csidCtxt[deviceId].baseAddr);}return result;
}

 

/**********************************************************************************************//**
@briefInitialize low level drivers.It should be called only once as part of IFE open call. So there is no mutex protection.No param checks are done as its already done in caller function.@param deviceIdDevice id for the IFE device.It should be in between 0 to 1 less than max number of IFE devices supported by platform.@returnError code defined in QCDStatus_e if driver init call fails else QCD_SUCCESS
**************************************************************************************************/
static QCDStatus_e InitializeDriver(uint32_t deviceId)
{QCDStatus_e result  = QCD_ERROR_GENERAL_FAILURE;uint32_t    rdiIdx  = 0U;// Initialize VFE Driverresult = VFEDriver_Init(&g_vfeCtxt[deviceId]);if (QCD_SUCCESS != result){QCD_LOG(LOG_IFE, LOG_LEVEL_ERROR,"IFE %u : VFEDriver_Init failed with result = 0x%08x",deviceId, result);}else{
#ifdef QCD_AIS_MIX_USECASEresult = OSAL_CriticalSectionCreate(&g_csidCtxt[deviceId].hRegMutex);if (QCD_SUCCESS != result){QCD_LOG(LOG_IFE, LOG_LEVEL_ERROR,"CSID ID = %u : hRegMutex Create failed with result 0x%x",deviceId, result);}result = CSIDDriver_InitFFIGolderRegister(&g_csidCtxt[deviceId]);if (QCD_SUCCESS != result){QCD_LOG(LOG_IFE, LOG_LEVEL_ERROR,"IFE %u : CSIDDriver_InitFFIGolderRegister failed with result = 0x%08x",deviceId, result);}
#endif// Update RDI state//更新vfe CSID outputRDICtxt[rdiIdx].state 的状态for (rdiIdx = 0U; rdiIdx < MAX_IFE_OUTPUT_PATH; rdiIdx++){g_vfeCtxt[deviceId].outputRDICtxt[rdiIdx].state = VFE_STATE_INIT;}QCD_LOG(LOG_IFE, LOG_LEVEL_HIGH,"IFE %u : VFEDriver_Init success",deviceId);// Initialize CSID Driverg_csidCtxt[deviceId].state = CSID_STATE_INIT;for (rdiIdx = 0U; rdiIdx < MAX_IFE_OUTPUT_PATH; rdiIdx++){g_csidCtxt[deviceId].outputRDICtxt[rdiIdx].state = CSID_STATE_INIT;}}return result;
}

VFEDriver_Init

创建四个队列进行图像数据的数据获取

/**********************************************************************************************//**
@briefInitialize VFE Driver by creating OSAL resource and internal buffers for device and eachRDI paths. It internally calls deinit if init call fails to cleanup resources.This should be the first call to VFE driver.All context initilization and handling and mutex protection for the same is expected byIFE manager.@param pCtxtPointer to the device context@returnError code defined in QCDStatus_e if param validation or resource allocation fails elseQCD_SUCCESS.
**************************************************************************************************/
QCDStatus_e VFEDriver_Init(VFEDeviceCtxt_t *pCtxt)
{QCDStatus_e     result          = QCD_ERROR_GENERAL_FAILURE;QCDStatus_e     deinitResult    = QCD_ERROR_GENERAL_FAILURE;//RDI pathuint32_t        outputPath      = (uint32_t)IFE_OUTPUT_PATH_RDI0;uint32_t        outputOffset    = 0U;if (NULL == pCtxt){result = QCD_ERROR_BAD_PARAM;QCD_LOG(LOG_IFE, LOG_LEVEL_ERROR, "Null Device Context Provided");}else{QCD_LOG(LOG_IFE, LOG_LEVEL_HIGH,"VFE ID = %u : VFEDriver_Init for device ctxt 0x%p",pCtxt->vfeId, pCtxt);// Create resource for the deviceresult = CreateDeviceResource(pCtxt);}if (QCD_SUCCESS == result){QCD_LOG(LOG_IFE, LOG_LEVEL_DEBUG,"VFE ID = %u : IST Event Queue created successfully with handle = 0x%p",pCtxt->vfeId, pCtxt->hISTQueue);// Initialize RDI output contextfor (outputPath = (uint32_t)IFE_OUTPUT_PATH_RDI0;outputPath < pCtxt->rdiPathMax;outputPath++){/** Frame Header is allocated as contiguous memory at once for all RDI* and the same is used as different offset by all RDIs.* Hence setting the offset here so each RDI can access different memory location*///获取对应的IFE 输出图相对于IFE base的偏移值outputOffset = outputPath * FRAME_HEADER_CHUNK_PER_INTERF;//获取图像数据的物理地址pCtxt->outputRDICtxt[outputPath].frameHeaderDA =pCtxt->frameHeaderBuf.pDA + outputOffset;//将图像物理地址转换成虚拟地址pCtxt->outputRDICtxt[outputPath].frameHeaderVA =(uintptr_t)((uint8_t*)pCtxt->frameHeaderBuf.pVA + outputOffset);// Create resource for RDIresult = CreateRDIResource(pCtxt, (IFEOutputPathType_e)outputPath);if (QCD_SUCCESS != result){break;}}}if (QCD_SUCCESS == result){QCD_LOG(LOG_IFE, LOG_LEVEL_HIGH,"VFE ID = %u : VFEDriver_Init success for device ctxt 0x%p",pCtxt->vfeId, pCtxt);}else{// De-initialize VFE if initialization failsdeinitResult = VFEDriver_DeInit(pCtxt);if (QCD_SUCCESS != deinitResult){if (NULL == pCtxt){QCD_LOG(LOG_IFE, LOG_LEVEL_ERROR,"VFEDriver_DeInit failed for device ctxt 0x%p with result = 0x%08x",pCtxt, deinitResult);}else{QCD_LOG(LOG_IFE, LOG_LEVEL_ERROR,"VFE ID = %u : VFEDriver_DeInit failed with result = 0x%08x",pCtxt->vfeId, deinitResult);}}}
#ifdef QCD_AIS_MIX_USECASEif (QCD_SUCCESS == result){result = VFEDriver_InitFFIGolderRegister(pCtxt);if (QCD_SUCCESS != result){QCD_LOG(LOG_IFE, LOG_LEVEL_ERROR, "VFE ID = %u : FFI Init Failed", pCtxt->vfeId);}}
#endifreturn result;
}

//创建IEF event pCtxt->hEventQueue 事件队列和中断队列pCtxt->hISTQueue

/**********************************************************************************************//**
@briefCreate OSAL resources for the device/driver.- Create event for async event handling.- Create non-blocking queues between driver and manager for event handling.- Allocate buffer for frame header dump.It should be called only once as part VFEDriver_Init. So there is no mutex protection.No param checks are done as its already done in caller function.@param pCtxtPointer to the device context.@returnError code defined in QCDStatus_e if this call fails to allocate resource else QCD_SUCCESSif all above steps is success
**************************************************************************************************/
static QCDStatus_e CreateDeviceResource(VFEDeviceCtxt_t *pCtxt)
{QCDStatus_e result = QCD_ERROR_GENERAL_FAILURE;// Create async event for reset event handlingresult = OSAL_EventCreate(&pCtxt->hAsyncEvent);if (QCD_SUCCESS != result){QCD_LOG(LOG_IFE, LOG_LEVEL_ERROR,"VFE ID = %u : Async event creation failed with result = 0x%08x",pCtxt->vfeId, result);}if (QCD_SUCCESS == result){QCD_LOG(LOG_IFE, LOG_LEVEL_DEBUG,"VFE ID = %u : Async Event created successfully with handle = 0x%p",pCtxt->vfeId, pCtxt->hAsyncEvent);// Create non blocking event queue for event handling between driver and managerresult = OSAL_QueueCreate(sizeof(IFEEventMsgType_t),IFE_EVENT_QUEUE_SIZE,FALSE,&pCtxt->hEventQueue);if (QCD_SUCCESS != result){QCD_LOG(LOG_IFE, LOG_LEVEL_ERROR,"VFE ID = %u : OSAL_QueueCreate failed with result = 0x%08x",pCtxt->vfeId, result);}}if (QCD_SUCCESS == result){QCD_LOG(LOG_IFE, LOG_LEVEL_DEBUG,"VFE ID = %u : Event Queue created successfully with handle = 0x%p",pCtxt->vfeId, pCtxt->hEventQueue);/** Create non blocking IST event queue for 1st stage IST events handling between* driver and manager*/result = OSAL_QueueCreate(sizeof(IFEEventMsgType_t),IFE_EVENT_QUEUE_SIZE,FALSE,&pCtxt->hISTQueue);if (QCD_SUCCESS != result){QCD_LOG(LOG_IFE, LOG_LEVEL_ERROR,"VFE ID = %u : OSAL_QueueCreate failed with result = 0x%08x",pCtxt->vfeId, result);}}if (QCD_SUCCESS == result){//对framebuffer 分配内存用于接受IFE的图像数据// Allocate frame header buffer for all RDI pathspCtxt->frameHeaderBuf.size = CAM_ALIGN_SIZE(FRAME_HEADER_CHUNK_PER_INTERF,FRAME_HEADER_BUFFER_SIZE);pCtxt->frameHeaderBuf.flags = CAMERA_BUFFER_FLAG_CONTIGOUS;result = PLM_CameraBufferAlloc(&pCtxt->frameHeaderBuf, 0U, CAMERA_HWBLOCK_IFE);if (QCD_SUCCESS != result){QCD_LOG(LOG_IFE, LOG_LEVEL_ERROR,"VFE ID = %u : PLM_CameraBufferAlloc failed for header with result = 0x%08x",pCtxt->vfeId, result);}}
#ifdef QCD_AIS_MIX_USECASEif (QCD_SUCCESS == result){result = OSAL_CriticalSectionCreate(&pCtxt->hRegMutex);if (QCD_SUCCESS != result){QCD_LOG(LOG_IFE, LOG_LEVEL_ERROR,"VFE ID = %u : hRegMutex Create failed with result 0x%x",pCtxt->vfeId, result);}}
#endifreturn result;
}

 

/**********************************************************************************************//**
@briefCreate OSAL resources for the RDI paths.- Create non blocking queues for buffer management- Create mutex for buffer handling.It should be called only once per RDI as part VFEDriver_Init. So there is no mutex protection.No param checks are done as its already done in caller function.@param pCtxtPointer to the device context@param outputPathRDI Path for which resource will be allocated.Valid range is 0 to 1 less than max number of RDI path supported by platform.@returnError code defined in QCDStatus_e if this call fails to allocate resource else QCD_SUCCESSif all above steps is success
**************************************************************************************************/
static QCDStatus_e CreateRDIResource(VFEDeviceCtxt_t *pCtxt, IFEOutputPathType_e outputPath)
{QCDStatus_e result = QCD_ERROR_GENERAL_FAILURE;// Create non blocking output buffer queue//创建bufferqueue判断IFE driver 获取到了数据, 通知StreamManager获取数据result = OSAL_QueueCreate(sizeof(VFEOutputBuffer_t),MAX_NUM_OF_OUTPUT_BUFFERS,FALSE,&pCtxt->outputRDICtxt[outputPath].hOutputBufferQueue);if (QCD_SUCCESS != result){QCD_LOG(LOG_IFE, LOG_LEVEL_ERROR,"VFE ID = %u : OSAL_QueueCreate failed for RDI %d with result = 0x%08x",pCtxt->vfeId, outputPath, result);}else{QCD_LOG(LOG_IFE, LOG_LEVEL_DEBUG,"VFE ID = %u : Output buffer queue created for RDI = %d Handle = 0x%p",pCtxt->vfeId, outputPath,pCtxt->outputRDICtxt[outputPath].hOutputBufferQueue);//创建VFE hardware FIFO判断物理层有数据 让IFE驱动获取数据// Create non blocking FIFO queue which replicate VFE hardware FIFOresult = OSAL_QueueCreate(sizeof(VFEOutputBuffer_t),MAX_FIFO_DEPTH,FALSE,&pCtxt->outputRDICtxt[outputPath].hHwFIFO);if (QCD_SUCCESS != result){QCD_LOG(LOG_IFE, LOG_LEVEL_ERROR,"VFE ID = %u : OSAL_QueueCreate failed : RDI %d with result = 0x%08x",pCtxt->vfeId, outputPath, result);}}if (QCD_SUCCESS == result){QCD_LOG(LOG_IFE, LOG_LEVEL_DEBUG,"VFE ID = %u : FIFO queue created for RDI = %d with handle = 0x%p",pCtxt->vfeId, outputPath, pCtxt->outputRDICtxt[outputPath].hHwFIFO);// Create mutex for buffer handlingresult = OSAL_CriticalSectionCreate(&pCtxt->outputRDICtxt[outputPath].hMutex);if (QCD_SUCCESS == result){QCD_LOG(LOG_IFE, LOG_LEVEL_DEBUG,"VFE ID = %u : Mutex created for RDI = %d with handle = 0x%p",pCtxt->vfeId, outputPath, pCtxt->outputRDICtxt[outputPath].hMutex);}else{QCD_LOG(LOG_IFE, LOG_LEVEL_ERROR,"VFE ID = %u : Mutex creation failed for RDI %d with result = 0x%08x",pCtxt->vfeId, outputPath, result);}}return result;
}

 CSID init 

QCDStatus_e CSIDDriver_InitFFIGolderRegister(CSIDDeviceCtxt_t *pCtxt)
{QCDStatus_e result = QCD_SUCCESS;result = OSAL_CriticalSectionEnter(pCtxt->hRegMutex);if (QCD_SUCCESS != result){QCD_LOG(LOG_IFE, LOG_LEVEL_ERROR,"CSID %u : OSAL_CriticalSectionEnter failed with result 0x%x",pCtxt->csidId, result);}else{error_type ret;uint32_t i = 0U;for (i = 0U; i < CSID_MONITOR_REG_SIZE; i++){g_csidGoldenRegs[pCtxt->csidId][i] = csidRegNeedToMonitor[i].defaultValue;}ret = crc32_generate((char*)g_csidGoldenRegs[pCtxt->csidId],sizeof(g_csidGoldenRegs[pCtxt->csidId]), &pCtxt->regGoldenCRC);if (SUCCESS != ret){QCD_LOG(LOG_IFE, LOG_LEVEL_ERROR, "CRC Generate Failed: %x", ret);result = QCD_ERROR_INVALID_STATE;}result = OSAL_CriticalSectionLeave(pCtxt->hRegMutex);if (QCD_SUCCESS != result){QCD_LOG(LOG_IFE, LOG_LEVEL_ERROR,"CSID %u : OSAL_CriticalSectionLeave failed with result 0x%x",pCtxt->csidId, result);}}return result;
}

RegisterISTCallback

注册中断回调函数VFEProcessIST CSIDProcessIST

    It should be called only once as part of IFE open call. So there is no mutex protection.No param checks are done as its already done in caller function.@param deviceIdDevice id for the IFE device.It should be in between 0 to 1 less than max number of IFE devices supported by platform.@returnError code defined in QCDStatus_e if PLM call fails else QCD_SUCCESS
**************************************************************************************************/
static QCDStatus_e RegisterISTCallback(uint32_t deviceId)
{QCDStatus_e result = QCD_ERROR_GENERAL_FAILURE;// Register IST for VFE//注册VFE中断函数g_vfeCtxt[deviceId].hIST = PLM_ISTInit(CAMERA_HWBLOCK_IFE, deviceId, &VFEProcessIST);if (NULL == g_vfeCtxt[deviceId].hIST){result = QCD_ERROR_GENERAL_FAILURE;QCD_LOG(LOG_IFE, LOG_LEVEL_ERROR,"IFE %u : PLM_ISTInit failed with result = 0x%08x",deviceId, result);}else{// Update result as VFE IST registration is successfulresult = QCD_SUCCESS;}if (QCD_SUCCESS == result){QCD_LOG(LOG_IFE, LOG_LEVEL_HIGH,"IFE %u : PLM_ISTInit success Handle 0x%p",deviceId, g_vfeCtxt[deviceId].hIST);// Register IST for CSID//注册CSID 中断函数g_csidCtxt[deviceId].hIST = PLM_ISTInit(CAMERA_HWBLOCK_IFE_CSID,deviceId,&CSIDProcessIST);if (NULL == g_csidCtxt[deviceId].hIST){result = QCD_ERROR_GENERAL_FAILURE;QCD_LOG(LOG_IFE, LOG_LEVEL_ERROR,"IFE %u : PLM_ISTInit failed with result = 0x%08x",deviceId, result);}}if (QCD_SUCCESS == result){QCD_LOG(LOG_IFE, LOG_LEVEL_HIGH,"IFE %u : PLM_ISTInit success Handle 0x%p",deviceId, g_csidCtxt[deviceId].hIST);}return result;
}

 

@briefInterrupt handler thread function@param hwBlock [in]Camera HW block.@param devId [in]HW block Id.@param cbFunc [in]Callback function.@returnInterrupt thread handle.
**************************************************************************************************/
OsalThread_t  PLM_ISTInit(const CameraHWBlock_e hwBlock,const uint32_t devId,const PLM_ISTCbFunc_t cbFunc)
{OsalThread_t  hThreadIST    = NULL;QCDStatus_e   rc            = QCD_ERROR_GENERAL_FAILURE;if (NULL == cbFunc){QCD_LOG(LOG_PLATFORM, LOG_LEVEL_ERROR, "PLM_ISTInit called with NULL callback function");}else{PlatformISTCtxt_t *pISTCtxt = (PlatformISTCtxt_t*) OSAL_Malloc(sizeof(PlatformISTCtxt_t));if (NULL == pISTCtxt){QCD_LOG(LOG_PLATFORM, LOG_LEVEL_ERROR, "OSAL_Malloc failed");}else{rc = OSAL_CriticalSectionEnter(g_PlmIstMutex);if (QCD_SUCCESS == rc){pISTCtxt->hwDef.hwBlock   = hwBlock;pISTCtxt->hwDef.id        = devId;rc = PLM_GetInfo(GET_HWBLOCK_IRQ,&pISTCtxt->hwDef,(uint32_t)sizeof(CameraHWBlockDef_t));if (QCD_SUCCESS == rc){SIGEV_INTR_INIT(&pISTCtxt->sigEvent);pISTCtxt->istCb = cbFunc;//注册中断回调函数rc = OSAL_ThreadCreate( g_OSALThreadISTPriority,PLM_IST,pISTCtxt,0,(const char*)pISTCtxt->hwDef.name,&hThreadIST);if (QCD_SUCCESS != rc){QCD_LOG(LOG_PLATFORM, LOG_LEVEL_ERROR, "IST Thread create failure");OSAL_Free(pISTCtxt);}else{g_pCtxtList[g_CtxtCount] = pISTCtxt;g_CtxtCount++;}}else{QCD_LOG(LOG_PLATFORM, LOG_LEVEL_ERROR, "PLM_GetInfo failed (rc = 0x%x)", rc);}rc = OSAL_CriticalSectionLeave(g_PlmIstMutex);if (QCD_SUCCESS != rc){QCD_LOG(LOG_PLATFORM,LOG_LEVEL_ERROR,"OSAL_CriticalSectionLeave failed (rc = 0x%x)",rc);}}else{QCD_LOG(LOG_PLATFORM,LOG_LEVEL_ERROR,"OSAL_CriticalSectionEnter failed (rc = 0x%x)",rc);}}}return hThreadIST;
}

 获取IFE中的HWBLOCK中的硬件映射的irq number

@param pData [in]structure to fill/read info to.@param size [in]size of pData.@returnError code defined in QCDStatus_e
**************************************************************************************************/
static QCDStatus_e GetHWBlockIRQ(void *pData, size_t size)
{QCDStatus_e rc = QCD_ERROR_GENERAL_FAILURE;if (size != sizeof(CameraHWBlockDef_t)){QCD_LOG(LOG_PLATFORM, LOG_LEVEL_ERROR, "invalid size");rc = QCD_ERROR_BAD_PARAM;}else{const CameraHWDefinition_t     *pHwDef   = GetHWDef();CameraHWBlockDef_t             *pHWBlock = pData;if (pHwDef != NULL){for (int32_t i = 0; i < CAMERAPLATFORM_MAX_HWBLOCKS; i++){if ((pHWBlock->hwBlock == pHwDef->hwblocks[i].hwBlock) &&(pHWBlock->id == pHwDef->hwblocks[i].id)){pHWBlock->irq = pHwDef->hwblocks[i].irq;if (0 > snprintf(pHWBlock->name,sizeof(pHwDef->hwblocks[i].name),"%s_ist",pHwDef->hwblocks[i].name)){rc = QCD_ERROR_GENERAL_FAILURE;QCD_LOG(LOG_PLATFORM,LOG_LEVEL_ERROR,"Error setting IRQ thread name");}else{rc = QCD_SUCCESS;}break;}}}else{QCD_LOG(LOG_PLATFORM, LOG_LEVEL_ERROR, "hw block not found ");rc = QCD_ERROR_NOT_FOUND;}}return rc;
}

HwMgr_CSIPHYOpen

/**********************************************************************************************//**
@briefGet the CSIPHY device id base address from platform. Create the critical section and registerthe IST for handling the interrupt for the provided CSIPHY device Id.@param deviceIdDevice Id for the CSIPHY device and valid range of CSIPHY deviceId is 0-3.@returnIf funtion parameter deviceId is vaild, return value depends on the called functionreturn value otherwise bad param.
**************************************************************************************************/
QCDStatus_e HwMgr_CSIPHYOpen(uint32_t deviceId)
{QCDStatus_e result = QCD_ERROR_GENERAL_FAILURE;QCDStatus_e tempResult = QCD_ERROR_GENERAL_FAILURE;if (MAX_CSIPHY_DEVICES <= deviceId){result = QCD_ERROR_BAD_PARAM;QCD_LOG(LOG_CSIPHY, LOG_LEVEL_ERROR, "Invalid CSIPHY device id %u", deviceId);}else if (CSIPHY_STATE_INVALID != g_csiphyCtxt[deviceId].state){result = QCD_ERROR_INVALID_STATE;QCD_LOG(LOG_CSIPHY, LOG_LEVEL_ERROR,"CSIPHY %u : Invalid CSIPHY state %d",deviceId, g_csiphyCtxt[deviceId].state);}else{// Initialize CSIPHY Deviceg_csiphyCtxt[deviceId].state = CSIPHY_STATE_INVALID;g_csiphyCtxt[deviceId].refCount = 0U;g_csiphyCtxt[deviceId].csiphyId = deviceId;g_csiphyCtxt[deviceId].baseAddr = 0U;g_csiphyCtxt[deviceId].hMutex = NULL;g_csiphyCtxt[deviceId].hIST = NULL;
#ifdef QCD_AIS_MIX_USECASEg_csiphyCtxt[deviceId].hRegMutex = NULL;g_csiphyCtxt[deviceId].regGoldenCRC = 0U;
#endifresult = QCD_SUCCESS;}if (QCD_SUCCESS == result){result =  CSIPHYDeviceInit(deviceId);}if (QCD_SUCCESS != result){QCD_LOG(LOG_CSIPHY, LOG_LEVEL_ERROR,"CSIPHY %u : CSIPHYOpen device init failed with result 0x%x",deviceId, result);// Close CSIPHY DevicetempResult = HwMgr_CSIPHYClose(deviceId);if (QCD_SUCCESS != tempResult){QCD_LOG(LOG_CSIPHY, LOG_LEVEL_ERROR,"CSIPHY %u : HwMgr_CSIPHYClose failed with result 0x%x",deviceId, tempResult);}}else{// Update the CSIPHY stateg_csiphyCtxt[deviceId].state = CSIPHY_STATE_OPEN;QCD_LOG(LOG_CSIPHY, LOG_LEVEL_HIGH,"CSIPHY %u : CSIPHYOpen success",deviceId);}return result;
}

 

/**********************************************************************************************//**
@briefDuring HwMgr_CSIPHYOpen, the CSIPHYDeviceInit Initialize CSIPHY Device, Create the criticalsection and register the IST handler per CSIPHY device block.@precaller's function responsible to ensure calling parameters are valid.@param deviceIdDevice Id for the CSIPHY Device (valid range of device Id is from 0-3)@returnIf called function return success, return value is QCD_SUCCESS otherwise return error codeas defined in QCDStatus_e.
**************************************************************************************************/
static QCDStatus_e CSIPHYDeviceInit(uint32_t deviceId)
{QCDStatus_e result       = QCD_ERROR_GENERAL_FAILURE;CameraHWBlockDef_t hwDef;// Get CSIPHY device info from platform managerhwDef.hwBlock = CAMERA_HWBLOCK_CSIPHY;hwDef.id = deviceId;result = PLM_GetInfo(GET_HWBLOCK_VADDR, &hwDef, (uint32_t)sizeof(hwDef));if (QCD_SUCCESS != result){QCD_LOG(LOG_CSIPHY, LOG_LEVEL_ERROR,"CSIPHY %u : PLM_GetInfo failed with result 0x%x", deviceId, result);}else{//获取csiphy的基地址g_csiphyCtxt[deviceId].baseAddr = hwDef.offset;QCD_LOG(LOG_CSIPHY, LOG_LEVEL_HIGH,"CSIPHY %u : Virtual address for CSIPHY 0x%lx",deviceId, g_csiphyCtxt[deviceId].baseAddr);// Create mutex for CSIPHY deviceresult = OSAL_CriticalSectionCreate(&g_csiphyCtxt[deviceId].hMutex);if (QCD_SUCCESS != result){QCD_LOG(LOG_CSIPHY, LOG_LEVEL_ERROR,"CSIPHY %u : OSAL_CriticalSectionCreate failed with result 0x%x",deviceId, result);}}
#ifdef QCD_AIS_MIX_USECASEif (QCD_SUCCESS == result){result = OSAL_CriticalSectionCreate(&g_csiphyCtxt[deviceId].hRegMutex);if (QCD_SUCCESS != result){QCD_LOG(LOG_CSIPHY, LOG_LEVEL_ERROR,"CSIPHY %u : hRegMutex Create failed with result 0x%x",deviceId, result);}}
#endifif (QCD_SUCCESS == result){// Register IST for CSIPHY//注册CSIPHY 中断函数g_csiphyCtxt[deviceId].hIST = PLM_ISTInit(CAMERA_HWBLOCK_CSIPHY,deviceId, &CSIPHYProcessIST);if (NULL == g_csiphyCtxt[deviceId].hIST){result = QCD_ERROR_GENERAL_FAILURE;QCD_LOG(LOG_CSIPHY, LOG_LEVEL_ERROR,"CSIPHY %u : PLM_ISTInit failed with result 0x%x",deviceId, result);}else{
#ifdef QCD_AIS_MIX_USECASEresult = CSIPHYDriver_Init(&g_csiphyCtxt[deviceId]);if (QCD_SUCCESS == result)
#endifQCD_LOG(LOG_CSIPHY, LOG_LEVEL_HIGH,"CSIPHY %u : PLM_ISTInit success",deviceId);}}return result;
}

 

QCDStatus_e CSIPHYDriver_Init(CSIPHYDeviceCtxt_t *pCtxt)
{QCDStatus_e result = QCD_SUCCESS;result = OSAL_CriticalSectionEnter(pCtxt->hRegMutex);if (QCD_SUCCESS != result){QCD_LOG(LOG_CSIPHY, LOG_LEVEL_ERROR,"CSIPHY %u : OSAL_CriticalSectionEnter failed with result 0x%x",pCtxt->csiphyId, result);}else{error_type ret;memset(g_csiphyGoldenRegs[pCtxt->csiphyId], 0U, sizeof(g_csiphyGoldenRegs[pCtxt->csiphyId]));ret = crc32_generate((char*)g_csiphyGoldenRegs[pCtxt->csiphyId],sizeof(g_csiphyGoldenRegs[pCtxt->csiphyId]), &pCtxt->regGoldenCRC);if (SUCCESS != ret){QCD_LOG(LOG_CSIPHY, LOG_LEVEL_ERROR, "CRC Generate Failed: %x", ret);result = QCD_ERROR_INVALID_STATE;}result = OSAL_CriticalSectionLeave(pCtxt->hRegMutex);if (QCD_SUCCESS != result){QCD_LOG(LOG_CSIPHY, LOG_LEVEL_ERROR,"CSIPHY %u : OSAL_CriticalSectionLeave failed with result 0x%x",pCtxt->csiphyId, result);}}return result;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/27919.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【每日一题】—— B. Maximum Rounding(Codeforces Round 891 (Div. 3))

&#x1f30f;博客主页&#xff1a;PH_modest的博客主页 &#x1f6a9;当前专栏&#xff1a;每日一题 &#x1f48c;其他专栏&#xff1a; &#x1f534; 每日反刍 &#x1f7e1; C跬步积累 &#x1f7e2; C语言跬步积累 &#x1f308;座右铭&#xff1a;广积粮&#xff0c;缓称…

VIM 编辑器: Bram Moolenaar

VIM 用了很长时间&#xff0c; 个人的 VIM 配置文件差不多10年没有更新了。以前写程序的时候&#xff0c; 编辑都用这个。 linux kernel&#xff0c; boost规模的代码都不在话下。现在虽然代码写的少了&#xff0c;依然是我打开文件的首选。 现在用手机了&#xff0c;配个蓝牙键…

无涯教程-Perl - endnetent函数

描述 此功能告诉系统您不再希望使用getnetent从网络列表中读取条目。 语法 以下是此函数的简单语法- endnetent返回值 此函数不返回任何值。 例 以下是显示其基本用法的示例代码- #!/usr/bin/perluse Socket;while ( ($name, $aliases, $addrtype, $net) getnetent() )…

[RTKLIB]模糊度固定相关问题(一)

文章目录 一、改进的模糊度固定算法1. 简述模糊度固定算法思路2. 详解 manage_amb_LAMBDA()函数3. 我的思考和疑问 一、改进的模糊度固定算法 早些时候在阅读RTKLIB源码时&#xff0c;就对其模糊度固定算法感觉非常恐惧&#xff0c;首先是觉得整数最小二乘降相关是一个非常难的…

Android 9-- 源码角度: Home键的监听和拦截

在做应用层APP需求的过程中&#xff0c;HOME键的监听&#xff0c;Back键的监听&#xff0c;这都是很常见的问题&#xff0c;那你有试过&#xff0c;去拦截HOME键的事件吗&#xff0c;有去了解过如何处理吗&#xff0c;流程如何 首先大家应该先了解一种情况&#xff0c;就是Andr…

16-4_Qt 5.9 C++开发指南_Qt 应用程序的发布

文章目录 1. 应用程序发布方式2. Windows 平台上的应用程序发布 1. 应用程序发布方式 用 Qt 开发一个应用程序后&#xff0c;将应用程序提供给用户在其他计算机上使用就是应用程序的发布。应用程序发布一般会提供一个安装程序&#xff0c;将应用程序的可执行文件及需要的运行库…

Python 网络请求之requests

1.导入 requests 库 安装 requests 库 pip3 install "requests2.29.0"这里安装2,29.0版本&#xff0c;我在安装最新版本时&#xff0c;请求接口会报错&#xff1a; urllib3 v2.0 only supports OpenSSL 1.1.1, currently the ssl module is compiled with LibreSS…

Vue中使用Sortable:A和B拖拽交换位置之后,B和A又神奇得换回去了,但下面的数据确实已交换

参考&#xff1a;Vue中使用Sortable 1 问题 Vue中使用Sortable拖拽表头达到改变列的位置的效果&#xff0c; 想法&#xff1a;使用该组件进行拖拽列&#xff0c;但它不会切换原表头下的数据&#xff0c;因此&#xff1a;1、先拖拽&#xff0c;2、手动交换原先vue中的表头&am…

嵌入式面试4 Linux编程

23.设fp已定义,执行语句fpfopen(“file”,“w”);后,以下针对文本文件file操作叙述的选项错误的是:&#xff08;ACD&#xff09;【多选】 A 可以随意读和写 B 只能写不能读 C 可以在原有内容后追加写 D 写操作结束后可以从头开始读 权重&#xff1a;高 备注&#xff1a;还要理解…

Linux tun虚拟网卡通信初识

什么是linux tun设备 Linux TUN 设备是一种虚拟网络设备&#xff0c;用于在用户空间和内核空间之间建立数据通道&#xff0c;使用户空间程序可以通过这个设备与内核网络栈进行交互。TUN 设备是一种通用的网络隧道设备&#xff0c;常用于实现虚拟专用网络&#xff08;VPN&#…

【湍流介质的三维传播模拟器】全衍射3-D传播模拟器,用于在具有随机和背景结构的介质中传播无线电和光传播(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…

华为OD机试(含B卷)真题2023 算法分类版,58道20个算法分类,如果距离机考时间不多了,就看这个吧,稳稳的

目录 一、数据结构1、线性表2、优先队列3、滑动窗口4、二叉树5、并查集6、栈 二、算法1、基础算法2、字符串3、图4、动态规划5、数学 三、漫画算法2&#xff1a;小灰的算法进阶参与方式 很多小伙伴问我&#xff0c;华为OD机试算法题太多了&#xff0c;知识点繁杂&#xff0c;如…

中文版开源Llama 2同时有了语言、多模态大模型,完全可商用

可以说&#xff0c;AI 初创公司 LinkSoul.Al 的这些开源项目让海外开源大模型在国内的普及和推广速度与国际几乎保持了一致。 7 月 19 日&#xff0c;Meta 终于发布了免费可商用版本 Llama 2&#xff0c;让开源大模型领域的格局发生了巨大变化。 Llama 2 模型系列包含 70 亿、…

Python识别抖音Tiktok、巨量引擎滑块验证码识别

由于最近比较忙&#xff0c;所以本周搞了一个相对简单的验证码&#xff0c;就是抖音Tiktok的滑块验证码&#xff0c;这也是接到客户的一个需求。这种验证码通常在电脑端登录抖音、巨量引擎的的时候出现。 首先看一下最终的效果&#xff1a; 验证码识别过程 1、利用爬虫采集图…

查看单元测试用例覆盖率新姿势:IDEA 集成 JaCoCo

1、什么是 IDEA IDEA 全称 IntelliJ IDEA&#xff0c;是 Java 编程语言开发的集成环境。IntelliJ 在业界被公认为最好的 Java 开发工具&#xff0c;尤其在智能代码助手、代码自动提示、重构、JavaEE 支持、各类版本工具(git、SVN 等)、JUnit、CVS 整合、代码分析、 创新的 GUI…

04-5_Qt 5.9 C++开发指南_QComboBox和QPlainTextEdit

文章目录 1. 实例功能概述2. 源码2.1 可视化UI设计2.2 widget.h2.3 widget.cpp 1. 实例功能概述 QComboBox 是下拉列表框组件类&#xff0c;它提供一个下拉列表供用户选择&#xff0c;也可以直接当作一个QLineEdit 用作输入。OComboBox 除了显示可见下拉列表外&#xff0c;每个…

【Python学习】Python大版本新增内容精选

&#x1f308;据说&#xff0c;看我文章时 关注、点赞、收藏 的 帅哥美女们 心情都会不自觉的好起来。 前言&#xff1a; &#x1f9e1;作者简介&#xff1a;大家好我是 user_from_future &#xff0c;意思是 “ 来自未来的用户 ” &#xff0c;寓意着未来的自己一定很棒~ ✨个…

【Docker 学习笔记】Docker 命令大全

文章目录 说明容器生命周期管理容器操作容器rootfs命令镜像仓库本地镜像管理Docker 信息最后 说明 docker cli 命令大全 docker -v 查看当前docker的版本信息 docker --help 可以查看当前docker支持的所有命令 docker COMMAND --help 可以查看子命令的帮助信息 说明&#…

golang云原生怎么学?

学习golang云原生有哪些好处&#xff0c;他们的优缺点又有哪些&#xff1f; 一.好处有哪些&#xff1f; 1.高效性能&#xff1a;Golang是一门编译型语言&#xff0c;具有卓越的执行效率和并发处理能力。在云原生环境中&#xff0c;高效的性能对于应对大规模和高负载的分布式系…

flutter:Future、Stream、RxDart

Future 在Flutter中&#xff0c;Future是Dart语言中的一个类&#xff0c;用于表示异步操作的结果。与Future相关的的重要关键字包括async和await。 async&#xff1a;这个关键字用于在方法或函数声明前添加&#xff0c;以指示该方法为异步方法。在异步方法中&#xff0c;执行…