diff --git a/actledPi.c b/actledPi.c
new file mode 100644
index 0000000..904f2d1
--- /dev/null
+++ b/actledPi.c
@@ -0,0 +1,259 @@
+/*
+ * This is free and unencumbered software released into the public domain.
+ *
+ * Anyone is free to copy, modify, publish, use, compile, sell, or
+ * distribute this software, either in source code form or as a compiled
+ * binary, for any purpose, commercial or non-commercial, and by any
+ * means.
+ *
+ * In jurisdictions that recognize copyright laws, the author or authors
+ * of this software dedicate any and all copyright interest in the
+ * software to the public domain. We make this dedication for the benefit
+ * of the public at large and to the detriment of our heirs and
+ * successors. We intend this dedication to be an overt act of
+ * relinquishment in perpetuity of all present and future rights to this
+ * software under copyright law.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
+ * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
+ * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
+ * IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR
+ * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
+ * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
+ * OTHER DEALINGS IN THE SOFTWARE.
+ *
+ * For more information, please refer to
+ *
+ **************************************************************************
+ *
+ * A disk activity light for the Raspberry Pi.
+ * Blinks the ACT led on all mass storage I/O, i.e. not only the SD card.
+ * Based on hddled.c - http://members.optusnet.com.au/foonly/whirlpool/code/hddled.c -
+ *
+ *
+ * To compile:
+ * gcc -Wall -O3 -o actledPi actledPi.c
+ *
+ * Options:
+ * -d, --detach Detach from terminal (become a daemon)
+ * -r, --refresh=VALUE Refresh interval (default: 20 ms)
+ *
+ */
+
+
+#define VMSTAT "/proc/vmstat"
+#define ACTLED "/sys/class/leds/led0/brightness"
+#define TRGCTL "/sys/class/leds/led0/trigger"
+#define LOW 0
+#define HIGH 1
+
+#define _GNU_SOURCE
+
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+
+static unsigned int o_refresh = 20; /* milliseconds */
+static int o_detach = 0;
+
+static volatile sig_atomic_t running = 1;
+static char *line = NULL;
+static size_t len = 0;
+
+FILE *actled = NULL;
+FILE *trigger_ctl = NULL;
+
+/* Reread the vmstat file */
+int activity(FILE *vmstat) {
+ static unsigned int prev_pgpgin, prev_pgpgout;
+ unsigned int pgpgin, pgpgout;
+ int found_pgpgin, found_pgpgout;
+ int result;
+
+ /* Reload the vmstat file */
+ result = TEMP_FAILURE_RETRY(fseek(vmstat, 0L, SEEK_SET));
+ if (result) {
+ perror("Could not rewind " VMSTAT);
+ return result;
+ }
+
+ /* Clear glibc's buffer */
+ result = TEMP_FAILURE_RETRY(fflush(vmstat));
+ if (result) {
+ perror("Could not flush input stream");
+ return result;
+ }
+
+ /* Extract the I/O stats */
+ found_pgpgin = found_pgpgout = 0;
+ while (getline(&line, &len, vmstat) != -1 && errno != EINTR) {
+ if (sscanf(line, "pgpgin %u", &pgpgin))
+ found_pgpgin++;
+ else if (sscanf(line, "pgpgout %u", &pgpgout))
+ found_pgpgout++;
+ if (found_pgpgin && found_pgpgout)
+ break;
+ }
+ if (!found_pgpgin || !found_pgpgout) {
+ fprintf(stderr, "Could not find required lines in " VMSTAT);
+ return -1;
+ }
+
+ /* Anything changed? */
+ result =
+ (prev_pgpgin != pgpgin) ||
+ (prev_pgpgout != pgpgout);
+ prev_pgpgin = pgpgin;
+ prev_pgpgout = pgpgout;
+
+ return result;
+}
+
+/* Update the LED */
+void led(int on) {
+ static int current = 1; /* Ensure the LED turns off on first call */
+ if (current == on)
+ return;
+
+ if (on) {
+ fputs("255\n", actled);
+ } else {
+ fputs("0\n", actled);
+ }
+ fflush(actled);
+ current = on;
+}
+
+/* Signal handler -- break out of the main loop */
+void shutdown(int sig) {
+ running = 0;
+}
+
+/* Argp parser function */
+error_t parse_options(int key, char *arg, struct argp_state *state) {
+ switch (key) {
+ case 'd':
+ o_detach = 1;
+ break;
+ case 'r':
+ o_refresh = strtol(arg, NULL, 10);
+ if (o_refresh < 10)
+ argp_failure(state, EXIT_FAILURE, 0,
+ "refresh interval must be at least 10");
+ break;
+ }
+ return 0;
+}
+
+int main(int argc, char **argv) {
+ struct argp_option options[] = {
+ { "detach", 'd', NULL, 0, "Detach from terminal" },
+ { "refresh", 'r', "VALUE", 0, "Refresh interval (default: 20 ms)" },
+ { 0 },
+ };
+ struct argp parser = {
+ NULL, parse_options, NULL,
+ "Show disk activity on all disks.",
+ NULL, NULL, NULL
+ };
+ int status = EXIT_FAILURE;
+ FILE *vmstat = NULL;
+ struct timespec delay;
+
+ /* Parse the command-line */
+ parser.options = options;
+ if (argp_parse(&parser, argc, argv, ARGP_NO_ARGS, NULL, NULL))
+ goto out;
+
+ delay.tv_sec = o_refresh / 1000;
+ delay.tv_nsec = 1000000 * (o_refresh % 1000);
+
+
+ /* Open the vmstat file */
+ vmstat = fopen(VMSTAT, "r");
+ if (!vmstat) {
+ perror("Could not open " VMSTAT " for reading");
+ goto out;
+ }
+
+ /* Change the trigger on the OK/Act LED to "none" */
+
+ trigger_ctl = fopen(TRGCTL, "rw");
+ if (!trigger_ctl) {
+ perror("Unable to change LED trigger");
+ goto out;
+ }
+ fputs ("none\n", trigger_ctl);
+ fclose (trigger_ctl);
+
+
+ /* Open the actled file */
+ actled = fopen(ACTLED, "w");
+ if (!actled) {
+ perror("Could not open " ACTLED " for writing");
+ goto out;
+ }
+
+ /* Ensure the LED is off */
+ led(LOW);
+
+ if (activity(vmstat) < 0)
+ goto out;
+
+ /* Detach from terminal? */
+ if (o_detach) {
+ pid_t child = fork();
+ if (child < 0) {
+ perror("Could not detach from terminal");
+ goto out;
+ }
+ if (child) {
+ /* I am the parent */
+ status = EXIT_SUCCESS;
+ goto out;
+ }
+ }
+
+ /* We catch these signals so we can clean up */
+ {
+ struct sigaction action;
+ memset(&action, 0, sizeof(action));
+ action.sa_handler = shutdown;
+ sigemptyset(&action.sa_mask);
+ action.sa_flags = 0; /* We block on usleep; don't use SA_RESTART */
+ sigaction(SIGHUP, &action, NULL);
+ sigaction(SIGINT, &action, NULL);
+ sigaction(SIGTERM, &action, NULL);
+ }
+
+ /* Loop until signal received */
+ while (running) {
+ int a;
+ if (nanosleep(&delay, NULL) < 0)
+ break;
+ a = activity(vmstat);
+ if (a < 0)
+ break;
+ led(a);
+ }
+
+ /* Ensure the LED is off */
+ led(LOW);
+
+ status = EXIT_SUCCESS;
+
+out:
+ if (line) free(line);
+ if (vmstat) fclose(vmstat);
+ if (actled) {
+ fclose(actled);
+ trigger_ctl = fopen(TRGCTL, "rw");
+ fputs("mmc0\n", trigger_ctl);
+ fclose(trigger_ctl);
+ }
+ return status;
+}