当前位置: 首页 > news >正文

RK3568平台 RTC时间框架

一.RTC时间框架概述

RTC(Real Time Clock)是一种用于计时的模块,可以是再soc内部,也可以是外部模块。对于soc内部的RTC,只需要读取寄存器即可,对于外部模块的RTC,一般需要使用到I2C接口进行读取。至于如何供电,都是可以在主电源断开后使用纽扣电源辅助供电,达到设备断电仍然可以计时的效果。

对于soc内部的RTC,需要额外的晶振,而外部模块的RTC不需要。

在linux内核里,rtc是一个字符设备,内核实现了一个通用的字符设备层,提供给应用层系统调用,给底下的RTC硬件驱动层提供注册接口。每款RTC需要编写对应的硬件驱动,填写读取和设置时间的回调函数,并且注册进系统。从上到下分别是应用层、rtc通用字符设备层、rtc底层驱动层。

Linux 内核中,RTC 驱动的结构图如下所示, 可以分为三个层次:

接口层,负责向用户空间提供操作的结点以及相关接口。
• RTC Core, 为rtc 驱动提供了一套API, 完成设备和驱动的注册等。
• RTC 驱动层,负责具体的RTC 驱动实现,如设置时间、闹钟等设置寄存器的操作。

二.RTC设备驱动

申请rtc_device,RTC注册函数:

struct rtc_device *devm_rtc_device_register(struct device *dev,  const char *name, const struct rtc_class_ops *ops,  struct module *owner)

这个接口会自动帮你从设备树的aliases节点获取rtcx的序号,如果没有填写就自动分配一个序号。

注销rtc_device,RTC注销函数:

void devm_rtc_device_unregister(struct device *dev,  struct rtc_device *rtc)

内核RTC框架提供了3中数据结构:

struct rtc_time、struct rtc_device和struct rtc_device_ops结构,其定义如下:

struct rtc_time {int tm_sec, tm_min, tm_hour, tm_mday, tm_mon, tm_year/*从1900开始*/;int tm_wday, tm_yday, tm_isdst/*夏令时标志*/;
};struct rtc_device {struct device dev;struct module *owner;int id; // 由内核提供给rtc设备的全局索引,/dev/rtc<id>char name[RTC_DEVICE_NAME_SIZE];const struct rtc_class_ops *ops; // 一组操作,例如 设置/读取 时间/闹钟struct mutex ops_lock;struct cdev char_dev;ulong flags;ulong irq_data;spinlock_t irq_lock;wait_queue_head_t irq_queue;struct rtc_task *irq_task;spinlock_t irq_task_lock;int irq_freq;int max_user_freq;struct work_struct irqwork;
};struct rtc_class_ops {int (*open)(struct device *dev); // 用户在设备/dev/rtc1上调用read时的callbackint (*release)(struct device *dev);// 用户在设备/dev/rtc1上调用close时的callbackint (*ioctl)(struct device *dev, uint cmd, ulong arg); // 用户在设备/dev/rtc1上调用ioctl时的callbackint (*read_time)(struct device *dev, struct rtc_time *tm); // rtc内核的回调函数int (*set_time)(struct device *dev, struct rtc_time *tm);int (*read_alarm)(struct device *dev, struct rtc_wkalrm *alarm);int (*set_alarm)(struct device *dev, struct rtc_wkalrm *alarm);int (*read_callback)(struct device *dev, int data); // 用户在设备/dev/rtc1上调用read时的callbackint (*alarm_irq_enable)(struct device *dev, uint enabled);
};// rtc模块提供了宏to_rtc_device(d)把 struct device结构转变为struct rtc_device{}结构

读取和设置时间 :

驱动程序负责提供读取和设置设备时间的函数。RTC总是以二进制编码的格式存储/恢复时间,其中每个4位表示0~9而非0~F,内核提供bcd2bin()和bin2bcd()两个转换的宏。提供辅助函数rtc_valid_tm(struct rtm_time *)用于确定struct rtc_time{}表示合法的时间。

函数使用示例:

