MT6737的lens HAL层分析

2024-01-25 02:08
文章标签 分析 hal lens mt6737

本文主要是介绍MT6737的lens HAL层分析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

lens驱动和HAL分析
一.6737的lens 搜索HAL层分析
Mcu_drv.cpp (vendor\mediatek\proprietary\hardware\mtkcam\legacy\platform\mt6735\core\featureio\drv\lens)	
根据摄像头搜索lens
MCUDrv::lensSearch( unsigned int a_u4CurrSensorDev, unsigned int a_u4CurrSensorId)//Camera_custom_msdk.cpp (vendor\mediatek\proprietary\custom\mt6735\hal\d1\camera_3a)LensCustomInit(a_u4CurrSensorDev); //得到lens列表,  //单独分析1   LensCustomGetInitFunc(&MCUDrv::m_LensInitFunc_main[0]);  //把LensCustomInit得到的数组赋值到a_pLensInitFuncmemcpy(a_pLensInitFunc, &LensInitFunc[0], sizeof(MSDK_LENS_INIT_FUNCTION_STRUCT) * MAX_NUM_OF_SUPPORT_LENS);MCUDrv::m_u4CurrLensIdx_main = 0;for (i=0; i<MAX_NUM_OF_SUPPORT_LENS; i++) //搜索每个lensif (MCUDrv::m_LensInitFunc_main[i].LensId == DUMMY_LENS_ID)  //如果某个lens的ID为DUMMY_LENS_ID,记录,这里是选择最后一个DUMMY_LENS_IDMCUDrv::m_u4CurrLensIdx_main = i;for (i=0; i<MAX_NUM_OF_SUPPORT_LENS; i++)//把当前的sensor id和数组里面的比较,如果有,就记录,并且跳槽,所以是选择匹配的第一个if ((MCUDrv::m_LensInitFunc_main[i].SensorId == a_u4CurrSensorId) && (a_u4CurrSensorId!=0xFFFF) && (a_u4CurrSensorId!=0x0))MCUDrv::m_u4CurrLensIdx_main = i;MCU_DRV_DBG("[idx]%d\n", i);break;LensCustomSetIndex(MCUDrv::m_u4CurrLensIdx_main); //记录下主摄像头的lens的IDgMainLensIdx = a_u4CurrIdx;//单独分析1//Lenslist.cpp (vendor\mediatek\proprietary\custom\mt6735\hal\d1\lens\src)GetLensInitFuncList(&LensInitFunc[0], a_u4CurrSensorDev);if(a_u4CurrSensorDev==2) //submemcpy(pLensList, &LensList_sub[0], sizeof(MSDK_LENS_INIT_FUNCTION_STRUCT)* MAX_NUM_OF_SUPPORT_LENS);else if(a_u4CurrSensorDev==4) //main 2memcpy(pLensList, &LensList_main2[0], sizeof(MSDK_LENS_INIT_FUNCTION_STRUCT)* MAX_NUM_OF_SUPPORT_LENS);else  // main or othersmemcpy(pLensList, &LensList_main[0], sizeof(MSDK_LENS_INIT_FUNCTION_STRUCT)* MAX_NUM_OF_SUPPORT_LENS);  //拷贝到pLensListMSDK_LENS_INIT_FUNCTION_STRUCT LensList_main[MAX_NUM_OF_SUPPORT_LENS] =
{{DUMMY_SENSOR_ID, DUMMY_LENS_ID, "Dummy", pDummy_getDefaultData},#if defined(SENSORDRIVE)	{OV3640_SENSOR_ID, SENSOR_DRIVE_LENS_ID, "kd_camera_hw", pSensorDrive_getDefaultData},#endif
#if defined(FM50AF){DUMMY_SENSOR_ID, FM50AF_LENS_ID, "FM50AF", pFM50AF_getDefaultData},#endif#if defined(DW9714AF){MN34152_SENSOR_ID, DW9714AF_LENS_ID, "DW9714AF", pDW9714AF_getDefaultData},{IMX219_SENSOR_ID, DW9714AF_LENS_ID, "DW9714AF", pDW9714AF_getDefaultData},#endif
}二.lens HAL层调用分析
Lens_drv.cpp (vendor\mediatek\proprietary\hardware\mtkcam\legacy\platform\mt6735\core\featureio\drv\lens)1.初始化
LensDrv::init(unsigned int a_u4CurrSensorDev )sprintf(cBuf, "/dev/%s", MCUDrv::m_LensInitFunc_main[a_u4CurrLensIdx].LensDrvName);m_fdMCU_main = open("/dev/MAINAF", O_RDWR); //打开这个节点memcpy(motorName.uMotorName, MCUDrv::m_LensInitFunc_main[a_u4CurrLensIdx].LensDrvName, 32); //得到AF的名字int err = ioctl(m_fdMCU_main,mcuIOC_S_SETDRVNAME,&motorName);   //设置AF的名字,驱动的AFIOC_S_SETDRVNAME,后面有分析2.移动马达
LensDrv::moveMCU(int a_i4FocusPos,unsigned int a_u4CurrSensorDev )a_fdMCU=m_fdMCU_main;//设置马达位置,操作/dev/MAINAF,a_i4FocusPos为知(0--1023),去的default,后面有分析err = ioctl(a_fdMCU,mcuIOC_T_MOVETO,(unsigned long)a_i4FocusPos);  3.获取马达信息
LensDrv::getMCUInfo(mcuMotorInfo *a_pMotorInfo, unsigned int a_u4CurrSensorDev )err = ioctl(a_fdMCU,mcuIOC_G_MOTORINFO, &motorInfo);。。。。。。。。其他操作。。。。。。。。。三.Main_lens公共驱动分析
Main_lens.c (kernel-3.18\drivers\misc\mediatek\lens)#define PLATFORM_DRIVER_NAME "lens_actuator_main_af"/* platform structure */
static struct platform_driver g_stAF_Driver = {.probe = AF_probe,.remove = AF_remove,.suspend = AF_suspend,.resume = AF_resume,.driver = {.name = PLATFORM_DRIVER_NAME,.owner = THIS_MODULE,}
};static struct platform_device g_stAF_device = {.name = PLATFORM_DRIVER_NAME,.id = 0,.dev = {}
};1.初始化,注册平台设备驱动
MAINAF_i2C_initplatform_device_register(&g_stAF_device) //平台驱动匹配,进入probe函数platform_driver_register(&g_stAF_Driver)2.注册I2C设备
#if I2C_CONFIG_SETTING == 2
static const struct of_device_id MAINAF_of_match[] = {{.compatible = "mediatek,CAMERA_MAIN_AF"},{},
};
#endif
static struct i2c_driver AF_i2c_driver = {.probe = AF_i2c_probe,.remove = AF_i2c_remove,.driver.name = AF_DRVNAME,
#if I2C_CONFIG_SETTING == 2.driver.of_match_table = MAINAF_of_match,
#endif.id_table = AF_i2c_id,
};AF_probei2c_add_driver(&AF_i2c_driver); //匹配进入AF_i2c_probe函数,生成节点/sys/bus/i2c/drivers/MAINAF3.I2C初始化static const struct file_operations g_stAF_fops = {.owner = THIS_MODULE,.open = AF_Open,.release = AF_Release,.unlocked_ioctl = AF_Ioctl,
#ifdef CONFIG_COMPAT.compat_ioctl = AF_Ioctl,
#endif
};AF_i2c_probeg_pstAF_I2Cclient = client; //保存client,应该和camera一样,先注册个假的地址i4RetValue = Register_AF_CharDrv();alloc_chrdev_region(&g_AF_devno, 0, 1, AF_DRVNAME) //分配设备号g_pAF_CharDrv = cdev_alloc(); //分配字符设备#define AF_DRVNAME "MAINAF"cdev_init(g_pAF_CharDrv, &g_stAF_fops); 初始化操作函数cdev_add(g_pAF_CharDrv, g_AF_devno, 1) //注册进系统,生成节点/dev/MAINAFactuator_class = class_create(THIS_MODULE, AF_DRIVER_CLASS_NAME);   //创建classvcm_device = device_create(actuator_class, NULL, g_AF_devno, NULL, AF_DRVNAME); //创建设备节点sys/bus/platform/drivers/lens_actuator_main_af4.AF_Ioctl,提供接口给上层使用
//lens的list
static stAF_DrvList g_stAF_DrvList[MAX_NUM_OF_LENS] = {#ifdef CONFIG_MTK_LENS_BU6424AF_SUPPORT{1, AFDRV_BU6424AF, BU6424AF_SetI2Cclient, BU6424AF_Ioctl, BU6424AF_Release},#endif
}switch (a_u4Command)case AFIOC_S_SETDRVNAME: //设置名字和把变量传入具体驱动i4RetValue = AF_SetMotorName((__user stAF_MotorName *)(a_u4Param));copy_from_user(&stMotorName , pstMotorName, sizeof(stAF_MotorName) //用户空间拷贝数据for (i = 0; i < MAX_NUM_OF_LENS; i++)if (strcmp(stMotorName.uMotorName, g_stAF_DrvList[i].uDrvName) == 0) //如果名字相同g_pstAF_CurDrv = &g_stAF_DrvList[i]; //设置当前设备g_pstAF_CurDrv->pAF_SetI2Cclient(g_pstAF_I2Cclient, &g_AF_SpinLock, &g_s4AF_Opened); //调用具体lens的pAF_SetI2CclientDW9714AF_SetI2Cclient //比如DW9714AF马达g_pstAF_I2Cclient = pstAF_I2Cclient;  //保存传进来的client指针,以后就可以在具体驱动中修改main_af的东西了g_pAF_SpinLock = pAF_SpinLock;g_pAF_Opened = pAF_Opened;  //open标志位default:i4RetValue = g_pstAF_CurDrv->pAF_Ioctl(a_pstFile, a_u4Command, a_u4Param); //调用具体驱动的ioctl四.具体AF驱动分析:DW9714AF为例DW9714AF.c (kernel-3.18\drivers\misc\mediatek\lens\common\dw9714af)
主要是ioctl分析
DW9714AF_Ioctlswitch (a_u4Command)case AFIOC_G_MOTORINFO: //获取一些信息,比如位置getAFInfo((__user stAF_MotorInfo *) (a_u4Param));case AFIOC_T_MOVETO:  //驱动到马达到具体位置i4RetValue = moveAF(a_u4Param); ret = s4AF_ReadReg(&InitPos); //单独分析1g_pstAF_I2Cclient->addr = AF_I2C_SLAVE_ADDR; case AFIOC_T_SETINFPOS: //设置AF的位置i4RetValue = setAFInf(a_u4Param);g_u4AF_INF = a_u4Position; /case AFIOC_T_SETMACROPOS: //设置AF最大的移动位置i4RetValue = setAFMacro(a_u4Param);g_u4AF_MACRO = a_u4Position; //单独分析1
ret = s4AF_ReadReg(&InitPos);g_pstAF_I2Cclient->addr = AF_I2C_SLAVE_ADDR;  //当前驱动的I2C地址g_pstAF_I2Cclient->addr = g_pstAF_I2Cclient->addr >> 1; i4RetValue = i2c_master_recv(g_pstAF_I2Cclient, pBuff, 2); /读写I2C

 

