/**********************************************************************
* 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下如何获取sd卡中的mbr

    答:使用dd命令,示例如下: dd if=/dev/mmcblk0 of=mbr.bin bs=512 count=1 解析: bs表示指定输入输出的块大小为512个字节 count表示指定读取输入的 ...

  2. 源码编译PHP提示zip错误

    本文来源:https://segmentfault.com/q/1010000002696561 /home/levi/soft/php-5.6.8/ext/zip/lib/zipint.h:118: ...

  3. JSON and XML Serialization in ASP.NET Web API

    https://docs.microsoft.com/en-us/aspnet/web-api/overview/formats-and-model-binding/json-and-xml-seri ...

  4. JavaScript的动态特性(通过eval,call,apply和bind来体现)

    JavaScript的动态特性(通过eval,call,apply和bind来体现) JavaScript是一种基于面向对象的.函数式的.动态的编程语言.现在发展到已经可以用在浏览器和服务器端了. 这 ...

  5. 酷到没朋友—— Cafflano便携式手磨手冲一体壶

    又一款外国新玩具~ 设计紧凑,手磨.滤架.滤壶融合的毫无ps痕迹! 简直是出差旅行,杀人越货必备良品!废话不多说,上图: 肿么样,一壶在手,天下我有~~~哈哈哈~~~

  6. 解决 对路径bin\roslyn..的访问被拒绝

    使用visual studio开发,一重新编译就会报错: 对路径“bin\roslyn\System.Reflection.Metadata.dll”的访问被拒绝 一开始的解决办法就是把bin下的文件 ...

  7. 关于JNDI那点事

    一.JNDI是什么? JNDI--Java 命名和目录接口(Java Naming and Directory Interface),是一组在Java应用中访问命名和目录服务的API. 二.JNDI好 ...

  8. pg_ctl -- 启动、停止、重启 PostgreSQL

    pg_ctl 名称 pg_ctl -- 启动.停止.重启 PostgreSQL 语法 pg_ctl start [-w] [-s] [-D datadir] [-l filename] [-o opt ...

  9. 【三小时学会Kubernetes!(零) 】系统结构及相关示例微服务介绍

    写在前面 牢牢占据容器技术统治地位的 Kubernetes,其重要性想必不言而喻,我保证本文是最详尽的 Kubernetes 技术文档,从我在后台排版了这么漫长的时间就能看出来.废话不多说 — — 以 ...

  10. 为什么CPU要从单核发展到多核?

    前言 这里首先直接给出结论:CPU从单核发展到多核的原因是如果维持单核,则为了提高CPU性能只能不断提高时钟频率,从而会导致CPU功耗急速上升,导致机箱过热,来不及散热. 历史 2004年,Intel ...