static int foo_rtc_read_time(struct device *dev, struct rtc_time *tm) {struct foo_regs regs;int error = foo_device_read(dev, &regs, 0, sizeof(regs));if (error) return error;tm->tm_sec = bcd2bin(regs.seconds);tm->tm_min = bcd2bin(regs.minutes);tm->tm_hour = bcd2bin(regs.cent_hours);tm->tm_mday = bcd2bin(regs.date);tm->tm_wday = bcd2bin(regs.day) - 1;tm->tm_mon = bcd2bin(regs.month) - 1;tm->tm_year = bcd2bin(regs.years) + 100; // 设备的epoch是2000,加回到1900return rtc_valid_tm(tm);
}static int foo_rtc_set_time(struct device *dev, struct rtc_time *tm) {struct foo_regs regs;regs.seconds = bin2bcd(tm->tm_sec);regs.minutes = bin2bcd(tm->tm_min);regs.cent_hours = bin2bcd(tm->tm_hour);regs.day = bin2bcd(tm->tm_wday + 1);regs.date = bin2bcd(tm->tm_mday);regs.month = bin2bcd(tm->tm_mon + 1);regs.cent_hours |= BQ32K_CENT;regs.years = bin2bcd(tm->tm_year % 100);return write_into_device(dev, &regs, 0, sizeof(regs));
}

三.RTC用户空间

 rtc在sys中拥有一个类,可以查看一些信息。

  • name 查看rtc型号

  • range 查看rtc支持的起始时间和结束时间

  • date 查看rtc当前日期

  • time 查看rtc当前时间

  • since_epoch 查看当前rtc时间距离epoch经历了多少秒,epoch是一个时间点1970 年 1 月 1 日凌晨零点零分零秒

  • hctosys 这个rtc是否在上电时候同步设置系统时间

  • max_user_freq 可读可写用于查看和设置RTC周期中断的最大频率,一般是1hz

  • offset 查看和设置当前的rtc校准精度的偏移值,和回调read_offset、set_offset有关,注意是ppb为单位,而且是可正可负

  • wakealarm 查看和设置闹钟时间,有的rtc会被隐藏起来这个属性,sys属性如何隐藏可以参考rtc_attr_group、rtc_attr_is_visible、rtc_does_wakealarm这三个。

在注册rtc驱动的时候,会自动在proc注册一个文件/proc/driver/rtc,此文件只可以读

http://www.lryc.cn/news/307362.html

相关文章:

  • 番外篇 | YOLOv5+DeepSort实现行人目标跟踪检测
  • 认识Sass
  • YOLOv9-Openvino和ONNXRuntime推理【CPU】
  • AIGC 架构:RAG (retrieval augumented generation) 应用可以使用 PostgreSQL 作为向量数据库组件吗?
  • leetcode:134.加油站
  • uniapp的微信小程序授权头像昵称(最新版)
  • Spring Boot到底是如何进行自动配置的?
  • 【王道数据结构】【chapter7查找】【P285t5】
  • 个人玩航拍,如何申请无人机空域?
  • ChatGPT带火的HBM是什么?
  • 10 款数据恢复软件功能和有效性对比(2024 年更新)
  • Python 与 pdfplumber:高效自动读取 PDF 的解决方案
  • Flutter 启动流程解析
  • 全量知识系统问题及SmartChat给出的答复 之4
  • Java架构师之路七、大数据:Hadoop、Spark、Hive、HBase、Kafka等
  • 图论基础(一)
  • 使用 React 和 MUI 创建多选 Checkbox 树组件
  • vue3里面使用el-image-vie出现图片预览导致页面卡顿停止加载问题
  • Leetcoder Day26| 回溯part06:总结+三道hard题
  • 浅谈 Linux 网络编程 - 网络字节序
  • Nginx网络服务六-----IP透传、调度算法和负载均衡
  • 【Linux进程】进程状态---进程僵尸与孤儿
  • MySQL数据库基础知识总结(适合小白入门使用)一
  • 历史新知网:寄快递寄个电脑显示器要多少钱?
  • 在两台CentOS 7服务器上部署MinIO集群。
  • 【计算机网络】深度学习使用应用层的HTTP协议
  • Ubuntu18.04 系统上配置并运行SuperGluePretrainedNetwork(仅使用CPU)
  • 协议-http协议-基础概念01-发展历程-http组成-http是什么-相关的应用-相关的协议
  • UI学习-学习内容
  • Flink CDC 提取记录变更时间作为事件时间和 Hudi 表的 precombine.field 以及1970-01-01 取值问题