RK3568平台 Regmap子系统

2024-02-09 02:04
文章标签 平台 子系统 rk3568 regmap

本文主要是介绍RK3568平台 Regmap子系统,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一.Regmap API 简介

Linux 下使用 i2c_transfer 来读写 I2C 设备中的寄存器,SPI 接口的话使用 spi_write/spi_read 等。I2C/SPI 芯片又非常的多,因此 Linux 内核里面就会充斥了大量的 i2c_transfer 这类的冗余 代码,再者,代码的复用性也会降低。

基于代码复用的原则,Linux 内核引入了 regmap 模型,regmap 将寄存器访问的共同逻辑抽 象出来,驱动开发人员不需要再去纠结使用 SPI 或者 I2C 接口 API 函数,统一使用 regmap API 函数。这样的好处就是统一使用 regmap,降低了代码冗余,提高了驱动的可以移植性。

Linux内核后引入了regmap模型,将寄存器访问的共同逻辑抽象出来,只需初始化时指定总线类型、寄存器位宽等关键参数,即可通过regmap模型接口来操作器件寄存器。当然,regmap同样适用于操作cpu自身的寄存器。将i2c、spi、mmio、irq都抽象出统一的接口regmap_read、regmap_write、regmap_update_bits等接口 ,从而提高代码的可重用性,并且使得在使用如上内核基础组件时变得更为简单易用。

二.Regmap 驱动框架

regmap整体上分为三层,从下到上分别为物理总线、regmap核心、regmap api。

底层,对接的是具体物理总线,目前regmap框架支持i2c、spi、mmio、spmi、ac97总线
核心层,regmap 核心实现
api,抽象通用接口
对于使用regmap框架来说,可以不用关心regmap核心的实现过程,只需根据物理总线类型,配置好相关参数信息,即可调用regmap api访问芯片寄存器。

三.Regmap 操作函数

Regmap 申请与初始化:

regmap 支持多种物理总线,比如 I2C 和 SPI,我们需要根据所使用的接口来选 择合适的 regmap 初始化函数。

SPI 接口 初始化函数为 regmap_init_spi,函数原型如下:

struct regmap * regmap_init_spi(struct spi_device *spi,const struct regmap_config *config)

I2C 接口的 regmap 初始化函数为 regmap_init_i2c,函数原型如下:

struct regmap * regmap_init_i2c(struct i2c_client *i2c,const struct regmap_config *config)

在退出驱动的时候需要释放掉申请到的 regmap,不管是什么接口,全部使用 regmap_exit 这 个函数来释放 regmap,函数原型如下:

void regmap_exit(struct regmap *map)

regmap 设备访问 API 函数:

不管是 I2C 还是 SPI 等接口,还是 SOC 内部的寄存器,对于寄存器的操作就两种:读和 写。regmap 提供了最核心的两个读写操作:regmap_read 和 regmap_write。

regmap_read 函数原型如下:

int regmap_read(struct regmap *map,unsigned int reg,unsigned int *val)

regmap_write 函数原型如下:

int regmap_write(struct regmap *map,unsigned int reg,unsigned int val)

四.Regmap实验程序

 本实验例子主要是对比IIC Regmap api与传统的i2c_transfer的区别。

 #include <linux/types.h>#include <linux/kernel.h>#include <linux/delay.h>#include <linux/ide.h>#include <linux/init.h>#include <linux/module.h>#include <linux/errno.h>#include <linux/gpio.h>#include <linux/cdev.h>#include <linux/device.h>#include <linux/of_gpio.h>#include <linux/semaphore.h>#include <linux/timer.h>#include <linux/i2c.h>#include <linux/miscdevice.h>#include <asm/uaccess.h>#include <asm/io.h>#include <linux/regmap.h>#include "regmap_iic.h"#define AP3216C_MINOR	255#define AP3216C_NAME "ap3216c"typedef struct {struct i2c_client *client;unsigned short ir, als, ps;struct regmap *regmap;struct regmap_config regmap_config;}ap3216c_dev;ap3216c_dev ap3216c;/** @description : 从 ap3216c 读取多个寄存器数据* @param – dev : ap3216c 设备* @param – reg : 要读取的寄存器首地址* @param – val : 读取到的数据* @param – len : 要读取的数据长度* @return : 操作结果*/
