2015-11-17 16:53:21 +03:00
/*
* builtin - config . c
*
* Copyright ( C ) 2015 , Taeung Song < treeze . taeung @ gmail . com >
*
*/
# include "builtin.h"
# include "perf.h"
# include "util/cache.h"
2015-12-15 18:39:39 +03:00
# include <subcmd/parse-options.h>
2015-11-17 16:53:21 +03:00
# include "util/util.h"
# include "util/debug.h"
2016-04-14 10:53:19 +03:00
# include "util/config.h"
2017-07-20 21:27:39 +03:00
# include <linux/string.h>
2015-11-17 16:53:21 +03:00
2016-02-10 20:51:17 +03:00
static bool use_system_config , use_user_config ;
2015-11-17 16:53:21 +03:00
static const char * const config_usage [ ] = {
2016-11-04 09:44:20 +03:00
" perf config [<file-option>] [options] [section.name[=value] ...] " ,
2015-11-17 16:53:21 +03:00
NULL
} ;
enum actions {
ACTION_LIST = 1
} actions ;
static struct option config_options [ ] = {
OPT_SET_UINT ( ' l ' , " list " , & actions ,
" show current config variables " , ACTION_LIST ) ,
2016-02-10 20:51:17 +03:00
OPT_BOOLEAN ( 0 , " system " , & use_system_config , " use system config file " ) ,
OPT_BOOLEAN ( 0 , " user " , & use_user_config , " use user config file " ) ,
2015-11-17 16:53:21 +03:00
OPT_END ( )
} ;
2016-11-04 09:44:20 +03:00
static int set_config ( struct perf_config_set * set , const char * file_name ,
const char * var , const char * value )
{
struct perf_config_section * section = NULL ;
struct perf_config_item * item = NULL ;
const char * first_line = " # this file is auto-generated. " ;
FILE * fp ;
if ( set = = NULL )
return - 1 ;
fp = fopen ( file_name , " w " ) ;
if ( ! fp )
return - 1 ;
2016-11-04 09:44:22 +03:00
perf_config_set__collect ( set , file_name , var , value ) ;
2016-11-04 09:44:20 +03:00
fprintf ( fp , " %s \n " , first_line ) ;
/* overwrite configvariables */
perf_config_items__for_each_entry ( & set - > sections , section ) {
2016-11-04 09:44:22 +03:00
if ( ! use_system_config & & section - > from_system_config )
continue ;
2016-11-04 09:44:20 +03:00
fprintf ( fp , " [%s] \n " , section - > name ) ;
perf_config_items__for_each_entry ( & section - > items , item ) {
2017-09-07 06:18:45 +03:00
if ( ! use_system_config & & item - > from_system_config )
2016-11-04 09:44:22 +03:00
continue ;
2016-11-04 09:44:20 +03:00
if ( item - > value )
fprintf ( fp , " \t %s = %s \n " ,
item - > name , item - > value ) ;
}
}
fclose ( fp ) ;
return 0 ;
}
2016-11-04 09:44:17 +03:00
static int show_spec_config ( struct perf_config_set * set , const char * var )
{
struct perf_config_section * section ;
struct perf_config_item * item ;
if ( set = = NULL )
return - 1 ;
perf_config_items__for_each_entry ( & set - > sections , section ) {
2017-07-20 21:27:39 +03:00
if ( ! strstarts ( var , section - > name ) )
2016-11-04 09:44:17 +03:00
continue ;
perf_config_items__for_each_entry ( & section - > items , item ) {
const char * name = var + strlen ( section - > name ) + 1 ;
if ( strcmp ( name , item - > name ) = = 0 ) {
char * value = item - > value ;
if ( value ) {
printf ( " %s=%s \n " , var , value ) ;
return 0 ;
}
}
}
}
return 0 ;
}
2016-04-14 10:53:19 +03:00
static int show_config ( struct perf_config_set * set )
2015-11-17 16:53:21 +03:00
{
2016-04-14 10:53:19 +03:00
struct perf_config_section * section ;
struct perf_config_item * item ;
if ( set = = NULL )
return - 1 ;
2016-06-23 17:14:32 +03:00
perf_config_set__for_each_entry ( set , section , item ) {
char * value = item - > value ;
2016-04-14 10:53:19 +03:00
2016-06-23 17:14:32 +03:00
if ( value )
printf ( " %s.%s=%s \n " , section - > name ,
item - > name , value ) ;
2016-04-14 10:53:19 +03:00
}
2015-11-17 16:53:21 +03:00
return 0 ;
}
2016-11-04 09:44:20 +03:00
static int parse_config_arg ( char * arg , char * * var , char * * value )
2016-11-04 09:44:19 +03:00
{
const char * last_dot = strchr ( arg , ' . ' ) ;
/*
* Since " var " actually contains the section name and the real
* config variable name separated by a dot , we have to know where the dot is .
*/
if ( last_dot = = NULL | | last_dot = = arg ) {
pr_err ( " The config variable does not contain a section name: %s \n " , arg ) ;
return - 1 ;
}
if ( ! last_dot [ 1 ] ) {
pr_err ( " The config variable does not contain a variable name: %s \n " , arg ) ;
return - 1 ;
}
2016-11-04 09:44:20 +03:00
* value = strchr ( arg , ' = ' ) ;
if ( * value = = NULL )
* var = arg ;
else if ( ! strcmp ( * value , " = " ) ) {
pr_err ( " The config variable does not contain a value: %s \n " , arg ) ;
return - 1 ;
} else {
* value = * value + 1 ; /* excluding a first character '=' */
* var = strsep ( & arg , " = " ) ;
if ( * var [ 0 ] = = ' \0 ' ) {
pr_err ( " invalid config variable: %s \n " , arg ) ;
return - 1 ;
}
}
2016-11-04 09:44:19 +03:00
return 0 ;
}
2017-03-27 17:47:20 +03:00
int cmd_config ( int argc , const char * * argv )
2015-11-17 16:53:21 +03:00
{
2017-06-17 06:46:42 +03:00
int i , ret = - 1 ;
2016-04-14 10:53:19 +03:00
struct perf_config_set * set ;
2016-02-10 20:51:17 +03:00
char * user_config = mkpath ( " %s/.perfconfig " , getenv ( " HOME " ) ) ;
2017-04-26 15:21:02 +03:00
const char * config_filename ;
2015-11-17 16:53:21 +03:00
argc = parse_options ( argc , argv , config_options , config_usage ,
PARSE_OPT_STOP_AT_NON_OPTION ) ;
2016-02-10 20:51:17 +03:00
if ( use_system_config & & use_user_config ) {
pr_err ( " Error: only one config file at a time \n " ) ;
parse_options_usage ( config_usage , config_options , " user " , 0 ) ;
parse_options_usage ( NULL , config_options , " system " , 0 ) ;
return - 1 ;
}
if ( use_system_config )
config_exclusive_filename = perf_etc_perfconfig ( ) ;
else if ( use_user_config )
config_exclusive_filename = user_config ;
2017-04-26 15:21:02 +03:00
if ( ! config_exclusive_filename )
config_filename = user_config ;
else
config_filename = config_exclusive_filename ;
2016-06-23 17:14:31 +03:00
/*
* At only ' config ' sub - command , individually use the config set
* because of reinitializing with options config file location .
*/
2016-04-14 10:53:19 +03:00
set = perf_config_set__new ( ) ;
2017-06-17 06:46:42 +03:00
if ( ! set )
2016-04-14 10:53:19 +03:00
goto out_err ;
2015-11-17 16:53:21 +03:00
switch ( actions ) {
case ACTION_LIST :
if ( argc ) {
pr_err ( " Error: takes no arguments \n " ) ;
parse_options_usage ( config_usage , config_options , " l " , 1 ) ;
} else {
2017-06-17 06:46:42 +03:00
if ( show_config ( set ) < 0 ) {
2015-11-17 16:53:21 +03:00
pr_err ( " Nothing configured, "
2016-02-10 20:51:17 +03:00
" please check your %s \n " , config_filename ) ;
2017-06-17 06:46:42 +03:00
goto out_err ;
}
2015-11-17 16:53:21 +03:00
}
break ;
default :
2017-05-08 14:07:30 +03:00
if ( ! argc ) {
usage_with_options ( config_usage , config_options ) ;
break ;
}
2016-11-04 09:44:19 +03:00
2017-05-08 14:07:30 +03:00
for ( i = 0 ; argv [ i ] ; i + + ) {
char * var , * value ;
char * arg = strdup ( argv [ i ] ) ;
2016-11-04 09:44:19 +03:00
2017-05-08 14:07:30 +03:00
if ( ! arg ) {
pr_err ( " %s: strdup failed \n " , __func__ ) ;
2017-06-17 06:46:42 +03:00
goto out_err ;
2017-05-08 14:07:30 +03:00
}
if ( parse_config_arg ( arg , & var , & value ) < 0 ) {
2016-11-04 09:44:19 +03:00
free ( arg ) ;
2017-06-17 06:46:42 +03:00
goto out_err ;
2016-11-04 09:44:19 +03:00
}
2017-05-08 14:07:30 +03:00
2017-06-17 06:46:37 +03:00
if ( value = = NULL ) {
2017-06-17 06:46:42 +03:00
if ( show_spec_config ( set , var ) < 0 ) {
2017-06-17 06:46:37 +03:00
pr_err ( " %s is not configured: %s \n " ,
var , config_filename ) ;
free ( arg ) ;
2017-06-17 06:46:42 +03:00
goto out_err ;
2017-06-17 06:46:37 +03:00
}
} else {
2017-06-17 06:46:42 +03:00
if ( set_config ( set , config_filename , var , value ) < 0 ) {
2017-06-17 06:46:37 +03:00
pr_err ( " Failed to set '%s=%s' on %s \n " ,
var , value , config_filename ) ;
free ( arg ) ;
2017-06-17 06:46:42 +03:00
goto out_err ;
2017-06-17 06:46:37 +03:00
}
}
2017-05-08 14:07:30 +03:00
free ( arg ) ;
}
2015-11-17 16:53:21 +03:00
}
2017-06-17 06:46:42 +03:00
ret = 0 ;
2016-04-14 10:53:19 +03:00
out_err :
2017-06-17 06:46:42 +03:00
perf_config_set__delete ( set ) ;
2015-11-17 16:53:21 +03:00
return ret ;
}