
Improve the example code to be at least useable, as in not causing harm (as shown below). Code can still be improved further, but this adds some basic safeguards. 1. hpfall *MUST* mlockall(MCL_CURRENT|MCL_FUTURE); itself! Since the Program sits and waits most of the time it becomes very likely swapped out. If it gets woken up when the laptop drops from the table while it is swapped out it actually triggers harddrive activity! 2. Daemonize hpfall using 'daemon(0,0)' (quick and dirty). 3. Give hpfall realtime priority. Should give a chance that it has less latency when woken up. Signed-off-by: Christian Thaeter <ct@pipapo.org> Signed-off-by: Frans Pop <elendil@planet.nl> Acked-by: Pavel Machek <pavel@ucw.cz> Signed-off-by: Jiri Kosina <jkosina@suse.cz>
108 lines
1.7 KiB
C
108 lines
1.7 KiB
C
/* Disk protection for HP machines.
|
|
*
|
|
* Copyright 2008 Eric Piel
|
|
* Copyright 2009 Pavel Machek <pavel@suse.cz>
|
|
*
|
|
* GPLv2.
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <unistd.h>
|
|
#include <fcntl.h>
|
|
#include <sys/stat.h>
|
|
#include <sys/types.h>
|
|
#include <string.h>
|
|
#include <stdint.h>
|
|
#include <errno.h>
|
|
#include <signal.h>
|
|
#include <sys/mman.h>
|
|
#include <sched.h>
|
|
|
|
void write_int(char *path, int i)
|
|
{
|
|
char buf[1024];
|
|
int fd = open(path, O_RDWR);
|
|
if (fd < 0) {
|
|
perror("open");
|
|
exit(1);
|
|
}
|
|
sprintf(buf, "%d", i);
|
|
if (write(fd, buf, strlen(buf)) != strlen(buf)) {
|
|
perror("write");
|
|
exit(1);
|
|
}
|
|
close(fd);
|
|
}
|
|
|
|
void set_led(int on)
|
|
{
|
|
write_int("/sys/class/leds/hp::hddprotect/brightness", on);
|
|
}
|
|
|
|
void protect(int seconds)
|
|
{
|
|
write_int("/sys/block/sda/device/unload_heads", seconds*1000);
|
|
}
|
|
|
|
int on_ac(void)
|
|
{
|
|
// /sys/class/power_supply/AC0/online
|
|
}
|
|
|
|
int lid_open(void)
|
|
{
|
|
// /proc/acpi/button/lid/LID/state
|
|
}
|
|
|
|
void ignore_me(void)
|
|
{
|
|
protect(0);
|
|
set_led(0);
|
|
}
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
int fd, ret;
|
|
struct sched_param param;
|
|
|
|
fd = open("/dev/freefall", O_RDONLY);
|
|
if (fd < 0) {
|
|
perror("open");
|
|
return EXIT_FAILURE;
|
|
}
|
|
|
|
daemon(0, 0);
|
|
param.sched_priority = sched_get_priority_max(SCHED_FIFO);
|
|
sched_setscheduler(0, SCHED_FIFO, ¶m);
|
|
mlockall(MCL_CURRENT|MCL_FUTURE);
|
|
|
|
signal(SIGALRM, ignore_me);
|
|
|
|
for (;;) {
|
|
unsigned char count;
|
|
|
|
ret = read(fd, &count, sizeof(count));
|
|
alarm(0);
|
|
if ((ret == -1) && (errno == EINTR)) {
|
|
/* Alarm expired, time to unpark the heads */
|
|
continue;
|
|
}
|
|
|
|
if (ret != sizeof(count)) {
|
|
perror("read");
|
|
break;
|
|
}
|
|
|
|
protect(21);
|
|
set_led(1);
|
|
if (1 || on_ac() || lid_open())
|
|
alarm(2);
|
|
else
|
|
alarm(20);
|
|
}
|
|
|
|
close(fd);
|
|
return EXIT_SUCCESS;
|
|
}
|