Skip to content
Navigation Menu
Toggle navigation
Sign in
In this repository
All GitHub Enterprise
↵
Jump to
↵
No suggested jump to results
In this repository
All GitHub Enterprise
↵
Jump to
↵
In this organization
All GitHub Enterprise
↵
Jump to
↵
In this repository
All GitHub Enterprise
↵
Jump to
↵
Sign in
Reseting focus
You signed in with another tab or window.
Reload
to refresh your session.
You signed out in another tab or window.
Reload
to refresh your session.
You switched accounts on another tab or window.
Reload
to refresh your session.
Dismiss alert
{{ message }}
mariux64
/
linux
Public
Notifications
You must be signed in to change notification settings
Fork
0
Star
0
Code
Issues
1
Pull requests
0
Actions
Projects
0
Wiki
Security
Insights
Additional navigation options
Code
Issues
Pull requests
Actions
Projects
Wiki
Security
Insights
Files
eff82a2
Documentation
LICENSES
arch
block
certs
crypto
drivers
fs
include
init
ipc
kernel
lib
mm
net
samples
scripts
security
sound
tools
accounting
arch
bpf
build
cgroup
crypto
debugging
firewire
firmware
gpio
hv
iio
include
io_uring
kvm
laptop
leds
lib
memory-model
nfsd
objtool
pci
pcmcia
perf
power
scripts
spi
testing
fault-injection
ktest
nvdimm
radix-tree
scatterlist
selftests
android
bpf
breakpoints
capabilities
cgroup
cpu-hotplug
cpufreq
drivers
efivarfs
exec
filesystems
firmware
ftrace
futex
gpio
ia64
intel_pstate
ipc
ir
kcmp
kexec
kmod
kselftest
kvm
lib
livepatch
locking
media_tests
membarrier
memfd
memory-hotplug
mount
mqueue
net
netfilter
networking
nsfs
ntb
pidfd
powerpc
prctl
proc
pstore
ptp
ptrace
rcutorture
rseq
rtc
.gitignore
Makefile
rtctest.c
setdate.c
safesetid
seccomp
sigaltstack
size
sparc64
splice
static_keys
sync
sysctl
tc-testing
timers
tmpfs
tpm2
uevent
user
vDSO
vm
watchdog
x86
zram
.gitignore
Makefile
gen_kselftest_tar.sh
kselftest.h
kselftest_harness.h
kselftest_install.sh
kselftest_module.h
kselftest_module.sh
lib.mk
vsock
thermal
time
usb
virtio
vm
wmi
Makefile
usr
virt
.clang-format
.cocciconfig
.get_maintainer.ignore
.gitattributes
.gitignore
.mailmap
COPYING
CREDITS
Kbuild
Kconfig
MAINTAINERS
Makefile
README
Breadcrumbs
linux
/
tools
/
testing
/
selftests
/
rtc
/
rtctest.c
Copy path
Blame
Blame
Latest commit
History
History
337 lines (262 loc) · 7.15 KB
Breadcrumbs
linux
/
tools
/
testing
/
selftests
/
rtc
/
rtctest.c
Top
File metadata and controls
Code
Blame
337 lines (262 loc) · 7.15 KB
Raw
// SPDX-License-Identifier: GPL-2.0 /* * Real Time Clock Driver Test Program * * Copyright (c) 2018 Alexandre Belloni <alexandre.belloni@bootlin.com> */ #include <errno.h> #include <fcntl.h> #include <linux/rtc.h> #include <stdio.h> #include <stdlib.h> #include <sys/ioctl.h> #include <sys/time.h> #include <sys/types.h> #include <time.h> #include <unistd.h> #include "../kselftest_harness.h" #define NUM_UIE 3 #define ALARM_DELTA 3 static char *rtc_file = "/dev/rtc0"; FIXTURE(rtc) { int fd; }; FIXTURE_SETUP(rtc) { self->fd = open(rtc_file, O_RDONLY); ASSERT_NE(-1, self->fd); } FIXTURE_TEARDOWN(rtc) { close(self->fd); } TEST_F(rtc, date_read) { int rc; struct rtc_time rtc_tm; /* Read the RTC time/date */ rc = ioctl(self->fd, RTC_RD_TIME, &rtc_tm); ASSERT_NE(-1, rc); TH_LOG("Current RTC date/time is %02d/%02d/%02d %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); } TEST_F_TIMEOUT(rtc, uie_read, NUM_UIE + 2) { int i, rc, irq = 0; unsigned long data; /* Turn on update interrupts */ rc = ioctl(self->fd, RTC_UIE_ON, 0); if (rc == -1) { ASSERT_EQ(EINVAL, errno); TH_LOG("skip update IRQs not supported."); return; } for (i = 0; i < NUM_UIE; i++) { /* This read will block */ rc = read(self->fd, &data, sizeof(data)); ASSERT_NE(-1, rc); irq++; } EXPECT_EQ(NUM_UIE, irq); rc = ioctl(self->fd, RTC_UIE_OFF, 0); ASSERT_NE(-1, rc); } TEST_F(rtc, uie_select) { int i, rc, irq = 0; unsigned long data; /* Turn on update interrupts */ rc = ioctl(self->fd, RTC_UIE_ON, 0); if (rc == -1) { ASSERT_EQ(EINVAL, errno); TH_LOG("skip update IRQs not supported."); return; } for (i = 0; i < NUM_UIE; i++) { struct timeval tv = { .tv_sec = 2 }; fd_set readfds; FD_ZERO(&readfds); FD_SET(self->fd, &readfds); /* The select will wait until an RTC interrupt happens. */ rc = select(self->fd + 1, &readfds, NULL, NULL, &tv); ASSERT_NE(-1, rc); ASSERT_NE(0, rc); /* This read won't block */ rc = read(self->fd, &data, sizeof(unsigned long)); ASSERT_NE(-1, rc); irq++; } EXPECT_EQ(NUM_UIE, irq); rc = ioctl(self->fd, RTC_UIE_OFF, 0); ASSERT_NE(-1, rc); } TEST_F(rtc, alarm_alm_set) { struct timeval tv = { .tv_sec = ALARM_DELTA + 2 }; unsigned long data; struct rtc_time tm; fd_set readfds; time_t secs, new; int rc; rc = ioctl(self->fd, RTC_RD_TIME, &tm); ASSERT_NE(-1, rc); secs = timegm((struct tm *)&tm) + ALARM_DELTA; gmtime_r(&secs, (struct tm *)&tm); rc = ioctl(self->fd, RTC_ALM_SET, &tm); if (rc == -1) { ASSERT_EQ(EINVAL, errno); TH_LOG("skip alarms are not supported."); return; } rc = ioctl(self->fd, RTC_ALM_READ, &tm); ASSERT_NE(-1, rc); TH_LOG("Alarm time now set to %02d:%02d:%02d.", tm.tm_hour, tm.tm_min, tm.tm_sec); /* Enable alarm interrupts */ rc = ioctl(self->fd, RTC_AIE_ON, 0); ASSERT_NE(-1, rc); FD_ZERO(&readfds); FD_SET(self->fd, &readfds); rc = select(self->fd + 1, &readfds, NULL, NULL, &tv); ASSERT_NE(-1, rc); ASSERT_NE(0, rc); /* Disable alarm interrupts */ rc = ioctl(self->fd, RTC_AIE_OFF, 0); ASSERT_NE(-1, rc); rc = read(self->fd, &data, sizeof(unsigned long)); ASSERT_NE(-1, rc); TH_LOG("data: %lx", data); rc = ioctl(self->fd, RTC_RD_TIME, &tm); ASSERT_NE(-1, rc); new = timegm((struct tm *)&tm); ASSERT_EQ(new, secs); } TEST_F(rtc, alarm_wkalm_set) { struct timeval tv = { .tv_sec = ALARM_DELTA + 2 }; struct rtc_wkalrm alarm = { 0 }; struct rtc_time tm; unsigned long data; fd_set readfds; time_t secs, new; int rc; rc = ioctl(self->fd, RTC_RD_TIME, &alarm.time); ASSERT_NE(-1, rc); secs = timegm((struct tm *)&alarm.time) + ALARM_DELTA; gmtime_r(&secs, (struct tm *)&alarm.time); alarm.enabled = 1; rc = ioctl(self->fd, RTC_WKALM_SET, &alarm); if (rc == -1) { ASSERT_EQ(EINVAL, errno); TH_LOG("skip alarms are not supported."); return; } rc = ioctl(self->fd, RTC_WKALM_RD, &alarm); ASSERT_NE(-1, rc); TH_LOG("Alarm time now set to %02d/%02d/%02d %02d:%02d:%02d.", alarm.time.tm_mday, alarm.time.tm_mon + 1, alarm.time.tm_year + 1900, alarm.time.tm_hour, alarm.time.tm_min, alarm.time.tm_sec); FD_ZERO(&readfds); FD_SET(self->fd, &readfds); rc = select(self->fd + 1, &readfds, NULL, NULL, &tv); ASSERT_NE(-1, rc); ASSERT_NE(0, rc); rc = read(self->fd, &data, sizeof(unsigned long)); ASSERT_NE(-1, rc); rc = ioctl(self->fd, RTC_RD_TIME, &tm); ASSERT_NE(-1, rc); new = timegm((struct tm *)&tm); ASSERT_EQ(new, secs); } TEST_F_TIMEOUT(rtc, alarm_alm_set_minute, 65) { struct timeval tv = { .tv_sec = 62 }; unsigned long data; struct rtc_time tm; fd_set readfds; time_t secs, new; int rc; rc = ioctl(self->fd, RTC_RD_TIME, &tm); ASSERT_NE(-1, rc); secs = timegm((struct tm *)&tm) + 60 - tm.tm_sec; gmtime_r(&secs, (struct tm *)&tm); rc = ioctl(self->fd, RTC_ALM_SET, &tm); if (rc == -1) { ASSERT_EQ(EINVAL, errno); TH_LOG("skip alarms are not supported."); return; } rc = ioctl(self->fd, RTC_ALM_READ, &tm); ASSERT_NE(-1, rc); TH_LOG("Alarm time now set to %02d:%02d:%02d.", tm.tm_hour, tm.tm_min, tm.tm_sec); /* Enable alarm interrupts */ rc = ioctl(self->fd, RTC_AIE_ON, 0); ASSERT_NE(-1, rc); FD_ZERO(&readfds); FD_SET(self->fd, &readfds); rc = select(self->fd + 1, &readfds, NULL, NULL, &tv); ASSERT_NE(-1, rc); ASSERT_NE(0, rc); /* Disable alarm interrupts */ rc = ioctl(self->fd, RTC_AIE_OFF, 0); ASSERT_NE(-1, rc); rc = read(self->fd, &data, sizeof(unsigned long)); ASSERT_NE(-1, rc); TH_LOG("data: %lx", data); rc = ioctl(self->fd, RTC_RD_TIME, &tm); ASSERT_NE(-1, rc); new = timegm((struct tm *)&tm); ASSERT_EQ(new, secs); } TEST_F_TIMEOUT(rtc, alarm_wkalm_set_minute, 65) { struct timeval tv = { .tv_sec = 62 }; struct rtc_wkalrm alarm = { 0 }; struct rtc_time tm; unsigned long data; fd_set readfds; time_t secs, new; int rc; rc = ioctl(self->fd, RTC_RD_TIME, &alarm.time); ASSERT_NE(-1, rc); secs = timegm((struct tm *)&alarm.time) + 60 - alarm.time.tm_sec; gmtime_r(&secs, (struct tm *)&alarm.time); alarm.enabled = 1; rc = ioctl(self->fd, RTC_WKALM_SET, &alarm); if (rc == -1) { ASSERT_EQ(EINVAL, errno); TH_LOG("skip alarms are not supported."); return; } rc = ioctl(self->fd, RTC_WKALM_RD, &alarm); ASSERT_NE(-1, rc); TH_LOG("Alarm time now set to %02d/%02d/%02d %02d:%02d:%02d.", alarm.time.tm_mday, alarm.time.tm_mon + 1, alarm.time.tm_year + 1900, alarm.time.tm_hour, alarm.time.tm_min, alarm.time.tm_sec); FD_ZERO(&readfds); FD_SET(self->fd, &readfds); rc = select(self->fd + 1, &readfds, NULL, NULL, &tv); ASSERT_NE(-1, rc); ASSERT_NE(0, rc); rc = read(self->fd, &data, sizeof(unsigned long)); ASSERT_NE(-1, rc); rc = ioctl(self->fd, RTC_RD_TIME, &tm); ASSERT_NE(-1, rc); new = timegm((struct tm *)&tm); ASSERT_EQ(new, secs); } static void __attribute__((constructor)) __constructor_order_last(void) { if (!__constructor_order) __constructor_order = _CONSTRUCTOR_ORDER_BACKWARD; } int main(int argc, char **argv) { switch (argc) { case 2: rtc_file = argv[1]; /* FALLTHROUGH */ case 1: break; default: fprintf(stderr, "usage: %s [rtcdev]\n", argv[0]); return 1; } return test_harness_run(argc, argv); }
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
You can’t perform that action at this time.