static int ap3216c_read_regs(ap3216c_dev *dev, u8 reg,void *val, int len){
#if 0int ret;struct i2c_msg msg[2];struct i2c_client *client = (struct i2c_client *)dev->client;/* msg[0]为发送要读取的首地址 */msg[0].addr = client->addr; /* ap3216c 地址 */msg[0].flags = 0; /* 标记为发送数据 */msg[0].buf = &reg; /* 读取的首地址 */msg[0].len = 1; /* reg 长度 *//* msg[1]读取数据 */msg[1].addr = client->addr; /* ap3216c 地址 */msg[1].flags = I2C_M_RD; /* 标记为读取数据 */msg[1].buf = val; /* 读取数据缓冲区 */msg[1].len = len; /* 要读取的数据长度 */ret = i2c_transfer(client->adapter, msg, 2);if(ret == 2) {ret = 0;} else {printk("i2c rd failed=%d reg=%06x len=%d\n",ret, reg, len);ret = -EREMOTEIO;}#endif int ret = 0;reg = regmap_read(dev->regmap,reg,(unsigned int*)val);return ret;}/** @description: 向 ap3216c 多个寄存器写入数据* @param - dev: ap3216c 设备* @param - reg: 要写入的寄存器首地址* @param - val: 要写入的数据缓冲区* @param - len: 要写入的数据长度* @return : 操作结果*/
static s32 ap3216c_write_regs(ap3216c_dev *dev, u8 reg,u8 *buf, u8 len)
{
#if 0
u8 b[256];
struct i2c_msg msg;
struct i2c_client *client = (struct i2c_client *)dev->client;b[0] = reg; /* 寄存器首地址 */
memcpy(&b[1],buf,len); /* 将要写入的数据拷贝到数组 b 里面 */msg.addr = client->addr; /* ap3216c 地址 */
msg.flags = 0; /* 标记为写数据 */msg.buf = b; /* 要写入的数据缓冲区 */msg.len = len + 1; /* 要写入的数据长度 */return i2c_transfer(client->adapter, &msg, 1);#endifregmap_write(dev->regmap, reg, (unsigned int) *buf);return 0;}/** @description : 读取 AP3216C 的数据,包括 ALS,PS 和 IR, 注意!如果同时* :打开 ALS,IR+PS 两次数据读取的时间间隔要大于 112.5ms* @param – ir : ir 数据* @param - ps : ps 数据* @param - ps : als 数据* @return : 无。*/void ap3216c_readdata(ap3216c_dev *dev){unsigned char i =0;unsigned char buf[6];/* 循环读取所有传感器数据 */for(i = 0; i < 6; i++) {ap3216c_read_regs(dev, AP3216C_IRDATALOW + i, buf+i, 1); }if(buf[0] & 0X80) /* IR_OF 位为 1,则数据无效 */dev->ir = 0; else /* 读取 IR 传感器的数据 */dev->ir = ((unsigned short)buf[1] << 2) | (buf[0] & 0X03); /* 读取ALS传感器的数据*/dev->als = ((unsigned short)buf[3] << 8) | buf[2]; if(buf[4] & 0x40) /* IR_OF 位为 1,则数据无效 */dev->ps = 0; else /* 读取 PS 传感器的数据 */dev->ps = ((unsigned short)(buf[5] & 0X3F) << 4) | (buf[4] &0X0F);}static int lcs_ap3216c_init(ap3216c_dev *dev)
{unsigned char buf = 0;/* 复位ap3216c */buf = 0x04;ap3216c_write_regs(dev, AP3216C_SYSTEMCONG, &buf, 1);/* 延时 */mdelay(50);/* 开启ALS、PS、IR*/buf = 0x03;ap3216c_write_regs(dev, AP3216C_SYSTEMCONG, &buf, 1);return 0;
}/** @description : 打开设备* @param – inode : 传递给驱动的 inode* @param - filp : 设备文件,file 结构体有个叫做 private_data 的成员变量* 一般在 open 的时候将 private_data 指向设备结构体。* @return : 0 成功;其他 失败*/
static int ap3216c_open(struct inode *inode, struct file *filp){/* 设置私有数据 */filp->private_data = &ap3216c;/* 初始化ap216c */lcs_ap3216c_init(&ap3216c);return 0;}/** @description : 从设备读取数据* @param - filp : 要打开的设备文件(文件描述符)* @param - buf : 返回给用户空间的数据缓冲区* @param - cnt : 要读取的数据长度* @param - offt : 相对于文件首地址的偏移* @return : 读取的字节数,如果为负值,表示读取失败*/
static ssize_t ap3216c_read(struct file *filp, char __user *buf,size_t cnt, loff_t *off){short data[3];long err = 0;ap3216c_dev *dev = (ap3216c_dev*)filp->private_data;/* 读取ap3216c数据*/ap3216c_readdata(dev);data[0] = dev->ir;data[1] = dev->als;data[2] = dev->ps;err = copy_to_user(buf, data, sizeof(data));return 0;}/** @description : 关闭/释放设备* @param - filp : 要关闭的设备文件(文件描述符)* @return : 0 成功;其他 失败*/static int ap3216c_release(struct inode *inode, struct file *filp){return 0;}/* AP3216C 操作函数 */static const struct file_operations ap3216c_ops = {.owner = THIS_MODULE,.open = ap3216c_open,.read = ap3216c_read,.release = ap3216c_release,};/* MISC设备结构体 */static struct miscdevice ap3216c_miscdev = {.minor = AP3216C_MINOR,.name = AP3216C_NAME,.fops = &ap3216c_ops,};/** @description : i2c 驱动的 probe 函数,当驱动与* 设备匹配以后此函数就会执行* @param – client : i2c 设备* @param - id : i2c 设备 ID* @return : 0,成功;其他负值,失败*/
static int ap3216c_probe(struct i2c_client *client,const struct i2c_device_id *id){int ret = 0;/* 注册MISC子系统 */ret = misc_register(&ap3216c_miscdev);if(ret < 0){printk("ap3216 misc device register failed!\r\n");ret = -EFAULT;}/* 获取I2C设备 */ap3216c.client = client;/* 初始化regmap_config设置 */ap3216c.regmap_config.reg_bits = 8;		/* 寄存器长度8bit */ap3216c.regmap_config.val_bits = 8;		/* 值长度8bit *//* 初始化I2C接口的regmap */ap3216c.regmap = regmap_init_i2c(client, &ap3216c.regmap_config);if(IS_ERR(ap3216c.regmap)){return PTR_ERR(ap3216c.regmap);}return 0;}/** @description : i2c 驱动的 remove 函数,移除 i2c 驱动的时候此函数会执行* @param - client : i2c 设备* @return : 0,成功;其他负值,失败*/
static int ap3216c_remove(struct i2c_client *client){int ret = 0;/*释放regmap*/regmap_exit(ap3216c.regmap);/*MISC 驱动框架卸载*/misc_deregister(&ap3216c_miscdev);return ret;}/* 传统匹配方式 ID 列表 */static const struct i2c_device_id ap3216c_id[] = {{"alientek,ap3216c", 0}, {}};/* 设备树匹配列表 */static const struct of_device_id ap3216c_of_match[] = {{ .compatible = "alientek,ap3216c" },{ /* Sentinel */ }};/* i2c 驱动结构体 */ static struct i2c_driver ap3216c_driver = {.probe = ap3216c_probe,.remove = ap3216c_remove,.driver = {.owner = THIS_MODULE,.name = "ap3216c",.of_match_table = ap3216c_of_match,},.id_table = ap3216c_id,};/** @description : 驱动入口函数* @param : 无* @return : 无*/static int __init ap3216c_init(void){int ret = 0;ret = i2c_add_driver(&ap3216c_driver);return ret;}static void __exit ap3216c_exit(void){i2c_del_driver(&ap3216c_driver);}/* module_i2c_driver(ap3216c_driver) */module_init(ap3216c_init);module_exit(ap3216c_exit);MODULE_LICENSE("GPL");MODULE_AUTHOR("ALIENTEK");MODULE_INFO(intree, "Y");

这篇关于RK3568平台 Regmap子系统的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

流媒体平台/视频监控/安防视频汇聚EasyCVR播放暂停后视频画面黑屏是什么原因?

视频智能分析/视频监控/安防监控综合管理系统EasyCVR视频汇聚融合平台,是TSINGSEE青犀视频垂直深耕音视频流媒体技术、AI智能技术领域的杰出成果。该平台以其强大的视频处理、汇聚与融合能力,在构建全栈视频监控系统中展现出了独特的优势。视频监控管理系统EasyCVR平台内置了强大的视频解码、转码、压缩等技术,能够处理多种视频流格式,并以多种格式(RTMP、RTSP、HTTP-FLV、WebS

综合安防管理平台LntonAIServer视频监控汇聚抖动检测算法优势

LntonAIServer视频质量诊断功能中的抖动检测是一个专门针对视频稳定性进行分析的功能。抖动通常是指视频帧之间的不必要运动,这种运动可能是由于摄像机的移动、传输中的错误或编解码问题导致的。抖动检测对于确保视频内容的平滑性和观看体验至关重要。 优势 1. 提高图像质量 - 清晰度提升:减少抖动,提高图像的清晰度和细节表现力,使得监控画面更加真实可信。 - 细节增强:在低光条件下,抖

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟&nbsp;开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚&nbsp;第一站:海量资源,应有尽有 走进“智听

如何解决线上平台抽佣高 线下门店客流少的痛点!

目前,许多传统零售店铺正遭遇客源下降的难题。尽管广告推广能带来一定的客流,但其费用昂贵。鉴于此,众多零售商纷纷选择加入像美团、饿了么和抖音这样的大型在线平台,但这些平台的高佣金率导致了利润的大幅缩水。在这样的市场环境下,商家之间的合作网络逐渐成为一种有效的解决方案,通过资源和客户基础的共享,实现共同的利益增长。 以最近在上海兴起的一个跨行业合作平台为例,该平台融合了环保消费积分系统,在短

Android平台播放RTSP流的几种方案探究(VLC VS ExoPlayer VS SmartPlayer)

技术背景 好多开发者需要遴选Android平台RTSP直播播放器的时候,不知道如何选的好,本文针对常用的方案,做个大概的说明: 1. 使用VLC for Android VLC Media Player(VLC多媒体播放器),最初命名为VideoLAN客户端,是VideoLAN品牌产品,是VideoLAN计划的多媒体播放器。它支持众多音频与视频解码器及文件格式,并支持DVD影音光盘,VCD影

【区块链 + 人才服务】区块链集成开发平台 | FISCO BCOS应用案例

随着区块链技术的快速发展,越来越多的企业开始将其应用于实际业务中。然而,区块链技术的专业性使得其集成开发成为一项挑战。针对此,广东中创智慧科技有限公司基于国产开源联盟链 FISCO BCOS 推出了区块链集成开发平台。该平台基于区块链技术,提供一套全面的区块链开发工具和开发环境,支持开发者快速开发和部署区块链应用。此外,该平台还可以提供一套全面的区块链开发教程和文档,帮助开发者快速上手区块链开发。

K8S(Kubernetes)开源的容器编排平台安装步骤详解

K8S(Kubernetes)是一个开源的容器编排平台,用于自动化部署、扩展和管理容器化应用程序。以下是K8S容器编排平台的安装步骤、使用方式及特点的概述: 安装步骤: 安装Docker:K8S需要基于Docker来运行容器化应用程序。首先要在所有节点上安装Docker引擎。 安装Kubernetes Master:在集群中选择一台主机作为Master节点,安装K8S的控制平面组件,如AP

衡石分析平台使用手册-单机安装及启动

单机安装及启动​ 本文讲述如何在单机环境下进行 HENGSHI SENSE 安装的操作过程。 在安装前请确认网络环境,如果是隔离环境,无法连接互联网时,请先按照 离线环境安装依赖的指导进行依赖包的安装,然后按照本文的指导继续操作。如果网络环境可以连接互联网,请直接按照本文的指导进行安装。 准备工作​ 请参考安装环境文档准备安装环境。 配置用户与安装目录。 在操作前请检查您是否有 sud

业务协同平台--简介

一、使用场景         1.多个系统统一在业务协同平台定义协同策略,由业务协同平台代替人工完成一系列的单据录入         2.同时业务协同平台将执行任务推送给pda、pad等执行终端,通知各人员、设备进行作业执行         3.作业过程中,可设置完成时间预警、作业节点通知,时刻了解作业进程         4.做完再给你做过程分析,给出优化建议         就问你这一套下

容器编排平台Kubernetes简介

目录 什么是K8s 为什么需要K8s 什么是容器(Contianer) K8s能做什么? K8s的架构原理  控制平面(Control plane)         kube-apiserver         etcd         kube-scheduler         kube-controller-manager         cloud-controlle