这篇关于MT6737的lens HAL层分析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/641769

相关文章

Springboot中分析SQL性能的两种方式详解

《Springboot中分析SQL性能的两种方式详解》文章介绍了SQL性能分析的两种方式:MyBatis-Plus性能分析插件和p6spy框架,MyBatis-Plus插件配置简单,适用于开发和测试环... 目录SQL性能分析的两种方式:功能介绍实现方式:实现步骤:SQL性能分析的两种方式:功能介绍记录

最长公共子序列问题的深度分析与Java实现方式

《最长公共子序列问题的深度分析与Java实现方式》本文详细介绍了最长公共子序列(LCS)问题,包括其概念、暴力解法、动态规划解法,并提供了Java代码实现,暴力解法虽然简单,但在大数据处理中效率较低,... 目录最长公共子序列问题概述问题理解与示例分析暴力解法思路与示例代码动态规划解法DP 表的构建与意义动

C#使用DeepSeek API实现自然语言处理,文本分类和情感分析

《C#使用DeepSeekAPI实现自然语言处理,文本分类和情感分析》在C#中使用DeepSeekAPI可以实现多种功能,例如自然语言处理、文本分类、情感分析等,本文主要为大家介绍了具体实现步骤,... 目录准备工作文本生成文本分类问答系统代码生成翻译功能文本摘要文本校对图像描述生成总结在C#中使用Deep

