summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorRoy Marples <roy@marples.name>2009-10-13 08:03:45 +0100
committerRoy Marples <roy@marples.name>2009-10-13 08:03:45 +0100
commita8f6a9b6540ca301af625b4767df53860511a6dc (patch)
treea1dd01c8796d117c986f96fd702c3f3c6f2cdba5
parentaaa0498bf86baa065656b6a5c37cff82c032eb6d (diff)
downloadopenrc-a8f6a9b6540ca301af625b4767df53860511a6dc.tar.gz
openrc-a8f6a9b6540ca301af625b4767df53860511a6dc.tar.bz2
openrc-a8f6a9b6540ca301af625b4767df53860511a6dc.tar.xz
Add new utility and init script swclock that sets the system time based on
the mtime of a file. It saves the shutdown time to this file also. This is handy for systems without a working RTC chip. Based on an idea by Michael A. Smith <michael@smith-li.com>. Fixes Gentoo #272073.
-rw-r--r--init.d.misc/.gitignore11
-rw-r--r--init.d/.gitignore1
-rw-r--r--init.d/Makefile2
-rw-r--r--init.d/savecache.in2
-rw-r--r--init.d/swclock.in27
-rw-r--r--src/includes/rc-misc.h1
-rw-r--r--src/rc/.gitignore2
-rw-r--r--src/rc/Makefile4
-rw-r--r--src/rc/builtins.h1
-rw-r--r--src/rc/rc-applets.c2
-rw-r--r--src/rc/swclock.c110
11 files changed, 159 insertions, 4 deletions
diff --git a/init.d.misc/.gitignore b/init.d.misc/.gitignore
new file mode 100644
index 0000000..9f1ce28
--- /dev/null
+++ b/init.d.misc/.gitignore
@@ -0,0 +1,11 @@
+avahi-dnsconfd
+avahid
+dhcpcd
+dbus
+hald
+named
+ntpd
+openvpn
+polkitd
+sshd
+wpa_supplicant
diff --git a/init.d/.gitignore b/init.d/.gitignore
index 5e1e1ec..94a40e2 100644
--- a/init.d/.gitignore
+++ b/init.d/.gitignore
@@ -34,6 +34,7 @@ rc-enabled
rpcbind
savecore
swap-blk
+swclock
syslogd
termencoding
ttys
diff --git a/init.d/Makefile b/init.d/Makefile
index b816ae7..0786e05 100644
--- a/init.d/Makefile
+++ b/init.d/Makefile
@@ -1,6 +1,6 @@
DIR= ${INITDIR}
SRCS= bootmisc.in fsck.in hostname.in local.in localmount.in netmount.in \
- network.in root.in savecache.in swap.in sysctl.in urandom.in
+ network.in root.in savecache.in swap.in swclock.in sysctl.in urandom.in
BIN= ${OBJS}
# Build our old net foo or not
diff --git a/init.d/savecache.in b/init.d/savecache.in
index 799b8bc..2c981c6 100644
--- a/init.d/savecache.in
+++ b/init.d/savecache.in
@@ -22,7 +22,7 @@ start()
fi
fi
local save=
- for x in deptree depconfig softlevel nettree rc.log; do
+ for x in deptree depconfig shutdowntime softlevel nettree rc.log; do
[ -e "$RC_SVCDIR/$x" ] && save="$save $RC_SVCDIR/$x"
done
if [ -n "$save" ]; then
diff --git a/init.d/swclock.in b/init.d/swclock.in
new file mode 100644
index 0000000..ab076c3
--- /dev/null
+++ b/init.d/swclock.in
@@ -0,0 +1,27 @@
+#!@PREFIX@/sbin/runscript
+# Copyright (c) 2009 Roy Marples <roy@marples.name>
+# All rights reserved. Released under the 2-clause BSD license.
+
+description="Sets the local clock to the mtime of a given file."
+
+depend()
+{
+ before *
+ keyword -openvz -prefix -uml -vserver -xenu
+}
+
+# swclock is an OpenRC built in
+
+start()
+{
+ ebegin "Setting the local clock based on last shutdown time"
+ swclock
+ eend $?
+}
+
+stop()
+{
+ ebegin "Saving the shutdown time"
+ swclock --save
+ eend $?
+}
diff --git a/src/includes/rc-misc.h b/src/includes/rc-misc.h
index de6888e..798df55 100644
--- a/src/includes/rc-misc.h
+++ b/src/includes/rc-misc.h
@@ -35,6 +35,7 @@
#include <sys/stat.h>
#include <errno.h>
#include <stdbool.h>
+#include <stdlib.h>
#include <string.h>
#define RC_LEVEL_BOOT "boot"
diff --git a/src/rc/.gitignore b/src/rc/.gitignore
index d06e857..e14788f 100644
--- a/src/rc/.gitignore
+++ b/src/rc/.gitignore
@@ -38,6 +38,7 @@ service_crashed
checkpath
fstabinfo
mountinfo
+swclock
rc-depend
service_get_value
service_set_value
@@ -59,6 +60,7 @@ checkpath.o
fstabinfo.o
mountinfo.o
start-stop-daemon.o
+swclock.o
rc-applets.o
rc-depend.o
rc-logger.o
diff --git a/src/rc/Makefile b/src/rc/Makefile
index 78b862d..d9f8e11 100644
--- a/src/rc/Makefile
+++ b/src/rc/Makefile
@@ -2,7 +2,7 @@ PROG= rc
SRCS= checkpath.c fstabinfo.c mountinfo.c start-stop-daemon.c \
rc-applets.c rc-depend.c rc-logger.c \
rc-misc.c rc-plugin.c rc-service.c rc-status.c rc-update.c \
- runscript.c rc.c
+ runscript.c rc.c swclock.c
CLEANFILES= version.h
@@ -26,7 +26,7 @@ RC_SBINLINKS= mark_service_starting mark_service_started \
mark_service_stopping mark_service_stopped \
mark_service_inactive mark_service_wasinactive \
mark_service_hotplugged mark_service_failed \
- rc-abort
+ rc-abort swclock
ALL_LINKS= ${BINLINKS} ${SBINLINKS} ${RC_BINLINKS} ${RC_SBINLINKS}
CLEANFILES+= ${ALL_LINKS}
diff --git a/src/rc/builtins.h b/src/rc/builtins.h
index 6ce2164..1acaccb 100644
--- a/src/rc/builtins.h
+++ b/src/rc/builtins.h
@@ -35,6 +35,7 @@ int rc_status(int, char **);
int rc_update(int, char **);
int runscript(int, char **);
int start_stop_daemon(int, char **);
+int swclock(int, char **);
void run_applets(int, char **);
diff --git a/src/rc/rc-applets.c b/src/rc/rc-applets.c
index 101097c..04e88de 100644
--- a/src/rc/rc-applets.c
+++ b/src/rc/rc-applets.c
@@ -449,6 +449,8 @@ run_applets(int argc, char **argv)
exit(start_stop_daemon(argc, argv));
else if (strcmp (applet, "checkpath") == 0)
exit(checkpath(argc, argv));
+ else if (strcmp(applet, "swclock") == 0)
+ exit(swclock(argc, argv));
/* These could also be applications in their own right */
if (strcmp(applet, "shell_var") == 0)
diff --git a/src/rc/swclock.c b/src/rc/swclock.c
new file mode 100644
index 0000000..39d6563
--- /dev/null
+++ b/src/rc/swclock.c
@@ -0,0 +1,110 @@
+/*
+ swclock.c
+ Sets the system time from the mtime of the given file.
+ This is useful for systems who do not have a working RTC and rely on ntp.
+ OpenRC relies on the correct system time for a lot of operations so this is needed
+ quite early.
+*/
+
+/*
+ * Copyright (c) 2009 Roy Marples <roy@marples.name>
+ * All rights reserved
+
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
+ * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+ * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+ * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+ * SUCH DAMAGE.
+ */
+
+#include <sys/time.h>
+#include <sys/types.h>
+#include <sys/stat.h>
+
+#include <errno.h>
+#include <fcntl.h>
+#include <getopt.h>
+#include <unistd.h>
+#include <utime.h>
+
+#include "builtins.h"
+#include "einfo.h"
+#include "rc-misc.h"
+
+#define RC_SHUTDOWNTIME RC_SVCDIR "/shutdowntime"
+
+extern const char *applet;
+
+#include "_usage.h"
+#define extraopts "file"
+#define getoptstring "s" getoptstring_COMMON
+static const struct option longopts[] = {
+ { "save", 0, NULL, 's' },
+ longopts_COMMON
+};
+static const char * const longopts_help[] = {
+ "saves the time",
+ longopts_help_COMMON
+};
+#include "_usage.c"
+
+int
+swclock(int argc, char **argv)
+{
+ int opt, sflag = 0;
+ const char *file = RC_SHUTDOWNTIME;
+ struct stat sb;
+ struct timeval tv;
+
+ while ((opt = getopt_long(argc, argv, getoptstring,
+ longopts, (int *) 0)) != -1)
+ {
+ switch (opt) {
+ case 's':
+ sflag = 1;
+ break;
+ case_RC_COMMON_GETOPT;
+ }
+ }
+
+ if (optind < argc)
+ file = argv[optind++];
+
+ if (sflag) {
+ if (stat(file, &sb) == -1) {
+ opt = open(file, O_WRONLY | O_CREAT, 0644);
+ if (opt == -1)
+ eerrorx("swclock: open: %s", strerror(errno));
+ close(opt);
+ } else
+ if (utime(file, NULL) == -1)
+ eerrorx("swclock: utime: %s", strerror(errno));
+ return 0;
+ }
+
+ if (stat(file, &sb) == -1)
+ eerrorx("swclock: `%s': %s", file, strerror(errno));
+
+ tv.tv_sec = sb.st_mtime;
+ tv.tv_usec = 0;
+
+ if (settimeofday(&tv, NULL) == -1)
+ eerrorx("swclock: settimeofday: %s", strerror(errno));
+
+ return 0;
+}