]> git.ipfire.org Git - thirdparty/systemd.git/blob - src/shared/hwclock.c
hwdb: Update database of Bluetooth company identifiers
[thirdparty/systemd.git] / src / shared / hwclock.c
1 /*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
2
3 /***
4 This file is part of systemd.
5
6 Copyright 2010-2012 Lennart Poettering
7
8 systemd is free software; you can redistribute it and/or modify it
9 under the terms of the GNU Lesser General Public License as published by
10 the Free Software Foundation; either version 2.1 of the License, or
11 (at your option) any later version.
12
13 systemd is distributed in the hope that it will be useful, but
14 WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 Lesser General Public License for more details.
17
18 You should have received a copy of the GNU Lesser General Public License
19 along with systemd; If not, see <http://www.gnu.org/licenses/>.
20 ***/
21
22 #include <assert.h>
23 #include <string.h>
24 #include <unistd.h>
25 #include <errno.h>
26 #include <stdlib.h>
27 #include <signal.h>
28 #include <stdio.h>
29 #include <sys/types.h>
30 #include <sys/stat.h>
31 #include <fcntl.h>
32 #include <sys/ioctl.h>
33 #include <stdarg.h>
34 #include <ctype.h>
35 #include <sys/prctl.h>
36 #include <sys/time.h>
37 #include <linux/rtc.h>
38
39 #include "macro.h"
40 #include "util.h"
41 #include "log.h"
42 #include "strv.h"
43 #include "hwclock.h"
44 #include "fileio.h"
45
46 int hwclock_get_time(struct tm *tm) {
47 _cleanup_close_ int fd = -1;
48
49 assert(tm);
50
51 fd = open("/dev/rtc", O_RDONLY|O_CLOEXEC);
52 if (fd < 0)
53 return -errno;
54
55 /* This leaves the timezone fields of struct tm
56 * uninitialized! */
57 if (ioctl(fd, RTC_RD_TIME, tm) < 0)
58 return -errno;
59
60 /* We don't know daylight saving, so we reset this in order not
61 * to confuse mktime(). */
62 tm->tm_isdst = -1;
63
64 return 0;
65 }
66
67 int hwclock_set_time(const struct tm *tm) {
68 _cleanup_close_ int fd = -1;
69
70 assert(tm);
71
72 fd = open("/dev/rtc", O_RDONLY|O_CLOEXEC);
73 if (fd < 0)
74 return -errno;
75
76 if (ioctl(fd, RTC_SET_TIME, tm) < 0)
77 return -errno;
78
79 return 0;
80 }
81
82 int hwclock_is_localtime(void) {
83 _cleanup_fclose_ FILE *f;
84
85 /*
86 * The third line of adjtime is "UTC" or "LOCAL" or nothing.
87 * # /etc/adjtime
88 * 0.0 0 0
89 * 0
90 * UTC
91 */
92 f = fopen("/etc/adjtime", "re");
93 if (f) {
94 char line[LINE_MAX];
95 bool b;
96
97 b = fgets(line, sizeof(line), f) &&
98 fgets(line, sizeof(line), f) &&
99 fgets(line, sizeof(line), f);
100 if (!b)
101 return -EIO;
102
103 truncate_nl(line);
104 return streq(line, "LOCAL");
105
106 } else if (errno != ENOENT)
107 return -errno;
108
109 return 0;
110 }
111
112 int hwclock_set_timezone(int *min) {
113 const struct timeval *tv_null = NULL;
114 struct timespec ts;
115 struct tm *tm;
116 int minutesdelta;
117 struct timezone tz;
118
119 assert_se(clock_gettime(CLOCK_REALTIME, &ts) == 0);
120 assert_se(tm = localtime(&ts.tv_sec));
121 minutesdelta = tm->tm_gmtoff / 60;
122
123 tz.tz_minuteswest = -minutesdelta;
124 tz.tz_dsttime = 0; /* DST_NONE*/
125
126 /*
127 * If the hardware clock does not run in UTC, but in local time:
128 * The very first time we set the kernel's timezone, it will warp
129 * the clock so that it runs in UTC instead of local time.
130 */
131 if (settimeofday(tv_null, &tz) < 0)
132 return -errno;
133 if (min)
134 *min = minutesdelta;
135 return 0;
136 }
137
138 int hwclock_reset_timezone(void) {
139 const struct timeval *tv_null = NULL;
140 struct timezone tz;
141
142 tz.tz_minuteswest = 0;
143 tz.tz_dsttime = 0; /* DST_NONE*/
144
145 /*
146 * The very first time we set the kernel's timezone, it will warp
147 * the clock. Do a dummy call here, so the time warping is sealed
148 * and we set only the timezone with next call.
149 */
150 if (settimeofday(tv_null, &tz) < 0)
151 return -errno;
152
153 return 0;
154 }