Redis主从/哨兵机制原理分析

《Redis主从/哨兵机制原理分析》本文介绍了Redis的主从复制和哨兵机制,主从复制实现了数据的热备份和负载均衡,而哨兵机制可以监控Redis集群,实现自动故障转移,哨兵机制通过监控、下线、选举和故... 目录一、主从复制1.1 什么是主从复制1.2 主从复制的作用1.3 主从复制原理1.3.1 全量复制

Redis主从复制的原理分析

《Redis主从复制的原理分析》Redis主从复制通过将数据镜像到多个从节点,实现高可用性和扩展性,主从复制包括初次全量同步和增量同步两个阶段,为优化复制性能,可以采用AOF持久化、调整复制超时时间、... 目录Redis主从复制的原理主从复制概述配置主从复制数据同步过程复制一致性与延迟故障转移机制监控与维

Redis连接失败:客户端IP不在白名单中的问题分析与解决方案

《Redis连接失败:客户端IP不在白名单中的问题分析与解决方案》在现代分布式系统中,Redis作为一种高性能的内存数据库,被广泛应用于缓存、消息队列、会话存储等场景,然而,在实际使用过程中,我们可能... 目录一、问题背景二、错误分析1. 错误信息解读2. 根本原因三、解决方案1. 将客户端IP添加到Re

