summaryrefslogtreecommitdiff
path: root/package/alix-switch
diff options
context:
space:
mode:
Diffstat (limited to 'package/alix-switch')
-rw-r--r--package/alix-switch/Makefile35
-rw-r--r--package/alix-switch/files/alix-switch19
-rw-r--r--package/alix-switch/files/alix-switch.init27
-rw-r--r--package/alix-switch/files/alix-switch.postinst3
-rw-r--r--package/alix-switch/src/alix-switchd.c108
5 files changed, 192 insertions, 0 deletions
diff --git a/package/alix-switch/Makefile b/package/alix-switch/Makefile
new file mode 100644
index 000000000..493a73053
--- /dev/null
+++ b/package/alix-switch/Makefile
@@ -0,0 +1,35 @@
+# This file is part of the OpenADK project. OpenADK is copyrighted
+# material, please see the LICENCE file in the top-level directory.
+
+include ${TOPDIR}/rules.mk
+
+PKG_NAME:= alix-switch
+PKG_VERSION:= 1.0
+PKG_RELEASE:= 1
+PKG_DESCR:= daemon listening on button events
+PKG_SECTION:= base
+
+PKG_TARGET_DEPENDS:= alix
+
+NO_DISTFILES:= 1
+
+include ${TOPDIR}/mk/package.mk
+
+$(eval $(call PKG_template,ALIX_SWITCH,${PKG_NAME},${PKG_VERSION}-${PKG_RELEASE},${PKG_DEPENDS},${PKG_DESCR},${PKG_SECTION}))
+
+PKGDFLT_ALIX_SWITCH= y
+
+CONFIG_STYLE:= manual
+BUILD_STYLE:= manual
+INSTALL_STYLE:= manual
+
+do-build:
+ ${TARGET_CC} -Wall ${TCPPFLAGS} ${TCFLAGS} \
+ -o ${WRKBUILD}/alix-switchd ${WRKBUILD}/alix-switchd.c
+
+do-install:
+ ${INSTALL_DIR} ${IDIR_ALIX_SWITCH}/usr/sbin ${IDIR_ALIX_SWITCH}/etc
+ ${INSTALL_BIN} ${WRKBUILD}/alix-switchd ${IDIR_ALIX_SWITCH}/usr/sbin
+ ${INSTALL_BIN} ./files/alix-switch ${IDIR_ALIX_SWITCH}/etc
+
+include ${TOPDIR}/mk/pkg-bottom.mk
diff --git a/package/alix-switch/files/alix-switch b/package/alix-switch/files/alix-switch
new file mode 100644
index 000000000..b61a6a48f
--- /dev/null
+++ b/package/alix-switch/files/alix-switch
@@ -0,0 +1,19 @@
+#!/bin/sh
+# launched by alix-switchd in case of button event
+# f.e. boot rescue system once
+case "$1" in
+ on)
+ echo "alix-switch: on"
+ mount /dev/sda1 /boot/grub
+ grub-reboot 1
+ umount /boot/grub
+ reboot
+ ;;
+ off)
+ echo "alix-switch: off"
+ ;;
+ *)
+ echo "Usage: $0 {on|off}"
+ ;;
+esac
+exit $?
diff --git a/package/alix-switch/files/alix-switch.init b/package/alix-switch/files/alix-switch.init
new file mode 100644
index 000000000..c783a1be5
--- /dev/null
+++ b/package/alix-switch/files/alix-switch.init
@@ -0,0 +1,27 @@
+#!/bin/sh
+#PKG alix-switch
+#INIT 10
+
+. /etc/rc.conf
+
+case $1 in
+autostop) ;;
+autostart)
+ test x"${alix_switch:-NO}" = x"NO" && exit 0
+ exec sh $0 start
+ ;;
+start)
+ alix-switchd -d
+ ;;
+stop)
+ pkill alix-switchd
+ ;;
+restart)
+ sh $0 stop
+ sh $0 start
+ ;;
+*)
+ echo "usage: $0 (start|stop|restart)"
+ exit 1
+esac
+exit $?
diff --git a/package/alix-switch/files/alix-switch.postinst b/package/alix-switch/files/alix-switch.postinst
new file mode 100644
index 000000000..28146bf44
--- /dev/null
+++ b/package/alix-switch/files/alix-switch.postinst
@@ -0,0 +1,3 @@
+#!/bin/sh
+. $IPKG_INSTROOT/etc/functions.sh
+add_rcconf alix_switch alix_switch YES
diff --git a/package/alix-switch/src/alix-switchd.c b/package/alix-switch/src/alix-switchd.c
new file mode 100644
index 000000000..78b52f19c
--- /dev/null
+++ b/package/alix-switch/src/alix-switchd.c
@@ -0,0 +1,108 @@
+/*
+* alix-switchd.c
+*
+* 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.
+*/
+
+#include <signal.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <syslog.h>
+#include <time.h>
+#include <unistd.h>
+#include <sys/io.h>
+#include <sys/stat.h>
+#include <sys/types.h>
+
+#define SCRIPT "/etc/alix-switch"
+#define GPIOBASE 0x6100
+
+typedef void (*sighandler_t)(int);
+
+static sighandler_t handle_signal (int sig_nr, sighandler_t signalhandler) {
+
+ struct sigaction neu_sig, alt_sig;
+
+ neu_sig.sa_handler = signalhandler;
+ sigemptyset(&neu_sig.sa_mask);
+ neu_sig.sa_flags = SA_RESTART;
+ if (sigaction (sig_nr, &neu_sig, &alt_sig) < 0)
+ return SIG_ERR;
+
+ return alt_sig.sa_handler;
+}
+
+static void start_daemon (void) {
+
+ int i;
+ pid_t pid, sid;
+
+ handle_signal(SIGHUP, SIG_IGN);
+ if ((pid = fork ()) != 0)
+ exit(EXIT_FAILURE);
+ umask(0);
+ if ((sid = setsid()) < 0)
+ exit(EXIT_FAILURE);
+ chdir("/");
+ for (i = sysconf(_SC_OPEN_MAX); i > 0; i--)
+ close(i);
+}
+
+
+int main(int argc, char *argv[]) {
+
+ int i;
+ unsigned long bPort = 0;
+ struct timespec sleep;
+ int bDaemon = 0, bSwitch = 0, bState = 0;
+
+ for(i = 1; i < argc; i++) {
+ if (!strcasecmp(argv[i], "-d") || !strcasecmp(argv[i], "--daemon")) {
+ bDaemon = 1;
+ } else {
+ printf( "\nusage: %s [-d | --daemon]\n", argv[0]);
+ exit(EXIT_FAILURE);
+ }
+ }
+
+ if (iopl(3)) {
+ fprintf( stderr, "Could not set I/O permissions to level 3\n");
+ exit(EXIT_FAILURE);
+ }
+
+ if (bDaemon)
+ start_daemon();
+
+ sleep.tv_sec = 0;
+ sleep.tv_nsec = 50000000;
+
+ while(1) {
+ bPort = inl(GPIOBASE + 0xB0);
+ if ((bPort & 0x100) == 0)
+ bState = 1;
+ else
+ bState = 0;
+
+ if (bState && !bSwitch)
+ system(SCRIPT " on");
+
+ bSwitch = bState;
+ nanosleep(&sleep, NULL);
+ }
+
+ if (iopl(0)) {
+ fprintf(stderr, "Could not set I/O permissions to level 0");
+ exit(EXIT_FAILURE);
+ }
+
+ return EXIT_SUCCESS;
+}