/**********************************************************************
* Linux RTC Test Example rtctest.c hacking
* 说明:
* 之前的Linux版本都没有这种针对硬件的测试代码示例,比较新的内核里
* 好像对了这些内核测试用例。
*
* 2017-8-15 深圳 龙华樟坑村 曾剑锋
*********************************************************************/ // 参考文档:
// 1. https://github.com/torvalds/linux/tree/master/tools/testing/selftests
// 2. Real Time Clock (RTC) Drivers for Linux
// https://android.googlesource.com/kernel/common/+/android-4.4/Documentation/rtc.txt
/*
* Real Time Clock Driver Test/Example Program
*
* Compile with:
* gcc -s -Wall -Wstrict-prototypes rtctest.c -o rtctest
*
* Copyright (C) 1996, Paul Gortmaker.
*
* Released under the GNU General Public License, version 2,
* included herein by reference.
*
*/ #include <stdio.h>
#include <linux/rtc.h>
#include <sys/ioctl.h>
#include <sys/time.h>
#include <sys/types.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdlib.h>
#include <errno.h> #ifndef ARRAY_SIZE
# define ARRAY_SIZE(x) (sizeof(x) / sizeof((x)[]))
#endif /*
* This expects the new RTC class driver framework, working with
* clocks that will often not be clones of what the PC-AT had.
* Use the command line to specify another RTC if you need one.
*/
// 默认rtc设备节点
static const char default_rtc[] = "/dev/rtc0"; // 后面用于设置rtc时间,获取rtc时间,然后对比设置、获取的时间来判断是否正确
static struct rtc_time cutoff_dates[] = {
{
.tm_year = , /* 1970 -1900 */
.tm_mday = ,
},
/* signed time_t 19/01/2038 3:14:08 */
{
.tm_year = ,
.tm_mday = ,
},
{
.tm_year = ,
.tm_mday = ,
},
{
.tm_year = , /* 2099 -1900 */
.tm_mday = ,
},
{
.tm_year = , /* 2100 -1900 */
.tm_mday = ,
},
/* unsigned time_t 07/02/2106 7:28:15*/
{
.tm_year = ,
.tm_mon = ,
.tm_mday = ,
},
{
.tm_year = ,
.tm_mon = ,
.tm_mday = ,
},
/* signed time on 64bit in nanoseconds 12/04/2262 01:47:16*/
{
.tm_year = ,
.tm_mon = ,
.tm_mday = ,
},
{
.tm_year = , /* 2262 -1900 */
.tm_mon = ,
.tm_mday = ,
},
}; // 时间对比函数
static int compare_dates(struct rtc_time *a, struct rtc_time *b)
{
if (a->tm_year != b->tm_year ||
a->tm_mon != b->tm_mon ||
a->tm_mday != b->tm_mday ||
a->tm_hour != b->tm_hour ||
a->tm_min != b->tm_min ||
((b->tm_sec - a->tm_sec) > ))
return ; return ;
} int main(int argc, char **argv)
{
int i, fd, retval, irqcount = , dangerous = ;
unsigned long tmp, data;
struct rtc_time rtc_tm;
const char *rtc = default_rtc;
struct timeval start, end, diff; // 这里的倒序的case可以借鉴
switch (argc) {
case :
if (*argv[] == 'd')
dangerous = ; // 没搞懂这里为什么叫dangerous
case :
rtc = argv[]; // rtc设备节点
/* FALLTHROUGH */
case :
break;
default:
fprintf(stderr, "usage: rtctest [rtcdev] [d]\n");
return ;
} fd = open(rtc, O_RDONLY); if (fd == -) {
perror(rtc);
exit(errno);
} fprintf(stderr, "\n\t\t\tRTC Driver Test Example.\n\n"); /* Turn on update interrupts (one per second) */
// 这里算是自动时间更新中断,有些芯片可能不支持的,我现在使用的就不支持
retval = ioctl(fd, RTC_UIE_ON, );
if (retval == -) {
if (errno == EINVAL) {
fprintf(stderr,
"\n...Update IRQs not supported.\n");
goto test_READ;
}
perror("RTC_UIE_ON ioctl");
exit(errno);
} // 直接读取
fprintf(stderr, "Counting 5 update (1/sec) interrupts from reading %s:",
rtc);
fflush(stderr);
for (i=; i<; i++) {
/* This read will block */
retval = read(fd, &data, sizeof(unsigned long));
if (retval == -) {
perror("read");
exit(errno);
}
fprintf(stderr, " %d",i);
fflush(stderr);
irqcount++;
} // select读取
fprintf(stderr, "\nAgain, from using select(2) on /dev/rtc:");
fflush(stderr);
for (i=; i<; i++) {
struct timeval tv = {, }; /* 5 second timeout on select */
fd_set readfds; FD_ZERO(&readfds);
FD_SET(fd, &readfds);
/* The select will wait until an RTC interrupt happens. */
retval = select(fd+, &readfds, NULL, NULL, &tv);
if (retval == -) {
perror("select");
exit(errno);
}
/* This read won't block unlike the select-less case above. */
retval = read(fd, &data, sizeof(unsigned long));
if (retval == -) {
perror("read");
exit(errno);
}
fprintf(stderr, " %d",i);
fflush(stderr);
irqcount++;
} /* Turn off update interrupts */
retval = ioctl(fd, RTC_UIE_OFF, );
if (retval == -) {
perror("RTC_UIE_OFF ioctl");
exit(errno);
} test_READ:
/* Read the RTC time/date */
// 读取RTC时间
retval = ioctl(fd, RTC_RD_TIME, &rtc_tm);
if (retval == -) {
perror("RTC_RD_TIME ioctl");
exit(errno);
} fprintf(stderr, "\n\nCurrent RTC date/time is %d-%d-%d, %02d:%02d:%02d.\n",
rtc_tm.tm_mday, rtc_tm.tm_mon + , rtc_tm.tm_year + ,
rtc_tm.tm_hour, rtc_tm.tm_min, rtc_tm.tm_sec); /* Set the alarm to 5 sec in the future, and check for rollover */
rtc_tm.tm_sec += ;
if (rtc_tm.tm_sec >= ) {
rtc_tm.tm_sec %= ;
rtc_tm.tm_min++;
}
if (rtc_tm.tm_min == ) {
rtc_tm.tm_min = ;
rtc_tm.tm_hour++;
}
if (rtc_tm.tm_hour == )
rtc_tm.tm_hour = ; // 设置闹钟
retval = ioctl(fd, RTC_ALM_SET, &rtc_tm);
if (retval == -) {
if (errno == EINVAL) {
fprintf(stderr,
"\n...Alarm IRQs not supported.\n");
goto test_PIE;
} perror("RTC_ALM_SET ioctl");
exit(errno);
} /* Read the current alarm settings */
retval = ioctl(fd, RTC_ALM_READ, &rtc_tm);
if (retval == -) {
perror("RTC_ALM_READ ioctl");
exit(errno);
} fprintf(stderr, "Alarm time now set to %02d:%02d:%02d.\n",
rtc_tm.tm_hour, rtc_tm.tm_min, rtc_tm.tm_sec); /* Enable alarm interrupts */
retval = ioctl(fd, RTC_AIE_ON, );
if (retval == -) {
if (errno == EINVAL) {
fprintf(stderr,
"\n...Alarm IRQs not supported.\n");
goto test_PIE;
} perror("RTC_AIE_ON ioctl");
exit(errno);
} fprintf(stderr, "Waiting 5 seconds for alarm...");
fflush(stderr);
/* This blocks until the alarm ring causes an interrupt */
retval = read(fd, &data, sizeof(unsigned long));
if (retval == -) {
perror("read");
exit(errno);
}
irqcount++;
fprintf(stderr, " okay. Alarm rang.\n"); /* Disable alarm interrupts */
retval = ioctl(fd, RTC_AIE_OFF, );
if (retval == -) {
perror("RTC_AIE_OFF ioctl");
exit(errno);
} test_PIE:
/* Read periodic IRQ rate */
retval = ioctl(fd, RTC_IRQP_READ, &tmp);
if (retval == -) {
/* not all RTCs support periodic IRQs */
if (errno == EINVAL) {
fprintf(stderr, "\nNo periodic IRQ support\n");
goto test_DATE;
}
perror("RTC_IRQP_READ ioctl");
exit(errno);
}
fprintf(stderr, "\nPeriodic IRQ rate is %ldHz.\n", tmp); fprintf(stderr, "Counting 20 interrupts at:");
fflush(stderr); /* The frequencies 128Hz, 256Hz, ... 8192Hz are only allowed for root. */
for (tmp=; tmp<=; tmp*=) { retval = ioctl(fd, RTC_IRQP_SET, tmp);
if (retval == -) {
/* not all RTCs can change their periodic IRQ rate */
if (errno == EINVAL) {
fprintf(stderr,
"\n...Periodic IRQ rate is fixed\n");
goto test_DATE;
}
perror("RTC_IRQP_SET ioctl");
exit(errno);
} fprintf(stderr, "\n%ldHz:\t", tmp);
fflush(stderr); /* Enable periodic interrupts */
retval = ioctl(fd, RTC_PIE_ON, );
if (retval == -) {
perror("RTC_PIE_ON ioctl");
exit(errno);
} for (i=; i<; i++) {
gettimeofday(&start, NULL);
/* This blocks */
retval = read(fd, &data, sizeof(unsigned long));
if (retval == -) {
perror("read");
exit(errno);
}
gettimeofday(&end, NULL);
timersub(&end, &start, &diff);
if (diff.tv_sec > ||
diff.tv_usec > ((1000000L / tmp) * 1.10)) {
fprintf(stderr, "\nPIE delta error: %ld.%06ld should be close to 0.%06ld\n",
diff.tv_sec, diff.tv_usec,
(1000000L / tmp));
fflush(stdout);
exit(-);
} fprintf(stderr, " %d",i);
fflush(stderr);
irqcount++;
} /* Disable periodic interrupts */
retval = ioctl(fd, RTC_PIE_OFF, );
if (retval == -) {
perror("RTC_PIE_OFF ioctl");
exit(errno);
}
} test_DATE:
if (!dangerous)
goto done; fprintf(stderr, "\nTesting problematic dates\n"); for (i = ; i < ARRAY_SIZE(cutoff_dates); i++) {
struct rtc_time current; /* Write the new date in RTC */
// 设置RTC时间
retval = ioctl(fd, RTC_SET_TIME, &cutoff_dates[i]);
if (retval == -) {
perror("RTC_SET_TIME ioctl");
close(fd);
exit(errno);
} /* Read back */
// 读取RTC时间
retval = ioctl(fd, RTC_RD_TIME, &current);
if (retval == -) {
perror("RTC_RD_TIME ioctl");
exit(errno);
} // 对比RTC时间
if(compare_dates(&cutoff_dates[i], &current)) {
fprintf(stderr,"Setting date %d failed\n",
cutoff_dates[i].tm_year + );
goto done;
} cutoff_dates[i].tm_sec += ; /* Write the new alarm in RTC */
// 设置闹钟时间
retval = ioctl(fd, RTC_ALM_SET, &cutoff_dates[i]);
if (retval == -) {
perror("RTC_ALM_SET ioctl");
close(fd);
exit(errno);
} /* Read back */
// 读取闹钟时间
retval = ioctl(fd, RTC_ALM_READ, &current);
if (retval == -) {
perror("RTC_ALM_READ ioctl");
exit(errno);
} // 对比闹钟时间
if(compare_dates(&cutoff_dates[i], &current)) {
fprintf(stderr,"Setting alarm %d failed\n",
cutoff_dates[i].tm_year + );
goto done;
} fprintf(stderr, "Setting year %d is OK \n",
cutoff_dates[i].tm_year + );
}
done:
fprintf(stderr, "\n\n\t\t\t *** Test complete ***\n"); close(fd); return ;
}

