You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
245 lines
5.9 KiB
245 lines
5.9 KiB
/* rtc-test.c
|
|
*
|
|
* Tests for the Real Time Clock driver.
|
|
*
|
|
* Copyright (c) Larsen & Toubro Infotech Ltd., 2010
|
|
* Copyright (c) 2013 Oracle and/or its affiliates. All Rights Reserved.
|
|
*
|
|
* Author : Silesh C V <Silesh.Vellattu@lntinfotech.com>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See
|
|
* the GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
|
|
|
|
#include <sys/ioctl.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <fcntl.h>
|
|
#include <unistd.h>
|
|
#include <linux/rtc.h>
|
|
#include <errno.h>
|
|
#include <time.h>
|
|
|
|
#include "test.h"
|
|
#include "safe_macros.h"
|
|
|
|
int rtc_fd = -1;
|
|
char *TCID = "rtc01";
|
|
int TST_TOTAL = 3;
|
|
|
|
static char *rtc_dev = "/dev/rtc";
|
|
static int dflag;
|
|
static const option_t options[] = {
|
|
{"d:", &dflag, &rtc_dev},
|
|
{NULL, NULL, NULL}
|
|
};
|
|
|
|
static void help(void)
|
|
{
|
|
printf(" -d x rtc device node, default is %s\n",
|
|
rtc_dev);
|
|
}
|
|
|
|
/* Read and Alarm Tests : Read test reads the Date/time from RTC
|
|
* while Alarm test, sets the alarm to 5 seconds in future and
|
|
* waits for it to ring.The ioctls tested in these tests are
|
|
* RTC_RD_TIME, RTC_ALM_SET, RTC_ALM_READ, RTC_AIE_OFF */
|
|
|
|
void read_alarm_test(void)
|
|
{
|
|
struct rtc_time rtc_tm;
|
|
int ret;
|
|
unsigned long data;
|
|
fd_set rfds;
|
|
struct timeval tv;
|
|
|
|
tst_resm(TINFO, "RTC READ TEST:");
|
|
|
|
/*Read RTC Time */
|
|
ret = ioctl(rtc_fd, RTC_RD_TIME, &rtc_tm);
|
|
if (ret == -1) {
|
|
tst_resm(TFAIL | TERRNO, "RTC_RD_TIME ioctl failed");
|
|
return;
|
|
}
|
|
|
|
tst_resm(TPASS, "RTC READ TEST Passed");
|
|
|
|
tst_resm(TINFO, "Current RTC date/time is %d-%d-%d, %02d:%02d:%02d.",
|
|
rtc_tm.tm_mday, rtc_tm.tm_mon + 1, rtc_tm.tm_year + 1900,
|
|
rtc_tm.tm_hour, rtc_tm.tm_min, rtc_tm.tm_sec);
|
|
|
|
tst_resm(TINFO, "RTC ALARM TEST :");
|
|
|
|
/*set Alarm to 5 Seconds */
|
|
rtc_tm.tm_sec += 5;
|
|
if (rtc_tm.tm_sec >= 60) {
|
|
rtc_tm.tm_sec %= 60;
|
|
rtc_tm.tm_min++;
|
|
}
|
|
|
|
if (rtc_tm.tm_min == 60) {
|
|
rtc_tm.tm_min = 0;
|
|
rtc_tm.tm_hour++;
|
|
}
|
|
|
|
if (rtc_tm.tm_hour == 24)
|
|
rtc_tm.tm_hour = 0;
|
|
|
|
ret = ioctl(rtc_fd, RTC_ALM_SET, &rtc_tm);
|
|
if (ret == -1) {
|
|
if (errno == EINVAL)
|
|
tst_resm(TCONF | TERRNO, "RTC_ALM_SET not supported");
|
|
else
|
|
tst_resm(TFAIL | TERRNO, "RTC_ALM_SET ioctl failed");
|
|
return;
|
|
}
|
|
|
|
/*Read current alarm time */
|
|
ret = ioctl(rtc_fd, RTC_ALM_READ, &rtc_tm);
|
|
if (ret == -1) {
|
|
if (errno == EINVAL) {
|
|
tst_resm(TCONF | TERRNO, "RTC_ALM_READ not suported");
|
|
} else {
|
|
tst_resm(TFAIL | TERRNO, "RTC_ALM_READ ioctl failed");
|
|
return;
|
|
}
|
|
} else {
|
|
tst_resm(TINFO, "Alarm time set to %02d:%02d:%02d.",
|
|
rtc_tm.tm_hour, rtc_tm.tm_min, rtc_tm.tm_sec);
|
|
}
|
|
|
|
/* Enable alarm interrupts */
|
|
ret = ioctl(rtc_fd, RTC_AIE_ON, 0);
|
|
if (ret == -1) {
|
|
tst_resm(TINFO | TERRNO, "RTC_AIE_ON ioctl failed");
|
|
return;
|
|
}
|
|
|
|
tst_resm(TINFO, "Waiting 5 seconds for the alarm...");
|
|
|
|
tv.tv_sec = 6; /*set 6 seconds as the time out */
|
|
tv.tv_usec = 0;
|
|
|
|
FD_ZERO(&rfds);
|
|
FD_SET(rtc_fd, &rfds);
|
|
|
|
ret = select(rtc_fd + 1, &rfds, NULL, NULL, &tv); /*wait for alarm */
|
|
|
|
if (ret == -1) {
|
|
tst_resm(TFAIL | TERRNO, "select failed");
|
|
return;
|
|
} else if (ret) {
|
|
ret = read(rtc_fd, &data, sizeof(unsigned long));
|
|
if (ret == -1) {
|
|
tst_resm(TFAIL | TERRNO, "read failed");
|
|
return;
|
|
}
|
|
tst_resm(TINFO, "Alarm rang.");
|
|
} else {
|
|
tst_resm(TFAIL, "Timed out waiting for the alarm");
|
|
return;
|
|
}
|
|
|
|
/* Disable alarm interrupts */
|
|
ret = ioctl(rtc_fd, RTC_AIE_OFF, 0);
|
|
if (ret == -1) {
|
|
tst_resm(TFAIL | TERRNO, "RTC_AIE_OFF ioctl failed");
|
|
return;
|
|
}
|
|
tst_resm(TPASS, "RTC ALARM TEST Passed");
|
|
}
|
|
|
|
/* Update_interrupts_test :Once the Update interrupts is enabled,
|
|
* the RTC gives interrupts (1/sec) on the interrupts line(if the rtc
|
|
* has one). This is tested by enabling the update interrupts
|
|
* and then waiting for 5 interrupts.*/
|
|
|
|
void update_interrupts_test(void)
|
|
{
|
|
int ret, i;
|
|
unsigned long data;
|
|
fd_set rfds;
|
|
struct timeval tv;
|
|
|
|
tst_resm(TINFO, "RTC UPDATE INTERRUPTS TEST :");
|
|
/*Turn on update interrupts */
|
|
ret = ioctl(rtc_fd, RTC_UIE_ON, 0);
|
|
if (ret == -1) {
|
|
if (errno == EINVAL)
|
|
tst_resm(TCONF | TERRNO, "RTC_UIE_ON not supported");
|
|
else
|
|
tst_resm(TFAIL | TERRNO, "RTC_UIE_ON ioctl failed");
|
|
return;
|
|
}
|
|
|
|
tst_resm(TINFO, "Waiting for 5 update interrupts...");
|
|
for (i = 1; i < 6; i++) {
|
|
|
|
tv.tv_sec = 2; /*2 sec time out for each interrupt */
|
|
tv.tv_usec = 0;
|
|
|
|
FD_ZERO(&rfds);
|
|
FD_SET(rtc_fd, &rfds);
|
|
|
|
ret = select(rtc_fd + 1, &rfds, NULL, NULL, &tv);
|
|
if (ret == -1) {
|
|
tst_resm(TFAIL | TERRNO, "select failed");
|
|
return;
|
|
} else if (ret) {
|
|
ret = read(rtc_fd, &data, sizeof(unsigned long));
|
|
if (ret == -1) {
|
|
tst_resm(TFAIL | TERRNO, "read failed");
|
|
return;
|
|
}
|
|
tst_resm(TINFO, "Update interrupt %d", i);
|
|
} else {
|
|
tst_resm(TFAIL,
|
|
"Timed out waiting for the update interrupt");
|
|
return;
|
|
}
|
|
}
|
|
|
|
/* Turn off update interrupts */
|
|
ret = ioctl(rtc_fd, RTC_UIE_OFF, 0);
|
|
if (ret == -1) {
|
|
tst_resm(TFAIL | TERRNO, "RTC_UIE_OFF ioctl failed");
|
|
return;
|
|
}
|
|
tst_resm(TPASS, "RTC UPDATE INTERRUPTS TEST Passed");
|
|
}
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
tst_parse_opts(argc, argv, options, help);
|
|
|
|
tst_require_root();
|
|
|
|
if (access(rtc_dev, F_OK) == -1)
|
|
tst_brkm(TCONF, NULL, "couldn't find rtc device '%s'", rtc_dev);
|
|
|
|
rtc_fd = SAFE_OPEN(NULL, rtc_dev, O_RDONLY);
|
|
|
|
/*Read and alarm tests */
|
|
read_alarm_test();
|
|
|
|
/*Update interrupts test */
|
|
update_interrupts_test();
|
|
|
|
close(rtc_fd);
|
|
|
|
tst_resm(TINFO, "RTC Tests Done!");
|
|
tst_exit();
|
|
}
|