2014-07-12 09:56:22 -07:00
/* Disk protection for HP/DELL machines.
2009-02-18 14:48:23 -08:00
*
* Copyright 2008 Eric Piel
2010-07-18 14:27:13 +02:00
* Copyright 2009 Pavel Machek < pavel @ ucw . cz >
2014-07-12 09:56:22 -07:00
* Copyright 2012 Sonal Santan
* Copyright 2014 Pali Rohár < pali . rohar @ gmail . com >
2009-02-18 14:48:23 -08:00
*
* 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>
2009-07-25 20:55:15 +02:00
# include <sys/mman.h>
# include <sched.h>
2014-07-12 09:56:22 -07:00
# include <syslog.h>
2009-02-18 14:48:23 -08:00
2014-07-12 09:56:22 -07:00
static int noled ;
static char unload_heads_path [ 64 ] ;
static char device_path [ 32 ] ;
static const char app_name [ ] = " FREE FALL " ;
2009-07-25 21:00:12 +02:00
2014-07-12 09:56:22 -07:00
static int set_unload_heads_path ( char * device )
2009-07-25 21:00:12 +02:00
{
char devname [ 64 ] ;
if ( strlen ( device ) < = 5 | | strncmp ( device , " /dev/ " , 5 ) ! = 0 )
return - EINVAL ;
2014-07-12 09:56:22 -07:00
strncpy ( devname , device + 5 , sizeof ( devname ) - 1 ) ;
2014-06-04 23:28:10 +02:00
devname [ sizeof ( devname ) - 1 ] = ' \0 ' ;
2014-07-12 09:56:22 -07:00
strncpy ( device_path , device , sizeof ( device_path ) - 1 ) ;
2009-07-25 21:00:12 +02:00
2013-12-30 12:35:43 +01:00
snprintf ( unload_heads_path , sizeof ( unload_heads_path ) - 1 ,
2009-07-25 21:00:12 +02:00
" /sys/block/%s/device/unload_heads " , devname ) ;
return 0 ;
}
2014-07-12 09:56:22 -07:00
static int valid_disk ( void )
2009-07-25 21:00:12 +02:00
{
int fd = open ( unload_heads_path , O_RDONLY ) ;
2014-07-12 09:56:22 -07:00
2009-07-25 21:00:12 +02:00
if ( fd < 0 ) {
perror ( unload_heads_path ) ;
return 0 ;
}
close ( fd ) ;
return 1 ;
}
2014-07-12 09:56:22 -07:00
static void write_int ( char * path , int i )
2009-02-18 14:48:23 -08:00
{
char buf [ 1024 ] ;
int fd = open ( path , O_RDWR ) ;
2014-07-12 09:56:22 -07:00
2009-02-18 14:48:23 -08:00
if ( fd < 0 ) {
perror ( " open " ) ;
exit ( 1 ) ;
}
2014-07-12 09:56:22 -07:00
2009-02-18 14:48:23 -08:00
sprintf ( buf , " %d " , i ) ;
2014-07-12 09:56:22 -07:00
2009-02-18 14:48:23 -08:00
if ( write ( fd , buf , strlen ( buf ) ) ! = strlen ( buf ) ) {
perror ( " write " ) ;
exit ( 1 ) ;
}
2014-07-12 09:56:22 -07:00
2009-02-18 14:48:23 -08:00
close ( fd ) ;
}
2014-07-12 09:56:22 -07:00
static void set_led ( int on )
2009-02-18 14:48:23 -08:00
{
2014-07-12 09:56:22 -07:00
if ( noled )
return ;
2009-02-18 14:48:23 -08:00
write_int ( " /sys/class/leds/hp::hddprotect/brightness " , on ) ;
}
2014-07-12 09:56:22 -07:00
static void protect ( int seconds )
2009-02-18 14:48:23 -08:00
{
2014-07-12 09:56:22 -07:00
const char * str = ( seconds = = 0 ) ? " Unparked " : " Parked " ;
2009-07-25 21:00:12 +02:00
write_int ( unload_heads_path , seconds * 1000 ) ;
2014-07-12 09:56:22 -07:00
syslog ( LOG_INFO , " %s %s disk head \n " , str , device_path ) ;
2009-02-18 14:48:23 -08:00
}
2014-07-12 09:56:22 -07:00
static int on_ac ( void )
2009-02-18 14:48:23 -08:00
{
2014-07-12 09:56:22 -07:00
/* /sys/class/power_supply/AC0/online */
return 1 ;
2009-02-18 14:48:23 -08:00
}
2014-07-12 09:56:22 -07:00
static int lid_open ( void )
2009-02-18 14:48:23 -08:00
{
2014-07-12 09:56:22 -07:00
/* /proc/acpi/button/lid/LID/state */
return 1 ;
2009-02-18 14:48:23 -08:00
}
2014-07-12 09:56:22 -07:00
static void ignore_me ( int signum )
2009-02-18 14:48:23 -08:00
{
protect ( 0 ) ;
set_led ( 0 ) ;
}
2009-07-25 21:00:12 +02:00
int main ( int argc , char * * argv )
2009-02-18 14:48:23 -08:00
{
2009-06-22 11:08:36 +02:00
int fd , ret ;
2014-07-12 09:56:22 -07:00
struct stat st ;
2009-07-25 20:55:15 +02:00
struct sched_param param ;
2009-02-18 14:48:23 -08:00
2009-07-25 21:00:12 +02:00
if ( argc = = 1 )
ret = set_unload_heads_path ( " /dev/sda " ) ;
else if ( argc = = 2 )
ret = set_unload_heads_path ( argv [ 1 ] ) ;
else
ret = - EINVAL ;
if ( ret | | ! valid_disk ( ) ) {
fprintf ( stderr , " usage: %s <device> (default: /dev/sda) \n " ,
argv [ 0 ] ) ;
exit ( 1 ) ;
}
2009-06-22 11:08:36 +02:00
fd = open ( " /dev/freefall " , O_RDONLY ) ;
if ( fd < 0 ) {
2009-07-25 21:00:12 +02:00
perror ( " /dev/freefall " ) ;
2009-06-22 11:08:36 +02:00
return EXIT_FAILURE ;
}
2009-02-18 14:48:23 -08:00
2014-07-12 09:56:22 -07:00
if ( stat ( " /sys/class/leds/hp::hddprotect/brightness " , & st ) )
noled = 1 ;
if ( daemon ( 0 , 0 ) ! = 0 ) {
perror ( " daemon " ) ;
return EXIT_FAILURE ;
}
openlog ( app_name , LOG_CONS | LOG_PID | LOG_NDELAY , LOG_LOCAL1 ) ;
2009-07-25 20:55:15 +02:00
param . sched_priority = sched_get_priority_max ( SCHED_FIFO ) ;
sched_setscheduler ( 0 , SCHED_FIFO , & param ) ;
mlockall ( MCL_CURRENT | MCL_FUTURE ) ;
2009-02-18 14:48:23 -08:00
signal ( SIGALRM , ignore_me ) ;
2009-06-22 11:08:36 +02:00
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 ) ;
}
2009-02-18 14:48:23 -08:00
2014-07-12 09:56:22 -07:00
closelog ( ) ;
2009-06-22 11:08:36 +02:00
close ( fd ) ;
return EXIT_SUCCESS ;
2009-02-18 14:48:23 -08:00
}