186470930SIngo Molnar /* 25f9273d6SNamhyung Kim * config.c 35f9273d6SNamhyung Kim * 45f9273d6SNamhyung Kim * Helper functions for parsing config items. 55f9273d6SNamhyung Kim * Originally copied from GIT source. 686470930SIngo Molnar * 786470930SIngo Molnar * Copyright (C) Linus Torvalds, 2005 886470930SIngo Molnar * Copyright (C) Johannes Schindelin, 2005 986470930SIngo Molnar * 1086470930SIngo Molnar */ 1186470930SIngo Molnar #include "util.h" 1286470930SIngo Molnar #include "cache.h" 134b6ab94eSJosh Poimboeuf #include <subcmd/exec-cmd.h> 140b93da17SNamhyung Kim #include "util/hist.h" /* perf_hist_config */ 15aa61fd05SWang Nan #include "util/llvm-utils.h" /* perf_llvm_config */ 1620105ca1STaeung Song #include "config.h" 1786470930SIngo Molnar 1886470930SIngo Molnar #define MAXNAME (256) 1986470930SIngo Molnar 2045de34bbSStephane Eranian #define DEBUG_CACHE_DIR ".debug" 2145de34bbSStephane Eranian 2245de34bbSStephane Eranian 2345de34bbSStephane Eranian char buildid_dir[MAXPATHLEN]; /* root dir for buildid, binary cache */ 2445de34bbSStephane Eranian 2586470930SIngo Molnar static FILE *config_file; 2686470930SIngo Molnar static const char *config_file_name; 2786470930SIngo Molnar static int config_linenr; 2886470930SIngo Molnar static int config_file_eof; 298a0a9c7eSTaeung Song static struct perf_config_set *config_set; 3086470930SIngo Molnar 31c7ac2417STaeung Song const char *config_exclusive_filename; 3286470930SIngo Molnar 3386470930SIngo Molnar static int get_next_char(void) 3486470930SIngo Molnar { 3586470930SIngo Molnar int c; 3686470930SIngo Molnar FILE *f; 3786470930SIngo Molnar 3886470930SIngo Molnar c = '\n'; 3986470930SIngo Molnar if ((f = config_file) != NULL) { 4086470930SIngo Molnar c = fgetc(f); 4186470930SIngo Molnar if (c == '\r') { 4286470930SIngo Molnar /* DOS like systems */ 4386470930SIngo Molnar c = fgetc(f); 4486470930SIngo Molnar if (c != '\n') { 4586470930SIngo Molnar ungetc(c, f); 4686470930SIngo Molnar c = '\r'; 4786470930SIngo Molnar } 4886470930SIngo Molnar } 4986470930SIngo Molnar if (c == '\n') 5086470930SIngo Molnar config_linenr++; 5186470930SIngo Molnar if (c == EOF) { 5286470930SIngo Molnar config_file_eof = 1; 5386470930SIngo Molnar c = '\n'; 5486470930SIngo Molnar } 5586470930SIngo Molnar } 5686470930SIngo Molnar return c; 5786470930SIngo Molnar } 5886470930SIngo Molnar 5986470930SIngo Molnar static char *parse_value(void) 6086470930SIngo Molnar { 6186470930SIngo Molnar static char value[1024]; 62f37a291cSIngo Molnar int quote = 0, comment = 0, space = 0; 63f37a291cSIngo Molnar size_t len = 0; 6486470930SIngo Molnar 6586470930SIngo Molnar for (;;) { 6686470930SIngo Molnar int c = get_next_char(); 67f37a291cSIngo Molnar 6886470930SIngo Molnar if (len >= sizeof(value) - 1) 6986470930SIngo Molnar return NULL; 7086470930SIngo Molnar if (c == '\n') { 7186470930SIngo Molnar if (quote) 7286470930SIngo Molnar return NULL; 7386470930SIngo Molnar value[len] = 0; 7486470930SIngo Molnar return value; 7586470930SIngo Molnar } 7686470930SIngo Molnar if (comment) 7786470930SIngo Molnar continue; 7886470930SIngo Molnar if (isspace(c) && !quote) { 7986470930SIngo Molnar space = 1; 8086470930SIngo Molnar continue; 8186470930SIngo Molnar } 8286470930SIngo Molnar if (!quote) { 8386470930SIngo Molnar if (c == ';' || c == '#') { 8486470930SIngo Molnar comment = 1; 8586470930SIngo Molnar continue; 8686470930SIngo Molnar } 8786470930SIngo Molnar } 8886470930SIngo Molnar if (space) { 8986470930SIngo Molnar if (len) 9086470930SIngo Molnar value[len++] = ' '; 9186470930SIngo Molnar space = 0; 9286470930SIngo Molnar } 9386470930SIngo Molnar if (c == '\\') { 9486470930SIngo Molnar c = get_next_char(); 9586470930SIngo Molnar switch (c) { 9686470930SIngo Molnar case '\n': 9786470930SIngo Molnar continue; 9886470930SIngo Molnar case 't': 9986470930SIngo Molnar c = '\t'; 10086470930SIngo Molnar break; 10186470930SIngo Molnar case 'b': 10286470930SIngo Molnar c = '\b'; 10386470930SIngo Molnar break; 10486470930SIngo Molnar case 'n': 10586470930SIngo Molnar c = '\n'; 10686470930SIngo Molnar break; 10786470930SIngo Molnar /* Some characters escape as themselves */ 10886470930SIngo Molnar case '\\': case '"': 10986470930SIngo Molnar break; 11086470930SIngo Molnar /* Reject unknown escape sequences */ 11186470930SIngo Molnar default: 11286470930SIngo Molnar return NULL; 11386470930SIngo Molnar } 11486470930SIngo Molnar value[len++] = c; 11586470930SIngo Molnar continue; 11686470930SIngo Molnar } 11786470930SIngo Molnar if (c == '"') { 11886470930SIngo Molnar quote = 1-quote; 11986470930SIngo Molnar continue; 12086470930SIngo Molnar } 12186470930SIngo Molnar value[len++] = c; 12286470930SIngo Molnar } 12386470930SIngo Molnar } 12486470930SIngo Molnar 12586470930SIngo Molnar static inline int iskeychar(int c) 12686470930SIngo Molnar { 1278dc7c651SArnaldo Carvalho de Melo return isalnum(c) || c == '-' || c == '_'; 12886470930SIngo Molnar } 12986470930SIngo Molnar 13086470930SIngo Molnar static int get_value(config_fn_t fn, void *data, char *name, unsigned int len) 13186470930SIngo Molnar { 13286470930SIngo Molnar int c; 13386470930SIngo Molnar char *value; 13486470930SIngo Molnar 13586470930SIngo Molnar /* Get the full name */ 13686470930SIngo Molnar for (;;) { 13786470930SIngo Molnar c = get_next_char(); 13886470930SIngo Molnar if (config_file_eof) 13986470930SIngo Molnar break; 14086470930SIngo Molnar if (!iskeychar(c)) 14186470930SIngo Molnar break; 14245de34bbSStephane Eranian name[len++] = c; 14386470930SIngo Molnar if (len >= MAXNAME) 14486470930SIngo Molnar return -1; 14586470930SIngo Molnar } 14686470930SIngo Molnar name[len] = 0; 14786470930SIngo Molnar while (c == ' ' || c == '\t') 14886470930SIngo Molnar c = get_next_char(); 14986470930SIngo Molnar 15086470930SIngo Molnar value = NULL; 15186470930SIngo Molnar if (c != '\n') { 15286470930SIngo Molnar if (c != '=') 15386470930SIngo Molnar return -1; 15486470930SIngo Molnar value = parse_value(); 15586470930SIngo Molnar if (!value) 15686470930SIngo Molnar return -1; 15786470930SIngo Molnar } 15886470930SIngo Molnar return fn(name, value, data); 15986470930SIngo Molnar } 16086470930SIngo Molnar 16186470930SIngo Molnar static int get_extended_base_var(char *name, int baselen, int c) 16286470930SIngo Molnar { 16386470930SIngo Molnar do { 16486470930SIngo Molnar if (c == '\n') 16586470930SIngo Molnar return -1; 16686470930SIngo Molnar c = get_next_char(); 16786470930SIngo Molnar } while (isspace(c)); 16886470930SIngo Molnar 16986470930SIngo Molnar /* We require the format to be '[base "extension"]' */ 17086470930SIngo Molnar if (c != '"') 17186470930SIngo Molnar return -1; 17286470930SIngo Molnar name[baselen++] = '.'; 17386470930SIngo Molnar 17486470930SIngo Molnar for (;;) { 17583a0944fSIngo Molnar int ch = get_next_char(); 17683a0944fSIngo Molnar 17783a0944fSIngo Molnar if (ch == '\n') 17886470930SIngo Molnar return -1; 17983a0944fSIngo Molnar if (ch == '"') 18086470930SIngo Molnar break; 18183a0944fSIngo Molnar if (ch == '\\') { 18283a0944fSIngo Molnar ch = get_next_char(); 18383a0944fSIngo Molnar if (ch == '\n') 18486470930SIngo Molnar return -1; 18586470930SIngo Molnar } 18683a0944fSIngo Molnar name[baselen++] = ch; 18786470930SIngo Molnar if (baselen > MAXNAME / 2) 18886470930SIngo Molnar return -1; 18986470930SIngo Molnar } 19086470930SIngo Molnar 19186470930SIngo Molnar /* Final ']' */ 19286470930SIngo Molnar if (get_next_char() != ']') 19386470930SIngo Molnar return -1; 19486470930SIngo Molnar return baselen; 19586470930SIngo Molnar } 19686470930SIngo Molnar 19786470930SIngo Molnar static int get_base_var(char *name) 19886470930SIngo Molnar { 19986470930SIngo Molnar int baselen = 0; 20086470930SIngo Molnar 20186470930SIngo Molnar for (;;) { 20286470930SIngo Molnar int c = get_next_char(); 20386470930SIngo Molnar if (config_file_eof) 20486470930SIngo Molnar return -1; 20586470930SIngo Molnar if (c == ']') 20686470930SIngo Molnar return baselen; 20786470930SIngo Molnar if (isspace(c)) 20886470930SIngo Molnar return get_extended_base_var(name, baselen, c); 20986470930SIngo Molnar if (!iskeychar(c) && c != '.') 21086470930SIngo Molnar return -1; 21186470930SIngo Molnar if (baselen > MAXNAME / 2) 21286470930SIngo Molnar return -1; 21386470930SIngo Molnar name[baselen++] = tolower(c); 21486470930SIngo Molnar } 21586470930SIngo Molnar } 21686470930SIngo Molnar 21786470930SIngo Molnar static int perf_parse_file(config_fn_t fn, void *data) 21886470930SIngo Molnar { 21986470930SIngo Molnar int comment = 0; 22086470930SIngo Molnar int baselen = 0; 22186470930SIngo Molnar static char var[MAXNAME]; 22286470930SIngo Molnar 22386470930SIngo Molnar /* U+FEFF Byte Order Mark in UTF8 */ 22486470930SIngo Molnar static const unsigned char *utf8_bom = (unsigned char *) "\xef\xbb\xbf"; 22586470930SIngo Molnar const unsigned char *bomptr = utf8_bom; 22686470930SIngo Molnar 22786470930SIngo Molnar for (;;) { 22849757c9cSJiri Olsa int line, c = get_next_char(); 22949757c9cSJiri Olsa 23086470930SIngo Molnar if (bomptr && *bomptr) { 23186470930SIngo Molnar /* We are at the file beginning; skip UTF8-encoded BOM 23286470930SIngo Molnar * if present. Sane editors won't put this in on their 23386470930SIngo Molnar * own, but e.g. Windows Notepad will do it happily. */ 23486470930SIngo Molnar if ((unsigned char) c == *bomptr) { 23586470930SIngo Molnar bomptr++; 23686470930SIngo Molnar continue; 23786470930SIngo Molnar } else { 23886470930SIngo Molnar /* Do not tolerate partial BOM. */ 23986470930SIngo Molnar if (bomptr != utf8_bom) 24086470930SIngo Molnar break; 24186470930SIngo Molnar /* No BOM at file beginning. Cool. */ 24286470930SIngo Molnar bomptr = NULL; 24386470930SIngo Molnar } 24486470930SIngo Molnar } 24586470930SIngo Molnar if (c == '\n') { 24686470930SIngo Molnar if (config_file_eof) 24786470930SIngo Molnar return 0; 24886470930SIngo Molnar comment = 0; 24986470930SIngo Molnar continue; 25086470930SIngo Molnar } 25186470930SIngo Molnar if (comment || isspace(c)) 25286470930SIngo Molnar continue; 25386470930SIngo Molnar if (c == '#' || c == ';') { 25486470930SIngo Molnar comment = 1; 25586470930SIngo Molnar continue; 25686470930SIngo Molnar } 25786470930SIngo Molnar if (c == '[') { 25886470930SIngo Molnar baselen = get_base_var(var); 25986470930SIngo Molnar if (baselen <= 0) 26086470930SIngo Molnar break; 26186470930SIngo Molnar var[baselen++] = '.'; 26286470930SIngo Molnar var[baselen] = 0; 26386470930SIngo Molnar continue; 26486470930SIngo Molnar } 26586470930SIngo Molnar if (!isalpha(c)) 26686470930SIngo Molnar break; 26786470930SIngo Molnar var[baselen] = tolower(c); 26849757c9cSJiri Olsa 26949757c9cSJiri Olsa /* 27049757c9cSJiri Olsa * The get_value function might or might not reach the '\n', 27149757c9cSJiri Olsa * so saving the current line number for error reporting. 27249757c9cSJiri Olsa */ 27349757c9cSJiri Olsa line = config_linenr; 27449757c9cSJiri Olsa if (get_value(fn, data, var, baselen+1) < 0) { 27549757c9cSJiri Olsa config_linenr = line; 27686470930SIngo Molnar break; 27786470930SIngo Molnar } 27849757c9cSJiri Olsa } 27978f71c99STaeung Song pr_err("bad config file line %d in %s\n", config_linenr, config_file_name); 28078f71c99STaeung Song return -1; 28186470930SIngo Molnar } 28286470930SIngo Molnar 28386470930SIngo Molnar static int parse_unit_factor(const char *end, unsigned long *val) 28486470930SIngo Molnar { 28586470930SIngo Molnar if (!*end) 28686470930SIngo Molnar return 1; 28786470930SIngo Molnar else if (!strcasecmp(end, "k")) { 28886470930SIngo Molnar *val *= 1024; 28986470930SIngo Molnar return 1; 29086470930SIngo Molnar } 29186470930SIngo Molnar else if (!strcasecmp(end, "m")) { 29286470930SIngo Molnar *val *= 1024 * 1024; 29386470930SIngo Molnar return 1; 29486470930SIngo Molnar } 29586470930SIngo Molnar else if (!strcasecmp(end, "g")) { 29686470930SIngo Molnar *val *= 1024 * 1024 * 1024; 29786470930SIngo Molnar return 1; 29886470930SIngo Molnar } 29986470930SIngo Molnar return 0; 30086470930SIngo Molnar } 30186470930SIngo Molnar 30294c0655fSJiri Olsa static int perf_parse_llong(const char *value, long long *ret) 30394c0655fSJiri Olsa { 30494c0655fSJiri Olsa if (value && *value) { 30594c0655fSJiri Olsa char *end; 30694c0655fSJiri Olsa long long val = strtoll(value, &end, 0); 30794c0655fSJiri Olsa unsigned long factor = 1; 30894c0655fSJiri Olsa 30994c0655fSJiri Olsa if (!parse_unit_factor(end, &factor)) 31094c0655fSJiri Olsa return 0; 31194c0655fSJiri Olsa *ret = val * factor; 31294c0655fSJiri Olsa return 1; 31394c0655fSJiri Olsa } 31494c0655fSJiri Olsa return 0; 31594c0655fSJiri Olsa } 31694c0655fSJiri Olsa 31786470930SIngo Molnar static int perf_parse_long(const char *value, long *ret) 31886470930SIngo Molnar { 31986470930SIngo Molnar if (value && *value) { 32086470930SIngo Molnar char *end; 32186470930SIngo Molnar long val = strtol(value, &end, 0); 32286470930SIngo Molnar unsigned long factor = 1; 32386470930SIngo Molnar if (!parse_unit_factor(end, &factor)) 32486470930SIngo Molnar return 0; 32586470930SIngo Molnar *ret = val * factor; 32686470930SIngo Molnar return 1; 32786470930SIngo Molnar } 32886470930SIngo Molnar return 0; 32986470930SIngo Molnar } 33086470930SIngo Molnar 33186470930SIngo Molnar static void die_bad_config(const char *name) 33286470930SIngo Molnar { 33386470930SIngo Molnar if (config_file_name) 33486470930SIngo Molnar die("bad config value for '%s' in %s", name, config_file_name); 33586470930SIngo Molnar die("bad config value for '%s'", name); 33686470930SIngo Molnar } 33786470930SIngo Molnar 33894c0655fSJiri Olsa u64 perf_config_u64(const char *name, const char *value) 33994c0655fSJiri Olsa { 34094c0655fSJiri Olsa long long ret = 0; 34194c0655fSJiri Olsa 34294c0655fSJiri Olsa if (!perf_parse_llong(value, &ret)) 34394c0655fSJiri Olsa die_bad_config(name); 34494c0655fSJiri Olsa return (u64) ret; 34594c0655fSJiri Olsa } 34694c0655fSJiri Olsa 34786470930SIngo Molnar int perf_config_int(const char *name, const char *value) 34886470930SIngo Molnar { 34986470930SIngo Molnar long ret = 0; 35086470930SIngo Molnar if (!perf_parse_long(value, &ret)) 35186470930SIngo Molnar die_bad_config(name); 35286470930SIngo Molnar return ret; 35386470930SIngo Molnar } 35486470930SIngo Molnar 355a41794cdSArnaldo Carvalho de Melo static int perf_config_bool_or_int(const char *name, const char *value, int *is_bool) 35686470930SIngo Molnar { 35786470930SIngo Molnar *is_bool = 1; 35886470930SIngo Molnar if (!value) 35986470930SIngo Molnar return 1; 36086470930SIngo Molnar if (!*value) 36186470930SIngo Molnar return 0; 36286470930SIngo Molnar if (!strcasecmp(value, "true") || !strcasecmp(value, "yes") || !strcasecmp(value, "on")) 36386470930SIngo Molnar return 1; 36486470930SIngo Molnar if (!strcasecmp(value, "false") || !strcasecmp(value, "no") || !strcasecmp(value, "off")) 36586470930SIngo Molnar return 0; 36686470930SIngo Molnar *is_bool = 0; 36786470930SIngo Molnar return perf_config_int(name, value); 36886470930SIngo Molnar } 36986470930SIngo Molnar 37086470930SIngo Molnar int perf_config_bool(const char *name, const char *value) 37186470930SIngo Molnar { 37286470930SIngo Molnar int discard; 37386470930SIngo Molnar return !!perf_config_bool_or_int(name, value, &discard); 37486470930SIngo Molnar } 37586470930SIngo Molnar 376814b3f51SArnaldo Carvalho de Melo static const char *perf_config_dirname(const char *name, const char *value) 37745de34bbSStephane Eranian { 37845de34bbSStephane Eranian if (!name) 37945de34bbSStephane Eranian return NULL; 38045de34bbSStephane Eranian return value; 38145de34bbSStephane Eranian } 38245de34bbSStephane Eranian 3839cb5987cSTaeung Song static int perf_buildid_config(const char *var, const char *value) 3849cb5987cSTaeung Song { 3859cb5987cSTaeung Song /* same dir for all commands */ 3869cb5987cSTaeung Song if (!strcmp(var, "buildid.dir")) { 387d8e28654SVinson Lee const char *dir = perf_config_dirname(var, value); 3889cb5987cSTaeung Song 389d8e28654SVinson Lee if (!dir) 3909cb5987cSTaeung Song return -1; 391d8e28654SVinson Lee strncpy(buildid_dir, dir, MAXPATHLEN-1); 3929cb5987cSTaeung Song buildid_dir[MAXPATHLEN-1] = '\0'; 3939cb5987cSTaeung Song } 3949cb5987cSTaeung Song 3959cb5987cSTaeung Song return 0; 3969cb5987cSTaeung Song } 3979cb5987cSTaeung Song 3981d037ca1SIrina Tirdea static int perf_default_core_config(const char *var __maybe_unused, 3991d037ca1SIrina Tirdea const char *value __maybe_unused) 40086470930SIngo Molnar { 401395cf969SPaul Bolle /* Add other config variables here. */ 40286470930SIngo Molnar return 0; 40386470930SIngo Molnar } 40486470930SIngo Molnar 405c8302367SJiri Olsa static int perf_ui_config(const char *var, const char *value) 406c8302367SJiri Olsa { 407c8302367SJiri Olsa /* Add other config variables here. */ 408c8302367SJiri Olsa if (!strcmp(var, "ui.show-headers")) { 409c8302367SJiri Olsa symbol_conf.show_hist_headers = perf_config_bool(var, value); 410c8302367SJiri Olsa return 0; 411c8302367SJiri Olsa } 412c8302367SJiri Olsa return 0; 413c8302367SJiri Olsa } 414c8302367SJiri Olsa 4151d037ca1SIrina Tirdea int perf_default_config(const char *var, const char *value, 4161d037ca1SIrina Tirdea void *dummy __maybe_unused) 41786470930SIngo Molnar { 41886470930SIngo Molnar if (!prefixcmp(var, "core.")) 41986470930SIngo Molnar return perf_default_core_config(var, value); 42086470930SIngo Molnar 4210b93da17SNamhyung Kim if (!prefixcmp(var, "hist.")) 4220b93da17SNamhyung Kim return perf_hist_config(var, value); 4230b93da17SNamhyung Kim 424c8302367SJiri Olsa if (!prefixcmp(var, "ui.")) 425c8302367SJiri Olsa return perf_ui_config(var, value); 426c8302367SJiri Olsa 4272b9240caSNamhyung Kim if (!prefixcmp(var, "call-graph.")) 4282b9240caSNamhyung Kim return perf_callchain_config(var, value); 4292b9240caSNamhyung Kim 430aa61fd05SWang Nan if (!prefixcmp(var, "llvm.")) 431aa61fd05SWang Nan return perf_llvm_config(var, value); 432aa61fd05SWang Nan 4339cb5987cSTaeung Song if (!prefixcmp(var, "buildid.")) 4349cb5987cSTaeung Song return perf_buildid_config(var, value); 4359cb5987cSTaeung Song 436395cf969SPaul Bolle /* Add other config variables here. */ 43786470930SIngo Molnar return 0; 43886470930SIngo Molnar } 43986470930SIngo Molnar 440a41794cdSArnaldo Carvalho de Melo static int perf_config_from_file(config_fn_t fn, const char *filename, void *data) 44186470930SIngo Molnar { 44286470930SIngo Molnar int ret; 44386470930SIngo Molnar FILE *f = fopen(filename, "r"); 44486470930SIngo Molnar 44586470930SIngo Molnar ret = -1; 44686470930SIngo Molnar if (f) { 44786470930SIngo Molnar config_file = f; 44886470930SIngo Molnar config_file_name = filename; 44986470930SIngo Molnar config_linenr = 1; 45086470930SIngo Molnar config_file_eof = 0; 45186470930SIngo Molnar ret = perf_parse_file(fn, data); 45286470930SIngo Molnar fclose(f); 45386470930SIngo Molnar config_file_name = NULL; 45486470930SIngo Molnar } 45586470930SIngo Molnar return ret; 45686470930SIngo Molnar } 45786470930SIngo Molnar 458c7ac2417STaeung Song const char *perf_etc_perfconfig(void) 45986470930SIngo Molnar { 46086470930SIngo Molnar static const char *system_wide; 46186470930SIngo Molnar if (!system_wide) 46286470930SIngo Molnar system_wide = system_path(ETC_PERFCONFIG); 46386470930SIngo Molnar return system_wide; 46486470930SIngo Molnar } 46586470930SIngo Molnar 46686470930SIngo Molnar static int perf_env_bool(const char *k, int def) 46786470930SIngo Molnar { 46886470930SIngo Molnar const char *v = getenv(k); 46986470930SIngo Molnar return v ? perf_config_bool(k, v) : def; 47086470930SIngo Molnar } 47186470930SIngo Molnar 472a41794cdSArnaldo Carvalho de Melo static int perf_config_system(void) 47386470930SIngo Molnar { 47486470930SIngo Molnar return !perf_env_bool("PERF_CONFIG_NOSYSTEM", 0); 47586470930SIngo Molnar } 47686470930SIngo Molnar 477a41794cdSArnaldo Carvalho de Melo static int perf_config_global(void) 47886470930SIngo Molnar { 47986470930SIngo Molnar return !perf_env_bool("PERF_CONFIG_NOGLOBAL", 0); 48086470930SIngo Molnar } 48186470930SIngo Molnar 48220105ca1STaeung Song static struct perf_config_section *find_section(struct list_head *sections, 48320105ca1STaeung Song const char *section_name) 48420105ca1STaeung Song { 48520105ca1STaeung Song struct perf_config_section *section; 48620105ca1STaeung Song 48720105ca1STaeung Song list_for_each_entry(section, sections, node) 48820105ca1STaeung Song if (!strcmp(section->name, section_name)) 48920105ca1STaeung Song return section; 49020105ca1STaeung Song 49120105ca1STaeung Song return NULL; 49220105ca1STaeung Song } 49320105ca1STaeung Song 49420105ca1STaeung Song static struct perf_config_item *find_config_item(const char *name, 49520105ca1STaeung Song struct perf_config_section *section) 49620105ca1STaeung Song { 49720105ca1STaeung Song struct perf_config_item *item; 49820105ca1STaeung Song 49920105ca1STaeung Song list_for_each_entry(item, §ion->items, node) 50020105ca1STaeung Song if (!strcmp(item->name, name)) 50120105ca1STaeung Song return item; 50220105ca1STaeung Song 50320105ca1STaeung Song return NULL; 50420105ca1STaeung Song } 50520105ca1STaeung Song 50620105ca1STaeung Song static struct perf_config_section *add_section(struct list_head *sections, 50720105ca1STaeung Song const char *section_name) 50820105ca1STaeung Song { 50920105ca1STaeung Song struct perf_config_section *section = zalloc(sizeof(*section)); 51020105ca1STaeung Song 51120105ca1STaeung Song if (!section) 51220105ca1STaeung Song return NULL; 51320105ca1STaeung Song 51420105ca1STaeung Song INIT_LIST_HEAD(§ion->items); 51520105ca1STaeung Song section->name = strdup(section_name); 51620105ca1STaeung Song if (!section->name) { 51720105ca1STaeung Song pr_debug("%s: strdup failed\n", __func__); 51820105ca1STaeung Song free(section); 51920105ca1STaeung Song return NULL; 52020105ca1STaeung Song } 52120105ca1STaeung Song 52220105ca1STaeung Song list_add_tail(§ion->node, sections); 52320105ca1STaeung Song return section; 52420105ca1STaeung Song } 52520105ca1STaeung Song 52620105ca1STaeung Song static struct perf_config_item *add_config_item(struct perf_config_section *section, 52720105ca1STaeung Song const char *name) 52820105ca1STaeung Song { 52920105ca1STaeung Song struct perf_config_item *item = zalloc(sizeof(*item)); 53020105ca1STaeung Song 53120105ca1STaeung Song if (!item) 53220105ca1STaeung Song return NULL; 53320105ca1STaeung Song 53420105ca1STaeung Song item->name = strdup(name); 53520105ca1STaeung Song if (!item->name) { 53620105ca1STaeung Song pr_debug("%s: strdup failed\n", __func__); 53720105ca1STaeung Song free(item); 53820105ca1STaeung Song return NULL; 53920105ca1STaeung Song } 54020105ca1STaeung Song 54120105ca1STaeung Song list_add_tail(&item->node, §ion->items); 54220105ca1STaeung Song return item; 54320105ca1STaeung Song } 54420105ca1STaeung Song 54520105ca1STaeung Song static int set_value(struct perf_config_item *item, const char *value) 54620105ca1STaeung Song { 54720105ca1STaeung Song char *val = strdup(value); 54820105ca1STaeung Song 54920105ca1STaeung Song if (!val) 55020105ca1STaeung Song return -1; 55120105ca1STaeung Song 55220105ca1STaeung Song zfree(&item->value); 55320105ca1STaeung Song item->value = val; 55420105ca1STaeung Song return 0; 55520105ca1STaeung Song } 55620105ca1STaeung Song 55720105ca1STaeung Song static int collect_config(const char *var, const char *value, 55820105ca1STaeung Song void *perf_config_set) 55920105ca1STaeung Song { 56020105ca1STaeung Song int ret = -1; 56120105ca1STaeung Song char *ptr, *key; 56220105ca1STaeung Song char *section_name, *name; 56320105ca1STaeung Song struct perf_config_section *section = NULL; 56420105ca1STaeung Song struct perf_config_item *item = NULL; 56520105ca1STaeung Song struct perf_config_set *set = perf_config_set; 5667db91f25STaeung Song struct list_head *sections; 56720105ca1STaeung Song 5687db91f25STaeung Song if (set == NULL) 5697db91f25STaeung Song return -1; 5707db91f25STaeung Song 5717db91f25STaeung Song sections = &set->sections; 57220105ca1STaeung Song key = ptr = strdup(var); 57320105ca1STaeung Song if (!key) { 57420105ca1STaeung Song pr_debug("%s: strdup failed\n", __func__); 57520105ca1STaeung Song return -1; 57620105ca1STaeung Song } 57720105ca1STaeung Song 57820105ca1STaeung Song section_name = strsep(&ptr, "."); 57920105ca1STaeung Song name = ptr; 58020105ca1STaeung Song if (name == NULL || value == NULL) 58120105ca1STaeung Song goto out_free; 58220105ca1STaeung Song 58320105ca1STaeung Song section = find_section(sections, section_name); 58420105ca1STaeung Song if (!section) { 58520105ca1STaeung Song section = add_section(sections, section_name); 58620105ca1STaeung Song if (!section) 58720105ca1STaeung Song goto out_free; 58820105ca1STaeung Song } 58920105ca1STaeung Song 59020105ca1STaeung Song item = find_config_item(name, section); 59120105ca1STaeung Song if (!item) { 59220105ca1STaeung Song item = add_config_item(section, name); 59320105ca1STaeung Song if (!item) 59420105ca1STaeung Song goto out_free; 59520105ca1STaeung Song } 59620105ca1STaeung Song 59708d090cfSTaeung Song /* perf_config_set can contain both user and system config items. 59808d090cfSTaeung Song * So we should know where each value is from. 59908d090cfSTaeung Song * The classification would be needed when a particular config file 60008d090cfSTaeung Song * is overwrited by setting feature i.e. set_config(). 60108d090cfSTaeung Song */ 60208d090cfSTaeung Song if (strcmp(config_file_name, perf_etc_perfconfig()) == 0) { 60308d090cfSTaeung Song section->from_system_config = true; 60408d090cfSTaeung Song item->from_system_config = true; 60508d090cfSTaeung Song } else { 60608d090cfSTaeung Song section->from_system_config = false; 60708d090cfSTaeung Song item->from_system_config = false; 60808d090cfSTaeung Song } 60908d090cfSTaeung Song 61020105ca1STaeung Song ret = set_value(item, value); 61120105ca1STaeung Song return ret; 61220105ca1STaeung Song 61320105ca1STaeung Song out_free: 61420105ca1STaeung Song free(key); 61520105ca1STaeung Song return -1; 61620105ca1STaeung Song } 61720105ca1STaeung Song 61808d090cfSTaeung Song int perf_config_set__collect(struct perf_config_set *set, const char *file_name, 619c6fc018aSTaeung Song const char *var, const char *value) 620c6fc018aSTaeung Song { 62108d090cfSTaeung Song config_file_name = file_name; 622c6fc018aSTaeung Song return collect_config(var, value, set); 623c6fc018aSTaeung Song } 624c6fc018aSTaeung Song 6258beeb00fSTaeung Song static int perf_config_set__init(struct perf_config_set *set) 6268beeb00fSTaeung Song { 6278beeb00fSTaeung Song int ret = -1; 6288beeb00fSTaeung Song const char *home = NULL; 6298beeb00fSTaeung Song 6308beeb00fSTaeung Song /* Setting $PERF_CONFIG makes perf read _only_ the given config file. */ 6318beeb00fSTaeung Song if (config_exclusive_filename) 6328beeb00fSTaeung Song return perf_config_from_file(collect_config, config_exclusive_filename, set); 6338beeb00fSTaeung Song if (perf_config_system() && !access(perf_etc_perfconfig(), R_OK)) { 6348beeb00fSTaeung Song if (perf_config_from_file(collect_config, perf_etc_perfconfig(), set) < 0) 6358beeb00fSTaeung Song goto out; 6368beeb00fSTaeung Song } 6378beeb00fSTaeung Song 6388beeb00fSTaeung Song home = getenv("HOME"); 6398beeb00fSTaeung Song if (perf_config_global() && home) { 6408beeb00fSTaeung Song char *user_config = strdup(mkpath("%s/.perfconfig", home)); 6418beeb00fSTaeung Song struct stat st; 6428beeb00fSTaeung Song 6438beeb00fSTaeung Song if (user_config == NULL) { 6448beeb00fSTaeung Song warning("Not enough memory to process %s/.perfconfig, " 6458beeb00fSTaeung Song "ignoring it.", home); 6468beeb00fSTaeung Song goto out; 6478beeb00fSTaeung Song } 6488beeb00fSTaeung Song 649*afc45cf5SArnaldo Carvalho de Melo if (stat(user_config, &st) < 0) { 650*afc45cf5SArnaldo Carvalho de Melo if (errno == ENOENT) 651*afc45cf5SArnaldo Carvalho de Melo ret = 0; 6528beeb00fSTaeung Song goto out_free; 653*afc45cf5SArnaldo Carvalho de Melo } 654*afc45cf5SArnaldo Carvalho de Melo 655*afc45cf5SArnaldo Carvalho de Melo ret = 0; 6568beeb00fSTaeung Song 6578beeb00fSTaeung Song if (st.st_uid && (st.st_uid != geteuid())) { 6588beeb00fSTaeung Song warning("File %s not owned by current user or root, " 6598beeb00fSTaeung Song "ignoring it.", user_config); 6608beeb00fSTaeung Song goto out_free; 6618beeb00fSTaeung Song } 6628beeb00fSTaeung Song 663*afc45cf5SArnaldo Carvalho de Melo if (st.st_size) 6648beeb00fSTaeung Song ret = perf_config_from_file(collect_config, user_config, set); 6658beeb00fSTaeung Song out_free: 6668beeb00fSTaeung Song free(user_config); 6678beeb00fSTaeung Song } 6688beeb00fSTaeung Song out: 6698beeb00fSTaeung Song return ret; 6708beeb00fSTaeung Song } 6718beeb00fSTaeung Song 67220105ca1STaeung Song struct perf_config_set *perf_config_set__new(void) 67320105ca1STaeung Song { 67420105ca1STaeung Song struct perf_config_set *set = zalloc(sizeof(*set)); 67520105ca1STaeung Song 67620105ca1STaeung Song if (set) { 67720105ca1STaeung Song INIT_LIST_HEAD(&set->sections); 6788beeb00fSTaeung Song if (perf_config_set__init(set) < 0) { 67925d8f48fSTaeung Song perf_config_set__delete(set); 68025d8f48fSTaeung Song set = NULL; 68125d8f48fSTaeung Song } 68220105ca1STaeung Song } 68320105ca1STaeung Song 68420105ca1STaeung Song return set; 68520105ca1STaeung Song } 68620105ca1STaeung Song 6878a0a9c7eSTaeung Song int perf_config(config_fn_t fn, void *data) 6888a0a9c7eSTaeung Song { 6898a0a9c7eSTaeung Song int ret = 0; 6908a0a9c7eSTaeung Song char key[BUFSIZ]; 6918a0a9c7eSTaeung Song struct perf_config_section *section; 6928a0a9c7eSTaeung Song struct perf_config_item *item; 6938a0a9c7eSTaeung Song 6948a0a9c7eSTaeung Song if (config_set == NULL) 6958a0a9c7eSTaeung Song return -1; 6968a0a9c7eSTaeung Song 6978a0a9c7eSTaeung Song perf_config_set__for_each_entry(config_set, section, item) { 6988a0a9c7eSTaeung Song char *value = item->value; 6998a0a9c7eSTaeung Song 7008a0a9c7eSTaeung Song if (value) { 7018a0a9c7eSTaeung Song scnprintf(key, sizeof(key), "%s.%s", 7028a0a9c7eSTaeung Song section->name, item->name); 7038a0a9c7eSTaeung Song ret = fn(key, value, data); 7048a0a9c7eSTaeung Song if (ret < 0) { 7058a0a9c7eSTaeung Song pr_err("Error: wrong config key-value pair %s=%s\n", 7068a0a9c7eSTaeung Song key, value); 7078a0a9c7eSTaeung Song break; 7088a0a9c7eSTaeung Song } 7098a0a9c7eSTaeung Song } 7108a0a9c7eSTaeung Song } 7118a0a9c7eSTaeung Song 7128a0a9c7eSTaeung Song return ret; 7138a0a9c7eSTaeung Song } 7148a0a9c7eSTaeung Song 7158a0a9c7eSTaeung Song void perf_config__init(void) 7168a0a9c7eSTaeung Song { 7178a0a9c7eSTaeung Song if (config_set == NULL) 7188a0a9c7eSTaeung Song config_set = perf_config_set__new(); 7198a0a9c7eSTaeung Song } 7208a0a9c7eSTaeung Song 7218a0a9c7eSTaeung Song void perf_config__exit(void) 7228a0a9c7eSTaeung Song { 7238a0a9c7eSTaeung Song perf_config_set__delete(config_set); 7248a0a9c7eSTaeung Song config_set = NULL; 7258a0a9c7eSTaeung Song } 7268a0a9c7eSTaeung Song 7278a0a9c7eSTaeung Song void perf_config__refresh(void) 7288a0a9c7eSTaeung Song { 7298a0a9c7eSTaeung Song perf_config__exit(); 7308a0a9c7eSTaeung Song perf_config__init(); 7318a0a9c7eSTaeung Song } 7328a0a9c7eSTaeung Song 73320105ca1STaeung Song static void perf_config_item__delete(struct perf_config_item *item) 73420105ca1STaeung Song { 73520105ca1STaeung Song zfree(&item->name); 73620105ca1STaeung Song zfree(&item->value); 73720105ca1STaeung Song free(item); 73820105ca1STaeung Song } 73920105ca1STaeung Song 74020105ca1STaeung Song static void perf_config_section__purge(struct perf_config_section *section) 74120105ca1STaeung Song { 74220105ca1STaeung Song struct perf_config_item *item, *tmp; 74320105ca1STaeung Song 74420105ca1STaeung Song list_for_each_entry_safe(item, tmp, §ion->items, node) { 74520105ca1STaeung Song list_del_init(&item->node); 74620105ca1STaeung Song perf_config_item__delete(item); 74720105ca1STaeung Song } 74820105ca1STaeung Song } 74920105ca1STaeung Song 75020105ca1STaeung Song static void perf_config_section__delete(struct perf_config_section *section) 75120105ca1STaeung Song { 75220105ca1STaeung Song perf_config_section__purge(section); 75320105ca1STaeung Song zfree(§ion->name); 75420105ca1STaeung Song free(section); 75520105ca1STaeung Song } 75620105ca1STaeung Song 75720105ca1STaeung Song static void perf_config_set__purge(struct perf_config_set *set) 75820105ca1STaeung Song { 75920105ca1STaeung Song struct perf_config_section *section, *tmp; 76020105ca1STaeung Song 76120105ca1STaeung Song list_for_each_entry_safe(section, tmp, &set->sections, node) { 76220105ca1STaeung Song list_del_init(§ion->node); 76320105ca1STaeung Song perf_config_section__delete(section); 76420105ca1STaeung Song } 76520105ca1STaeung Song } 76620105ca1STaeung Song 76720105ca1STaeung Song void perf_config_set__delete(struct perf_config_set *set) 76820105ca1STaeung Song { 769826424ccSTaeung Song if (set == NULL) 770826424ccSTaeung Song return; 771826424ccSTaeung Song 77220105ca1STaeung Song perf_config_set__purge(set); 77320105ca1STaeung Song free(set); 77420105ca1STaeung Song } 77520105ca1STaeung Song 77686470930SIngo Molnar /* 77786470930SIngo Molnar * Call this to report error for your variable that should not 77886470930SIngo Molnar * get a boolean value (i.e. "[my] var" means "true"). 77986470930SIngo Molnar */ 78086470930SIngo Molnar int config_error_nonbool(const char *var) 78186470930SIngo Molnar { 78286470930SIngo Molnar return error("Missing value for '%s'", var); 78386470930SIngo Molnar } 78445de34bbSStephane Eranian 78599ce8e9fSJiri Olsa void set_buildid_dir(const char *dir) 78645de34bbSStephane Eranian { 78799ce8e9fSJiri Olsa if (dir) 78899ce8e9fSJiri Olsa scnprintf(buildid_dir, MAXPATHLEN-1, "%s", dir); 78945de34bbSStephane Eranian 79045de34bbSStephane Eranian /* default to $HOME/.debug */ 79145de34bbSStephane Eranian if (buildid_dir[0] == '\0') { 79237194f44STaeung Song char *home = getenv("HOME"); 79337194f44STaeung Song 79437194f44STaeung Song if (home) { 79545de34bbSStephane Eranian snprintf(buildid_dir, MAXPATHLEN-1, "%s/%s", 79637194f44STaeung Song home, DEBUG_CACHE_DIR); 79745de34bbSStephane Eranian } else { 79845de34bbSStephane Eranian strncpy(buildid_dir, DEBUG_CACHE_DIR, MAXPATHLEN-1); 79945de34bbSStephane Eranian } 80045de34bbSStephane Eranian buildid_dir[MAXPATHLEN-1] = '\0'; 80145de34bbSStephane Eranian } 80245de34bbSStephane Eranian /* for communicating with external commands */ 80345de34bbSStephane Eranian setenv("PERF_BUILDID_DIR", buildid_dir, 1); 80445de34bbSStephane Eranian } 805