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
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/>.
* * */
# 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 ;
}
2010-07-12 20:16:44 +04:00
static int show_cgroup_one_by_path ( const char * path , const char * prefix , unsigned n_columns , bool more ) {
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 ;
2010-07-11 02:50:49 +04:00
pid_t pid , biggest = 0 ;
2010-07-08 23:01:42 +04:00
int r ;
2010-07-08 08:08:32 +04:00
2010-07-08 23:01:42 +04:00
if ( n_columns < = 0 )
n_columns = columns ( ) ;
2010-07-08 08:08:32 +04:00
2010-07-08 23:01:42 +04:00
if ( ! prefix )
prefix = " " ;
2010-07-12 20:16:44 +04:00
if ( ( r = cg_fix_path ( path , & p ) ) < 0 )
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
if ( n > = n_allocated ) {
pid_t * npids ;
n_allocated = MAX ( 16U , n * 2U ) ;
if ( ! ( npids = realloc ( pids , sizeof ( pid_t ) * n_allocated ) ) ) {
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 23:01:42 +04:00
2010-07-11 02:50:49 +04:00
if ( pid > biggest )
biggest = pid ;
2010-07-08 08:08:32 +04:00
}
2010-07-11 02:50:49 +04:00
if ( r < 0 )
goto finish ;
2010-07-08 08:08:32 +04:00
if ( n > 0 ) {
unsigned i , m ;
/* Filter duplicates */
m = 0 ;
for ( i = 0 ; i < n ; i + + ) {
unsigned j ;
for ( j = i + 1 ; j < n ; j + + )
if ( pids [ i ] = = pids [ j ] )
break ;
if ( j > = n )
pids [ m + + ] = pids [ i ] ;
}
n = m ;
/* And sort */
qsort ( pids , n , sizeof ( pid_t ) , compare ) ;
2010-07-08 23:01:42 +04:00
if ( n_columns > 8 )
n_columns - = 8 ;
2010-07-08 08:08:32 +04:00
else
2010-07-08 23:01:42 +04:00
n_columns = 20 ;
2010-07-08 08:08:32 +04:00
for ( i = 0 ; i < n ; i + + ) {
char * t = NULL ;
2010-07-08 23:01:42 +04:00
get_process_cmdline ( pids [ i ] , n_columns , & t ) ;
2010-07-08 08:08:32 +04:00
2010-07-08 23:01:42 +04:00
printf ( " %s%s %*lu %s \n " ,
2010-07-08 08:08:32 +04:00
prefix ,
2010-07-08 23:01:42 +04:00
( more | | i < n - 1 ) ? " \342 \224 \234 " : " \342 \224 \224 " ,
2010-08-12 00:04:22 +04:00
( int ) ilog10 ( biggest ) ,
2010-07-08 23:01:42 +04:00
( unsigned long ) pids [ i ] ,
strna ( t ) ) ;
2010-07-08 08:08:32 +04:00
free ( t ) ;
}
}
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 ;
}
2010-07-12 20:16:44 +04:00
int show_cgroup_by_path ( const char * path , const char * prefix , unsigned n_columns ) {
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 ;
if ( n_columns < = 0 )
n_columns = columns ( ) ;
if ( ! prefix )
prefix = " " ;
2010-07-12 20:16:44 +04:00
if ( ( r = cg_fix_path ( path , & fn ) ) < 0 )
return r ;
2010-07-08 23:01:42 +04:00
if ( ! ( d = opendir ( fn ) ) ) {
free ( fn ) ;
return - errno ;
}
2010-07-12 20:16:44 +04:00
while ( ( r = cg_read_subgroup ( d , & gn ) ) > 0 ) {
2010-07-08 23:01:42 +04:00
if ( ! shown_pids ) {
2010-07-12 20:16:44 +04:00
show_cgroup_one_by_path ( path , prefix , n_columns , true ) ;
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 ) ) ;
if ( ! p1 )
if ( ! ( p1 = strappend ( prefix , " \342 \224 \202 " ) ) ) {
r = - ENOMEM ;
goto finish ;
}
2010-07-12 20:16:44 +04:00
show_cgroup_by_path ( last , p1 , n_columns - 2 ) ;
2010-07-08 23:01:42 +04:00
free ( last ) ;
last = NULL ;
}
2010-07-12 20:16:44 +04:00
r = asprintf ( & last , " %s/%s " , fn , gn ) ;
free ( gn ) ;
if ( r < 0 ) {
2010-07-08 23:17:41 +04:00
r = - ENOMEM ;
2010-07-08 23:01:42 +04:00
goto finish ;
}
}
2010-07-12 20:16:44 +04:00
if ( r < 0 )
goto finish ;
2010-07-08 23:01:42 +04:00
if ( ! shown_pids )
2010-07-12 20:16:44 +04:00
show_cgroup_one_by_path ( path , prefix , n_columns , ! ! last ) ;
2010-07-08 23:01:42 +04:00
if ( last ) {
printf ( " %s \342 \224 \224 %s \n " , prefix , file_name_from_path ( last ) ) ;
if ( ! p2 )
if ( ! ( p2 = strappend ( prefix , " " ) ) ) {
r = - ENOMEM ;
goto finish ;
}
2010-07-12 20:16:44 +04:00
show_cgroup_by_path ( last , p2 , n_columns - 2 ) ;
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
int show_cgroup ( const char * controller , const char * path , const char * prefix , unsigned n_columns ) {
char * p ;
int r ;
assert ( controller ) ;
assert ( path ) ;
if ( ( r = cg_get_path ( controller , path , NULL , & p ) ) < 0 )
return r ;
r = show_cgroup_by_path ( p , prefix , n_columns ) ;
free ( p ) ;
return r ;
}