Linux RTC Test Example rtctest.c hacking的更多相关文章

  1. linux rtc 接口【转】

    转自:http://blog.csdn.net/goldfighter/article/details/6126178 Linux操作系统内核对RTC的编程详解 转自: http://xenyinze ...

  2. linux RTC 驱动模型分析【转】

    转自:http://blog.csdn.net/yaozhenguo2006/article/details/6824970 RTC(real time clock)实时时钟,主要作用是给Linux系 ...

  3. ti processor sdk linux am335x evm /bin/setup-host-check.sh hacking

    #!/bin/sh # # ti processor sdk linux am335x evm /bin/setup-host-check.sh hacking # 说明: # 本文主要对TI的sdk ...

  4. ti processor sdk linux am335x evm /bin/create-sdcard.sh hacking

    #!/bin/bash # # ti processor sdk linux am335x evm /bin/create-sdcard.sh hacking # 说明: # 本文主要对TI的sdk中 ...

  5. ti processor sdk linux am335x evm /bin/unshallow-repositories.sh hacking

    #!/bin/bash # # ti processor sdk linux am335x evm /bin/unshallow-repositories.sh hacking # 说明: # 本文主 ...

  6. ti processor sdk linux am335x evm /bin/setup-package-install.sh hacking

    #!/bin/sh # # ti processor sdk linux am335x evm /bin/setup-package-install.sh hacking # 说明: # 本文主要对T ...

  7. ti processor sdk linux am335x evm /bin/setup-uboot-env.sh hacking

    #!/bin/sh # # ti processor sdk linux am335x evm /bin/setup-uboot-env.sh hacking # 说明: # 本文主要对TI的sdk中 ...

  8. ti processor sdk linux am335x evm /bin/setup-minicom.sh hacking

    #!/bin/sh # # ti processor sdk linux am335x evm /bin/setup-minicom.sh hacking # 说明: # 本文主要对TI的sdk中的s ...

  9. ti processor sdk linux am335x evm /bin/setup-targetfs-nfs.sh hacking

    #!/bin/sh # # ti processor sdk linux am335x evm /bin/setup-targetfs-nfs.sh hacking # 说明: # 本文主要对TI的s ...

