2010-08-14 21:59:25 +04:00
/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
2010-07-08 08:08:32 +04:00
/***
This file is part of systemd .
Copyright 2010 Lennart Poettering
systemd is free software ; you can redistribute it and / or modify it
2012-04-12 02:20:58 +04:00
under the terms of the GNU Lesser General Public License as published by
the Free Software Foundation ; either version 2.1 of the License , or
2010-07-08 08:08:32 +04:00
( 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
2012-04-12 02:20:58 +04:00
Lesser General Public License for more details .
2010-07-08 08:08:32 +04:00
2012-04-12 02:20:58 +04:00
You should have received a copy of the GNU Lesser General Public License
2010-07-08 08:08:32 +04:00
along with systemd ; If not , see < http : //www.gnu.org/licenses/>.
* * */
# include <stdio.h>
# include <string.h>
2010-07-08 23:01:42 +04:00
# include <dirent.h>
# include <errno.h>
2010-07-08 08:08:32 +04:00
# include "util.h"
# include "macro.h"
2010-07-11 02:50:49 +04:00
# include "cgroup-util.h"
2010-07-08 08:08:32 +04:00
# include "cgroup-show.h"
static int compare ( const void * a , const void * b ) {
const pid_t * p = a , * q = b ;
if ( * p < * q )
return - 1 ;
if ( * p > * q )
return 1 ;
return 0 ;
}
2010-07-08 23:01:42 +04:00
static unsigned ilog10 ( unsigned long ul ) {
int n = 0 ;
while ( ul > 0 ) {
n + + ;
ul / = 10 ;
}
return n ;
}
2012-04-16 20:56:18 +04:00
static void show_pid_array ( int pids [ ] , unsigned n_pids , const char * prefix , unsigned n_columns , bool extra , bool more , bool kernel_threads ) {
unsigned i , m ;
pid_t biggest = 0 ;
/* Filter duplicates */
m = 0 ;
for ( i = 0 ; i < n_pids ; i + + ) {
unsigned j ;
if ( pids [ i ] > biggest )
biggest = pids [ i ] ;
for ( j = i + 1 ; j < n_pids ; j + + )
if ( pids [ i ] = = pids [ j ] )
break ;
if ( j > = n_pids )
pids [ m + + ] = pids [ i ] ;
}
n_pids = m ;
/* And sort */
qsort ( pids , n_pids , sizeof ( pid_t ) , compare ) ;
if ( n_columns > 8 )
n_columns - = 8 ;
else
n_columns = 20 ;
for ( i = 0 ; i < n_pids ; i + + ) {
char * t = NULL ;
get_process_cmdline ( pids [ i ] , n_columns , true , & t ) ;
printf ( " %s%s %*lu %s \n " ,
prefix ,
extra ? " \342 \200 \243 " : ( ( more | | i < n_pids - 1 ) ? " \342 \224 \234 " : " \342 \224 \224 " ) ,
( int ) ilog10 ( biggest ) ,
( unsigned long ) pids [ i ] ,
strna ( t ) ) ;
free ( t ) ;
}
}
2012-01-22 21:18:51 +04:00
static int show_cgroup_one_by_path ( const char * path , const char * prefix , unsigned n_columns , bool more , bool kernel_threads ) {
2010-07-08 08:08:32 +04:00
char * fn ;
FILE * f ;
size_t n = 0 , n_allocated = 0 ;
pid_t * pids = NULL ;
2010-07-08 23:01:42 +04:00
char * p ;
2012-04-16 20:56:18 +04:00
pid_t pid ;
2010-07-08 23:01:42 +04:00
int r ;
2010-07-08 08:08:32 +04:00
2012-04-16 20:56:18 +04:00
r = cg_fix_path ( path , & p ) ;
if ( r < 0 )
2010-07-12 20:16:44 +04:00
return r ;
2010-07-08 08:08:32 +04:00
2010-07-08 23:01:42 +04:00
r = asprintf ( & fn , " %s/cgroup.procs " , p ) ;
free ( p ) ;
if ( r < 0 )
return - ENOMEM ;
f = fopen ( fn , " re " ) ;
2010-07-08 08:08:32 +04:00
free ( fn ) ;
if ( ! f )
2010-07-08 23:01:42 +04:00
return - errno ;
2010-07-08 08:08:32 +04:00
2010-07-11 02:50:49 +04:00
while ( ( r = cg_read_pid ( f , & pid ) ) > 0 ) {
2010-07-08 08:08:32 +04:00
2012-01-22 21:18:51 +04:00
if ( ! kernel_threads & & is_kernel_thread ( pid ) > 0 )
continue ;
2010-07-08 08:08:32 +04:00
if ( n > = n_allocated ) {
pid_t * npids ;
n_allocated = MAX ( 16U , n * 2U ) ;
2012-04-16 20:56:18 +04:00
npids = realloc ( pids , sizeof ( pid_t ) * n_allocated ) ;
if ( ! npids ) {
2010-07-08 23:01:42 +04:00
r = - ENOMEM ;
2010-07-08 08:08:32 +04:00
goto finish ;
}
pids = npids ;
}
assert ( n < n_allocated ) ;
2010-07-11 02:50:49 +04:00
pids [ n + + ] = pid ;
2010-07-08 08:08:32 +04:00
}
2010-07-11 02:50:49 +04:00
if ( r < 0 )
goto finish ;
2012-04-16 20:56:18 +04:00
if ( n > 0 )
show_pid_array ( pids , n , prefix , n_columns , false , more , kernel_threads ) ;
2010-07-08 08:08:32 +04:00
2010-07-08 23:01:42 +04:00
r = 0 ;
2010-07-08 08:08:32 +04:00
finish :
free ( pids ) ;
if ( f )
fclose ( f ) ;
2010-07-08 23:01:42 +04:00
return r ;
}
2012-04-16 19:35:58 +04:00
int show_cgroup_by_path ( const char * path , const char * prefix , unsigned n_columns , bool kernel_threads , bool all ) {
2010-07-08 23:01:42 +04:00
DIR * d ;
char * last = NULL ;
2010-07-12 20:16:44 +04:00
char * p1 = NULL , * p2 = NULL , * fn = NULL , * gn = NULL ;
2010-07-08 23:01:42 +04:00
bool shown_pids = false ;
int r ;
2012-04-16 20:56:18 +04:00
assert ( path ) ;
2010-07-08 23:01:42 +04:00
if ( n_columns < = 0 )
n_columns = columns ( ) ;
if ( ! prefix )
prefix = " " ;
2012-04-16 19:35:58 +04:00
r = cg_fix_path ( path , & fn ) ;
if ( r < 0 )
2010-07-12 20:16:44 +04:00
return r ;
2010-07-08 23:01:42 +04:00
2012-04-16 19:35:58 +04:00
d = opendir ( fn ) ;
if ( ! d ) {
2010-07-08 23:01:42 +04:00
free ( fn ) ;
return - errno ;
}
2010-07-12 20:16:44 +04:00
while ( ( r = cg_read_subgroup ( d , & gn ) ) > 0 ) {
2012-04-16 19:35:58 +04:00
char * k ;
r = asprintf ( & k , " %s/%s " , fn , gn ) ;
free ( gn ) ;
if ( r < 0 ) {
r = - ENOMEM ;
goto finish ;
}
if ( ! all & & cg_is_empty_recursive ( NULL , k , false ) > 0 ) {
free ( k ) ;
continue ;
}
2010-07-08 23:01:42 +04:00
if ( ! shown_pids ) {
2012-01-22 21:18:51 +04:00
show_cgroup_one_by_path ( path , prefix , n_columns , true , kernel_threads ) ;
2010-07-08 23:01:42 +04:00
shown_pids = true ;
}
if ( last ) {
printf ( " %s \342 \224 \234 %s \n " , prefix , file_name_from_path ( last ) ) ;
2012-04-16 19:35:58 +04:00
if ( ! p1 ) {
p1 = strappend ( prefix , " \342 \224 \202 " ) ;
if ( ! p1 ) {
free ( k ) ;
2010-07-08 23:01:42 +04:00
r = - ENOMEM ;
goto finish ;
}
2012-04-16 19:35:58 +04:00
}
2010-07-08 23:01:42 +04:00
2012-04-16 19:35:58 +04:00
show_cgroup_by_path ( last , p1 , n_columns - 2 , kernel_threads , all ) ;
2010-07-08 23:01:42 +04:00
free ( last ) ;
}
2012-04-16 19:35:58 +04:00
last = k ;
2010-07-08 23:01:42 +04:00
}
2010-07-12 20:16:44 +04:00
if ( r < 0 )
goto finish ;
2010-07-08 23:01:42 +04:00
if ( ! shown_pids )
2012-01-22 21:18:51 +04:00
show_cgroup_one_by_path ( path , prefix , n_columns , ! ! last , kernel_threads ) ;
2010-07-08 23:01:42 +04:00
if ( last ) {
printf ( " %s \342 \224 \224 %s \n " , prefix , file_name_from_path ( last ) ) ;
2012-04-16 19:35:58 +04:00
if ( ! p2 ) {
p2 = strappend ( prefix , " " ) ;
if ( ! p2 ) {
2010-07-08 23:01:42 +04:00
r = - ENOMEM ;
goto finish ;
}
2012-04-16 19:35:58 +04:00
}
2010-07-08 23:01:42 +04:00
2012-04-16 19:35:58 +04:00
show_cgroup_by_path ( last , p2 , n_columns - 2 , kernel_threads , all ) ;
2010-07-08 23:01:42 +04:00
}
r = 0 ;
finish :
free ( p1 ) ;
free ( p2 ) ;
free ( last ) ;
free ( fn ) ;
closedir ( d ) ;
return r ;
2010-07-08 08:08:32 +04:00
}
2010-07-12 20:16:44 +04:00
2012-04-16 19:35:58 +04:00
int show_cgroup ( const char * controller , const char * path , const char * prefix , unsigned n_columns , bool kernel_threads , bool all ) {
2010-07-12 20:16:44 +04:00
char * p ;
int r ;
assert ( controller ) ;
assert ( path ) ;
2012-01-22 21:18:51 +04:00
r = cg_get_path ( controller , path , NULL , & p ) ;
if ( r < 0 )
2010-07-12 20:16:44 +04:00
return r ;
2012-04-16 19:35:58 +04:00
r = show_cgroup_by_path ( p , prefix , n_columns , kernel_threads , all ) ;
2010-07-12 20:16:44 +04:00
free ( p ) ;
return r ;
}
2012-04-16 20:56:18 +04:00
static int show_extra_pids ( const char * controller , const char * path , const char * prefix , unsigned n_columns , const pid_t pids [ ] , unsigned n_pids ) {
pid_t * copy ;
unsigned i , j ;
int r ;
assert ( controller ) ;
assert ( path ) ;
if ( n_pids < = 0 )
return 0 ;
if ( n_columns < = 0 )
n_columns = columns ( ) ;
if ( ! prefix )
prefix = " " ;
copy = new ( pid_t , n_pids ) ;
if ( ! copy )
return - ENOMEM ;
for ( i = 0 , j = 0 ; i < n_pids ; i + + ) {
char * k ;
r = cg_get_by_pid ( controller , pids [ i ] , & k ) ;
if ( r < 0 ) {
free ( copy ) ;
return r ;
}
if ( path_startswith ( k , path ) )
continue ;
copy [ j + + ] = pids [ i ] ;
}
show_pid_array ( copy , j , prefix , n_columns , true , false , false ) ;
free ( copy ) ;
return 0 ;
}
int show_cgroup_and_extra ( const char * controller , const char * path , const char * prefix , unsigned n_columns , bool kernel_threads , bool all , const pid_t extra_pids [ ] , unsigned n_extra_pids ) {
int r ;
assert ( controller ) ;
assert ( path ) ;
r = show_cgroup ( controller , path , prefix , n_columns , kernel_threads , all ) ;
if ( r < 0 )
return r ;
return show_extra_pids ( controller , path , prefix , n_columns , extra_pids , n_extra_pids ) ;
}
int show_cgroup_and_extra_by_spec ( const char * spec , const char * prefix , unsigned n_columns , bool kernel_threads , bool all , const pid_t extra_pids [ ] , unsigned n_extra_pids ) {
int r ;
char * controller , * path ;
assert ( spec ) ;
r = cg_split_spec ( spec , & controller , & path ) ;
if ( r < 0 )
return r ;
r = show_cgroup_and_extra ( controller , path , prefix , n_columns , kernel_threads , all , extra_pids , n_extra_pids ) ;
free ( controller ) ;
free ( path ) ;
return r ;
}