2010-01-26 21:06:50 +03:00
/*-*- Mode: C; c-basic-offset: 8 -*-*/
2010-02-03 15:03:47 +03:00
/***
This file is part of systemd .
Copyright 2010 Lennart Poettering
systemd 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 .
systemd 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 .
You should have received a copy of the GNU General Public License
along with systemd ; If not , see < http : //www.gnu.org/licenses/>.
* * */
2010-01-28 04:44:47 +03:00
# include <errno.h>
2010-04-06 04:43:58 +04:00
# include <signal.h>
2010-01-28 04:44:47 +03:00
2010-01-26 23:39:06 +03:00
# include "unit.h"
2010-01-26 21:06:50 +03:00
# include "target.h"
# include "load-fragment.h"
2010-01-27 08:33:27 +03:00
# include "log.h"
2010-04-18 05:08:16 +04:00
# include "dbus-target.h"
2010-01-26 21:06:50 +03:00
2010-01-27 08:33:27 +03:00
static const UnitActiveState state_translation_table [ _TARGET_STATE_MAX ] = {
[ TARGET_DEAD ] = UNIT_INACTIVE ,
[ TARGET_ACTIVE ] = UNIT_ACTIVE
} ;
static void target_set_state ( Target * t , TargetState state ) {
TargetState old_state ;
assert ( t ) ;
old_state = t - > state ;
t - > state = state ;
2010-04-10 19:53:17 +04:00
if ( state ! = old_state )
2010-04-23 22:25:55 +04:00
log_debug ( " %s changed %s -> %s " ,
2010-04-21 05:27:44 +04:00
UNIT ( t ) - > meta . id ,
target_state_to_string ( old_state ) ,
target_state_to_string ( state ) ) ;
2010-01-26 21:06:50 +03:00
2010-01-27 08:33:27 +03:00
unit_notify ( UNIT ( t ) , state_translation_table [ old_state ] , state_translation_table [ state ] ) ;
}
2010-04-21 05:27:44 +04:00
static int target_coldplug ( Unit * u ) {
Target * t = TARGET ( u ) ;
assert ( t ) ;
assert ( t - > state = = TARGET_DEAD ) ;
if ( t - > deserialized_state ! = t - > state )
target_set_state ( t , t - > deserialized_state ) ;
return 0 ;
}
static void target_dump ( Unit * u , FILE * f , const char * prefix ) {
Target * t = TARGET ( u ) ;
assert ( t ) ;
assert ( f ) ;
fprintf ( f ,
" %sTarget State: %s \n " ,
prefix , target_state_to_string ( t - > state ) ) ;
}
2010-01-27 08:33:27 +03:00
static int target_start ( Unit * u ) {
Target * t = TARGET ( u ) ;
assert ( t ) ;
assert ( t - > state = = TARGET_DEAD ) ;
target_set_state ( t , TARGET_ACTIVE ) ;
return 0 ;
}
2010-01-26 21:06:50 +03:00
2010-01-27 08:33:27 +03:00
static int target_stop ( Unit * u ) {
Target * t = TARGET ( u ) ;
assert ( t ) ;
assert ( t - > state = = TARGET_ACTIVE ) ;
target_set_state ( t , TARGET_DEAD ) ;
return 0 ;
2010-01-26 21:06:50 +03:00
}
2010-04-21 05:27:44 +04:00
static int target_serialize ( Unit * u , FILE * f , FDSet * fds ) {
Target * s = TARGET ( u ) ;
assert ( s ) ;
assert ( f ) ;
assert ( fds ) ;
unit_serialize_item ( u , f , " state " , target_state_to_string ( s - > state ) ) ;
return 0 ;
}
static int target_deserialize_item ( Unit * u , const char * key , const char * value , FDSet * fds ) {
Target * s = TARGET ( u ) ;
assert ( u ) ;
assert ( key ) ;
assert ( value ) ;
assert ( fds ) ;
if ( streq ( key , " state " ) ) {
TargetState state ;
if ( ( state = target_state_from_string ( value ) ) < 0 )
log_debug ( " Failed to parse state value %s " , value ) ;
else
s - > deserialized_state = state ;
} else
log_debug ( " Unknown serialization key '%s' " , key ) ;
return 0 ;
}
2010-01-26 23:39:06 +03:00
static UnitActiveState target_active_state ( Unit * u ) {
2010-01-27 08:33:27 +03:00
assert ( u ) ;
return state_translation_table [ TARGET ( u ) - > state ] ;
2010-01-26 21:06:50 +03:00
}
2010-04-13 22:59:01 +04:00
static const char * target_sub_state_to_string ( Unit * u ) {
assert ( u ) ;
2010-04-21 05:27:44 +04:00
return target_state_to_string ( TARGET ( u ) - > state ) ;
2010-04-13 22:59:01 +04:00
}
2010-04-10 06:52:21 +04:00
int target_get_runlevel ( Target * t ) {
static const struct {
const char * special ;
const int runlevel ;
} table [ ] = {
{ SPECIAL_RUNLEVEL5_TARGET , ' 5 ' } ,
{ SPECIAL_RUNLEVEL4_TARGET , ' 4 ' } ,
{ SPECIAL_RUNLEVEL3_TARGET , ' 3 ' } ,
{ SPECIAL_RUNLEVEL2_TARGET , ' 2 ' } ,
{ SPECIAL_RUNLEVEL1_TARGET , ' 1 ' } ,
{ SPECIAL_RUNLEVEL0_TARGET , ' 0 ' } ,
{ SPECIAL_RUNLEVEL6_TARGET , ' 6 ' } ,
} ;
unsigned i ;
assert ( t ) ;
/* Tries to determine if this is a SysV runlevel and returns
* it if that is so . */
for ( i = 0 ; i < ELEMENTSOF ( table ) ; i + + )
if ( unit_has_name ( UNIT ( t ) , table [ i ] . special ) )
return table [ i ] . runlevel ;
return 0 ;
}
2010-04-21 05:27:44 +04:00
static const char * const target_state_table [ _TARGET_STATE_MAX ] = {
[ TARGET_DEAD ] = " dead " ,
[ TARGET_ACTIVE ] = " active "
} ;
DEFINE_STRING_TABLE_LOOKUP ( target_state , TargetState ) ;
2010-01-26 23:39:06 +03:00
const UnitVTable target_vtable = {
2010-01-26 21:06:50 +03:00
. suffix = " .target " ,
2010-04-10 19:53:17 +04:00
. load = unit_load_fragment_and_dropin ,
2010-04-21 05:27:44 +04:00
. coldplug = target_coldplug ,
2010-01-27 08:33:27 +03:00
. dump = target_dump ,
. start = target_start ,
. stop = target_stop ,
2010-01-26 21:06:50 +03:00
2010-04-21 05:27:44 +04:00
. serialize = target_serialize ,
. deserialize_item = target_deserialize_item ,
2010-04-13 22:59:01 +04:00
. active_state = target_active_state ,
2010-04-18 05:08:16 +04:00
. sub_state_to_string = target_sub_state_to_string ,
. bus_message_handler = bus_target_message_handler
2010-01-26 21:06:50 +03:00
} ;