随机推荐

  1. 配置linux内核输出所有的log信息

    答案:修改内核文件include/linux/printk.h  (内核版本为4.9) 修改前 #define CONSOLE_LOGLEVEL_DEFAULT 7 /* anything MORE ...

  2. <The old man and the sea>

    Every day is a new day. It is better to be lucky. But i would rather be exact. Then when luck comes ...

  3. SEM关键词的三种分类方式

    关键词分类是为了使sem账户搭建结构清晰便于管理关键词.基于对需求人群的深入分析,每个账户都有其独特的分类方式,比如招商加盟行业更多的是地域分类,品牌类企业通常用词性分类即可,而冷门行业用人群分类比较 ...

  4. VirtualBox 安装增强功能失败 解决方法

    参考:virtualbox安装增强功能时[未能加载虚拟光盘] 在安装VirtualBox并配置虚拟机之后,想要安装增强功能:设备 => 安装增强功能. 出现未能加载虚拟光盘问题. 解决方法: 进 ...

  5. [小问题笔记(二)] 可能导致DropDownList的SelectedIndexChanged事件不触发的几种情况

    遇到SelectedIndexChanged事件没有触发,可以依次检查以下几种情况是否在程序中出现. 一.DropDownList的不同option设置了相同的value . 二.没有写 AutoPo ...

  6. php入门(三)

    PHP 面向对象: 在php5中 var就是public的别名. 变量 $this 代表自身的对象. PHP_EOL;为换行符 构造函数+析构函数 <?php class Site { /* 成 ...

  7. JavaScript权威指南--语句

    知识要点 在javascript中,表达式是短语,那么语句(statement)就是整句或命令.表达式计算出一个值,但语句用来执行以使某件事发生. 1.表达式语句 具有副作用的表达式是JavaScri ...

  8. python 中str format 格式化数字补0方法

      >>> "{0:03d}".format(1)'001'>>> "{0:03d}".format(10)'010'> ...

  9. 分布式缓存集群方案特性使用场景(Memcache/Redis(Twemproxy/Codis/Redis-cluster))优缺点对比及选型

    分布式缓存集群方案特性使用场景(Memcache/Redis(Twemproxy/Codis/Redis-cluster))优缺点对比及选型   分布式缓存特性: 1) 高性能:当传统数据库面临大规模 ...

  10. windows service 2008 内存吃尽解决方案

    最近用windows2008,发现系统运行一段时间后,内存吃紧,赶紧打开资源查看器,发现当前运行的程序占有内存都很小,后经查资料,原来是被windows2008的文件缓存吃尽了,这是windows20 ...