Redis主从复制实现原理分析

《Redis主从复制实现原理分析》Redis主从复制通过Sync和CommandPropagate阶段实现数据同步,2.8版本后引入Psync指令,根据复制偏移量进行全量或部分同步,优化了数据传输效率... 目录Redis主DodMIK从复制实现原理实现原理Psync: 2.8版本后总结Redis主从复制实

锐捷和腾达哪个好? 两个品牌路由器对比分析

《锐捷和腾达哪个好?两个品牌路由器对比分析》在选择路由器时,Tenda和锐捷都是备受关注的品牌,各自有独特的产品特点和市场定位,选择哪个品牌的路由器更合适,实际上取决于你的具体需求和使用场景,我们从... 在选购路由器时,锐捷和腾达都是市场上备受关注的品牌,但它们的定位和特点却有所不同。锐捷更偏向企业级和专

Spring中Bean有关NullPointerException异常的原因分析

《Spring中Bean有关NullPointerException异常的原因分析》在Spring中使用@Autowired注解注入的bean不能在静态上下文中访问,否则会导致NullPointerE... 目录Spring中Bean有关NullPointerException异常的原因问题描述解决方案总结

python中的与时间相关的模块应用场景分析

《python中的与时间相关的模块应用场景分析》本文介绍了Python中与时间相关的几个重要模块:`time`、`datetime`、`calendar`、`timeit`、`pytz`和`dateu... 目录1. time 模块2. datetime 模块3. calendar 模块4. timeit