1b2441318SGreg Kroah-Hartman // SPDX-License-Identifier: GPL-2.0 286470930SIngo Molnar /* 35f9273d6SNamhyung Kim * config.c 45f9273d6SNamhyung Kim * 55f9273d6SNamhyung Kim * Helper functions for parsing config items. 65f9273d6SNamhyung Kim * Originally copied from GIT source. 786470930SIngo Molnar * 886470930SIngo Molnar * Copyright (C) Linus Torvalds, 2005 986470930SIngo Molnar * Copyright (C) Johannes Schindelin, 2005 1086470930SIngo Molnar * 1186470930SIngo Molnar */ 12a43783aeSArnaldo Carvalho de Melo #include <errno.h> 13391e4206SArnaldo Carvalho de Melo #include <sys/param.h> 1486470930SIngo Molnar #include "util.h" 1586470930SIngo Molnar #include "cache.h" 164b6ab94eSJosh Poimboeuf #include <subcmd/exec-cmd.h> 173fcb10e4SMark Drayton #include "util/event.h" /* proc_map_timeout */ 180b93da17SNamhyung Kim #include "util/hist.h" /* perf_hist_config */ 19aa61fd05SWang Nan #include "util/llvm-utils.h" /* perf_llvm_config */ 2020105ca1STaeung Song #include "config.h" 217a8ef4c4SArnaldo Carvalho de Melo #include <sys/types.h> 227a8ef4c4SArnaldo Carvalho de Melo #include <sys/stat.h> 237a8ef4c4SArnaldo Carvalho de Melo #include <unistd.h> 248e99b6d4SArnaldo Carvalho de Melo #include <linux/string.h> 2586470930SIngo Molnar 263d689ed6SArnaldo Carvalho de Melo #include "sane_ctype.h" 273d689ed6SArnaldo Carvalho de Melo 2886470930SIngo Molnar #define MAXNAME (256) 2986470930SIngo Molnar 3045de34bbSStephane Eranian #define DEBUG_CACHE_DIR ".debug" 3145de34bbSStephane Eranian 3245de34bbSStephane Eranian 3345de34bbSStephane Eranian char buildid_dir[MAXPATHLEN]; /* root dir for buildid, binary cache */ 3445de34bbSStephane Eranian 3586470930SIngo Molnar static FILE *config_file; 3686470930SIngo Molnar static const char *config_file_name; 3786470930SIngo Molnar static int config_linenr; 3886470930SIngo Molnar static int config_file_eof; 398a0a9c7eSTaeung Song static struct perf_config_set *config_set; 4086470930SIngo Molnar 41c7ac2417STaeung Song const char *config_exclusive_filename; 4286470930SIngo Molnar 4386470930SIngo Molnar static int get_next_char(void) 4486470930SIngo Molnar { 4586470930SIngo Molnar int c; 4686470930SIngo Molnar FILE *f; 4786470930SIngo Molnar 4886470930SIngo Molnar c = '\n'; 4986470930SIngo Molnar if ((f = config_file) != NULL) { 5086470930SIngo Molnar c = fgetc(f); 5186470930SIngo Molnar if (c == '\r') { 5286470930SIngo Molnar /* DOS like systems */ 5386470930SIngo Molnar c = fgetc(f); 5486470930SIngo Molnar if (c != '\n') { 5586470930SIngo Molnar ungetc(c, f); 5686470930SIngo Molnar c = '\r'; 5786470930SIngo Molnar } 5886470930SIngo Molnar } 5986470930SIngo Molnar if (c == '\n') 6086470930SIngo Molnar config_linenr++; 6186470930SIngo Molnar if (c == EOF) { 6286470930SIngo Molnar config_file_eof = 1; 6386470930SIngo Molnar c = '\n'; 6486470930SIngo Molnar } 6586470930SIngo Molnar } 6686470930SIngo Molnar return c; 6786470930SIngo Molnar } 6886470930SIngo Molnar 6986470930SIngo Molnar static char *parse_value(void) 7086470930SIngo Molnar { 7186470930SIngo Molnar static char value[1024]; 72f37a291cSIngo Molnar int quote = 0, comment = 0, space = 0; 73f37a291cSIngo Molnar size_t len = 0; 7486470930SIngo Molnar 7586470930SIngo Molnar for (;;) { 7686470930SIngo Molnar int c = get_next_char(); 77f37a291cSIngo Molnar 7886470930SIngo Molnar if (len >= sizeof(value) - 1) 7986470930SIngo Molnar return NULL; 8086470930SIngo Molnar if (c == '\n') { 8186470930SIngo Molnar if (quote) 8286470930SIngo Molnar return NULL; 8386470930SIngo Molnar value[len] = 0; 8486470930SIngo Molnar return value; 8586470930SIngo Molnar } 8686470930SIngo Molnar if (comment) 8786470930SIngo Molnar continue; 8886470930SIngo Molnar if (isspace(c) && !quote) { 8986470930SIngo Molnar space = 1; 9086470930SIngo Molnar continue; 9186470930SIngo Molnar } 9286470930SIngo Molnar if (!quote) { 9386470930SIngo Molnar if (c == ';' || c == '#') { 9486470930SIngo Molnar comment = 1; 9586470930SIngo Molnar continue; 9686470930SIngo Molnar } 9786470930SIngo Molnar } 9886470930SIngo Molnar if (space) { 9986470930SIngo Molnar if (len) 10086470930SIngo Molnar value[len++] = ' '; 10186470930SIngo Molnar space = 0; 10286470930SIngo Molnar } 10386470930SIngo Molnar if (c == '\\') { 10486470930SIngo Molnar c = get_next_char(); 10586470930SIngo Molnar switch (c) { 10686470930SIngo Molnar case '\n': 10786470930SIngo Molnar continue; 10886470930SIngo Molnar case 't': 10986470930SIngo Molnar c = '\t'; 11086470930SIngo Molnar break; 11186470930SIngo Molnar case 'b': 11286470930SIngo Molnar c = '\b'; 11386470930SIngo Molnar break; 11486470930SIngo Molnar case 'n': 11586470930SIngo Molnar c = '\n'; 11686470930SIngo Molnar break; 11786470930SIngo Molnar /* Some characters escape as themselves */ 11886470930SIngo Molnar case '\\': case '"': 11986470930SIngo Molnar break; 12086470930SIngo Molnar /* Reject unknown escape sequences */ 12186470930SIngo Molnar default: 12286470930SIngo Molnar return NULL; 12386470930SIngo Molnar } 12486470930SIngo Molnar value[len++] = c; 12586470930SIngo Molnar continue; 12686470930SIngo Molnar } 12786470930SIngo Molnar if (c == '"') { 12886470930SIngo Molnar quote = 1-quote; 12986470930SIngo Molnar continue; 13086470930SIngo Molnar } 13186470930SIngo Molnar value[len++] = c; 13286470930SIngo Molnar } 13386470930SIngo Molnar } 13486470930SIngo Molnar 13586470930SIngo Molnar static inline int iskeychar(int c) 13686470930SIngo Molnar { 1378dc7c651SArnaldo Carvalho de Melo return isalnum(c) || c == '-' || c == '_'; 13886470930SIngo Molnar } 13986470930SIngo Molnar 14086470930SIngo Molnar static int get_value(config_fn_t fn, void *data, char *name, unsigned int len) 14186470930SIngo Molnar { 14286470930SIngo Molnar int c; 14386470930SIngo Molnar char *value; 14486470930SIngo Molnar 14586470930SIngo Molnar /* Get the full name */ 14686470930SIngo Molnar for (;;) { 14786470930SIngo Molnar c = get_next_char(); 14886470930SIngo Molnar if (config_file_eof) 14986470930SIngo Molnar break; 15086470930SIngo Molnar if (!iskeychar(c)) 15186470930SIngo Molnar break; 15245de34bbSStephane Eranian name[len++] = c; 15386470930SIngo Molnar if (len >= MAXNAME) 15486470930SIngo Molnar return -1; 15586470930SIngo Molnar } 15686470930SIngo Molnar name[len] = 0; 15786470930SIngo Molnar while (c == ' ' || c == '\t') 15886470930SIngo Molnar c = get_next_char(); 15986470930SIngo Molnar 16086470930SIngo Molnar value = NULL; 16186470930SIngo Molnar if (c != '\n') { 16286470930SIngo Molnar if (c != '=') 16386470930SIngo Molnar return -1; 16486470930SIngo Molnar value = parse_value(); 16586470930SIngo Molnar if (!value) 16686470930SIngo Molnar return -1; 16786470930SIngo Molnar } 16886470930SIngo Molnar return fn(name, value, data); 16986470930SIngo Molnar } 17086470930SIngo Molnar 17186470930SIngo Molnar static int get_extended_base_var(char *name, int baselen, int c) 17286470930SIngo Molnar { 17386470930SIngo Molnar do { 17486470930SIngo Molnar if (c == '\n') 17586470930SIngo Molnar return -1; 17686470930SIngo Molnar c = get_next_char(); 17786470930SIngo Molnar } while (isspace(c)); 17886470930SIngo Molnar 17986470930SIngo Molnar /* We require the format to be '[base "extension"]' */ 18086470930SIngo Molnar if (c != '"') 18186470930SIngo Molnar return -1; 18286470930SIngo Molnar name[baselen++] = '.'; 18386470930SIngo Molnar 18486470930SIngo Molnar for (;;) { 18583a0944fSIngo Molnar int ch = get_next_char(); 18683a0944fSIngo Molnar 18783a0944fSIngo Molnar if (ch == '\n') 18886470930SIngo Molnar return -1; 18983a0944fSIngo Molnar if (ch == '"') 19086470930SIngo Molnar break; 19183a0944fSIngo Molnar if (ch == '\\') { 19283a0944fSIngo Molnar ch = get_next_char(); 19383a0944fSIngo Molnar if (ch == '\n') 19486470930SIngo Molnar return -1; 19586470930SIngo Molnar } 19683a0944fSIngo Molnar name[baselen++] = ch; 19786470930SIngo Molnar if (baselen > MAXNAME / 2) 19886470930SIngo Molnar return -1; 19986470930SIngo Molnar } 20086470930SIngo Molnar 20186470930SIngo Molnar /* Final ']' */ 20286470930SIngo Molnar if (get_next_char() != ']') 20386470930SIngo Molnar return -1; 20486470930SIngo Molnar return baselen; 20586470930SIngo Molnar } 20686470930SIngo Molnar 20786470930SIngo Molnar static int get_base_var(char *name) 20886470930SIngo Molnar { 20986470930SIngo Molnar int baselen = 0; 21086470930SIngo Molnar 21186470930SIngo Molnar for (;;) { 21286470930SIngo Molnar int c = get_next_char(); 21386470930SIngo Molnar if (config_file_eof) 21486470930SIngo Molnar return -1; 21586470930SIngo Molnar if (c == ']') 21686470930SIngo Molnar return baselen; 21786470930SIngo Molnar if (isspace(c)) 21886470930SIngo Molnar return get_extended_base_var(name, baselen, c); 21986470930SIngo Molnar if (!iskeychar(c) && c != '.') 22086470930SIngo Molnar return -1; 22186470930SIngo Molnar if (baselen > MAXNAME / 2) 22286470930SIngo Molnar return -1; 22386470930SIngo Molnar name[baselen++] = tolower(c); 22486470930SIngo Molnar } 22586470930SIngo Molnar } 22686470930SIngo Molnar 22786470930SIngo Molnar static int perf_parse_file(config_fn_t fn, void *data) 22886470930SIngo Molnar { 22986470930SIngo Molnar int comment = 0; 23086470930SIngo Molnar int baselen = 0; 23186470930SIngo Molnar static char var[MAXNAME]; 23286470930SIngo Molnar 23386470930SIngo Molnar /* U+FEFF Byte Order Mark in UTF8 */ 23486470930SIngo Molnar static const unsigned char *utf8_bom = (unsigned char *) "\xef\xbb\xbf"; 23586470930SIngo Molnar const unsigned char *bomptr = utf8_bom; 23686470930SIngo Molnar 23786470930SIngo Molnar for (;;) { 23849757c9cSJiri Olsa int line, c = get_next_char(); 23949757c9cSJiri Olsa 24086470930SIngo Molnar if (bomptr && *bomptr) { 24186470930SIngo Molnar /* We are at the file beginning; skip UTF8-encoded BOM 24286470930SIngo Molnar * if present. Sane editors won't put this in on their 24386470930SIngo Molnar * own, but e.g. Windows Notepad will do it happily. */ 24486470930SIngo Molnar if ((unsigned char) c == *bomptr) { 24586470930SIngo Molnar bomptr++; 24686470930SIngo Molnar continue; 24786470930SIngo Molnar } else { 24886470930SIngo Molnar /* Do not tolerate partial BOM. */ 24986470930SIngo Molnar if (bomptr != utf8_bom) 25086470930SIngo Molnar break; 25186470930SIngo Molnar /* No BOM at file beginning. Cool. */ 25286470930SIngo Molnar bomptr = NULL; 25386470930SIngo Molnar } 25486470930SIngo Molnar } 25586470930SIngo Molnar if (c == '\n') { 25686470930SIngo Molnar if (config_file_eof) 25786470930SIngo Molnar return 0; 25886470930SIngo Molnar comment = 0; 25986470930SIngo Molnar continue; 26086470930SIngo Molnar } 26186470930SIngo Molnar if (comment || isspace(c)) 26286470930SIngo Molnar continue; 26386470930SIngo Molnar if (c == '#' || c == ';') { 26486470930SIngo Molnar comment = 1; 26586470930SIngo Molnar continue; 26686470930SIngo Molnar } 26786470930SIngo Molnar if (c == '[') { 26886470930SIngo Molnar baselen = get_base_var(var); 26986470930SIngo Molnar if (baselen <= 0) 27086470930SIngo Molnar break; 27186470930SIngo Molnar var[baselen++] = '.'; 27286470930SIngo Molnar var[baselen] = 0; 27386470930SIngo Molnar continue; 27486470930SIngo Molnar } 27586470930SIngo Molnar if (!isalpha(c)) 27686470930SIngo Molnar break; 27786470930SIngo Molnar var[baselen] = tolower(c); 27849757c9cSJiri Olsa 27949757c9cSJiri Olsa /* 28049757c9cSJiri Olsa * The get_value function might or might not reach the '\n', 28149757c9cSJiri Olsa * so saving the current line number for error reporting. 28249757c9cSJiri Olsa */ 28349757c9cSJiri Olsa line = config_linenr; 28449757c9cSJiri Olsa if (get_value(fn, data, var, baselen+1) < 0) { 28549757c9cSJiri Olsa config_linenr = line; 28686470930SIngo Molnar break; 28786470930SIngo Molnar } 28849757c9cSJiri Olsa } 28978f71c99STaeung Song pr_err("bad config file line %d in %s\n", config_linenr, config_file_name); 29078f71c99STaeung Song return -1; 29186470930SIngo Molnar } 29286470930SIngo Molnar 29386470930SIngo Molnar static int parse_unit_factor(const char *end, unsigned long *val) 29486470930SIngo Molnar { 29586470930SIngo Molnar if (!*end) 29686470930SIngo Molnar return 1; 29786470930SIngo Molnar else if (!strcasecmp(end, "k")) { 29886470930SIngo Molnar *val *= 1024; 29986470930SIngo Molnar return 1; 30086470930SIngo Molnar } 30186470930SIngo Molnar else if (!strcasecmp(end, "m")) { 30286470930SIngo Molnar *val *= 1024 * 1024; 30386470930SIngo Molnar return 1; 30486470930SIngo Molnar } 30586470930SIngo Molnar else if (!strcasecmp(end, "g")) { 30686470930SIngo Molnar *val *= 1024 * 1024 * 1024; 30786470930SIngo Molnar return 1; 30886470930SIngo Molnar } 30986470930SIngo Molnar return 0; 31086470930SIngo Molnar } 31186470930SIngo Molnar 31294c0655fSJiri Olsa static int perf_parse_llong(const char *value, long long *ret) 31394c0655fSJiri Olsa { 31494c0655fSJiri Olsa if (value && *value) { 31594c0655fSJiri Olsa char *end; 31694c0655fSJiri Olsa long long val = strtoll(value, &end, 0); 31794c0655fSJiri Olsa unsigned long factor = 1; 31894c0655fSJiri Olsa 31994c0655fSJiri Olsa if (!parse_unit_factor(end, &factor)) 32094c0655fSJiri Olsa return 0; 32194c0655fSJiri Olsa *ret = val * factor; 32294c0655fSJiri Olsa return 1; 32394c0655fSJiri Olsa } 32494c0655fSJiri Olsa return 0; 32594c0655fSJiri Olsa } 32694c0655fSJiri Olsa 32786470930SIngo Molnar static int perf_parse_long(const char *value, long *ret) 32886470930SIngo Molnar { 32986470930SIngo Molnar if (value && *value) { 33086470930SIngo Molnar char *end; 33186470930SIngo Molnar long val = strtol(value, &end, 0); 33286470930SIngo Molnar unsigned long factor = 1; 33386470930SIngo Molnar if (!parse_unit_factor(end, &factor)) 33486470930SIngo Molnar return 0; 33586470930SIngo Molnar *ret = val * factor; 33686470930SIngo Molnar return 1; 33786470930SIngo Molnar } 33886470930SIngo Molnar return 0; 33986470930SIngo Molnar } 34086470930SIngo Molnar 34125ce4bb8SArnaldo Carvalho de Melo static void bad_config(const char *name) 34286470930SIngo Molnar { 34386470930SIngo Molnar if (config_file_name) 34425ce4bb8SArnaldo Carvalho de Melo pr_warning("bad config value for '%s' in %s, ignoring...\n", name, config_file_name); 34525ce4bb8SArnaldo Carvalho de Melo else 34625ce4bb8SArnaldo Carvalho de Melo pr_warning("bad config value for '%s', ignoring...\n", name); 34786470930SIngo Molnar } 34886470930SIngo Molnar 34925ce4bb8SArnaldo Carvalho de Melo int perf_config_u64(u64 *dest, const char *name, const char *value) 35094c0655fSJiri Olsa { 35194c0655fSJiri Olsa long long ret = 0; 35294c0655fSJiri Olsa 35325ce4bb8SArnaldo Carvalho de Melo if (!perf_parse_llong(value, &ret)) { 35425ce4bb8SArnaldo Carvalho de Melo bad_config(name); 35525ce4bb8SArnaldo Carvalho de Melo return -1; 35694c0655fSJiri Olsa } 35794c0655fSJiri Olsa 35825ce4bb8SArnaldo Carvalho de Melo *dest = ret; 35925ce4bb8SArnaldo Carvalho de Melo return 0; 36025ce4bb8SArnaldo Carvalho de Melo } 36125ce4bb8SArnaldo Carvalho de Melo 36225ce4bb8SArnaldo Carvalho de Melo int perf_config_int(int *dest, const char *name, const char *value) 36386470930SIngo Molnar { 36486470930SIngo Molnar long ret = 0; 36525ce4bb8SArnaldo Carvalho de Melo if (!perf_parse_long(value, &ret)) { 36625ce4bb8SArnaldo Carvalho de Melo bad_config(name); 36725ce4bb8SArnaldo Carvalho de Melo return -1; 36825ce4bb8SArnaldo Carvalho de Melo } 36925ce4bb8SArnaldo Carvalho de Melo *dest = ret; 37025ce4bb8SArnaldo Carvalho de Melo return 0; 37186470930SIngo Molnar } 37286470930SIngo Molnar 373a41794cdSArnaldo Carvalho de Melo static int perf_config_bool_or_int(const char *name, const char *value, int *is_bool) 37486470930SIngo Molnar { 37525ce4bb8SArnaldo Carvalho de Melo int ret; 37625ce4bb8SArnaldo Carvalho de Melo 37786470930SIngo Molnar *is_bool = 1; 37886470930SIngo Molnar if (!value) 37986470930SIngo Molnar return 1; 38086470930SIngo Molnar if (!*value) 38186470930SIngo Molnar return 0; 38286470930SIngo Molnar if (!strcasecmp(value, "true") || !strcasecmp(value, "yes") || !strcasecmp(value, "on")) 38386470930SIngo Molnar return 1; 38486470930SIngo Molnar if (!strcasecmp(value, "false") || !strcasecmp(value, "no") || !strcasecmp(value, "off")) 38586470930SIngo Molnar return 0; 38686470930SIngo Molnar *is_bool = 0; 38725ce4bb8SArnaldo Carvalho de Melo return perf_config_int(&ret, name, value) < 0 ? -1 : ret; 38886470930SIngo Molnar } 38986470930SIngo Molnar 39086470930SIngo Molnar int perf_config_bool(const char *name, const char *value) 39186470930SIngo Molnar { 39286470930SIngo Molnar int discard; 39386470930SIngo Molnar return !!perf_config_bool_or_int(name, value, &discard); 39486470930SIngo Molnar } 39586470930SIngo Molnar 396814b3f51SArnaldo Carvalho de Melo static const char *perf_config_dirname(const char *name, const char *value) 39745de34bbSStephane Eranian { 39845de34bbSStephane Eranian if (!name) 39945de34bbSStephane Eranian return NULL; 40045de34bbSStephane Eranian return value; 40145de34bbSStephane Eranian } 40245de34bbSStephane Eranian 4039cb5987cSTaeung Song static int perf_buildid_config(const char *var, const char *value) 4049cb5987cSTaeung Song { 4059cb5987cSTaeung Song /* same dir for all commands */ 4069cb5987cSTaeung Song if (!strcmp(var, "buildid.dir")) { 407d8e28654SVinson Lee const char *dir = perf_config_dirname(var, value); 4089cb5987cSTaeung Song 409ecc4c561SArnaldo Carvalho de Melo if (!dir) { 410ecc4c561SArnaldo Carvalho de Melo pr_err("Invalid buildid directory!\n"); 4119cb5987cSTaeung Song return -1; 412ecc4c561SArnaldo Carvalho de Melo } 413d8e28654SVinson Lee strncpy(buildid_dir, dir, MAXPATHLEN-1); 4149cb5987cSTaeung Song buildid_dir[MAXPATHLEN-1] = '\0'; 4159cb5987cSTaeung Song } 4169cb5987cSTaeung Song 4179cb5987cSTaeung Song return 0; 4189cb5987cSTaeung Song } 4199cb5987cSTaeung Song 4201d037ca1SIrina Tirdea static int perf_default_core_config(const char *var __maybe_unused, 4211d037ca1SIrina Tirdea const char *value __maybe_unused) 42286470930SIngo Molnar { 4233fcb10e4SMark Drayton if (!strcmp(var, "core.proc-map-timeout")) 4243fcb10e4SMark Drayton proc_map_timeout = strtoul(value, NULL, 10); 4253fcb10e4SMark Drayton 426395cf969SPaul Bolle /* Add other config variables here. */ 42786470930SIngo Molnar return 0; 42886470930SIngo Molnar } 42986470930SIngo Molnar 430c8302367SJiri Olsa static int perf_ui_config(const char *var, const char *value) 431c8302367SJiri Olsa { 432c8302367SJiri Olsa /* Add other config variables here. */ 433ecc4c561SArnaldo Carvalho de Melo if (!strcmp(var, "ui.show-headers")) 434c8302367SJiri Olsa symbol_conf.show_hist_headers = perf_config_bool(var, value); 435ecc4c561SArnaldo Carvalho de Melo 436c8302367SJiri Olsa return 0; 437c8302367SJiri Olsa } 438c8302367SJiri Olsa 4391d037ca1SIrina Tirdea int perf_default_config(const char *var, const char *value, 4401d037ca1SIrina Tirdea void *dummy __maybe_unused) 44186470930SIngo Molnar { 4428e99b6d4SArnaldo Carvalho de Melo if (strstarts(var, "core.")) 44386470930SIngo Molnar return perf_default_core_config(var, value); 44486470930SIngo Molnar 4458e99b6d4SArnaldo Carvalho de Melo if (strstarts(var, "hist.")) 4460b93da17SNamhyung Kim return perf_hist_config(var, value); 4470b93da17SNamhyung Kim 4488e99b6d4SArnaldo Carvalho de Melo if (strstarts(var, "ui.")) 449c8302367SJiri Olsa return perf_ui_config(var, value); 450c8302367SJiri Olsa 4518e99b6d4SArnaldo Carvalho de Melo if (strstarts(var, "call-graph.")) 4522b9240caSNamhyung Kim return perf_callchain_config(var, value); 4532b9240caSNamhyung Kim 4548e99b6d4SArnaldo Carvalho de Melo if (strstarts(var, "llvm.")) 455aa61fd05SWang Nan return perf_llvm_config(var, value); 456aa61fd05SWang Nan 4578e99b6d4SArnaldo Carvalho de Melo if (strstarts(var, "buildid.")) 4589cb5987cSTaeung Song return perf_buildid_config(var, value); 4599cb5987cSTaeung Song 460395cf969SPaul Bolle /* Add other config variables here. */ 46186470930SIngo Molnar return 0; 46286470930SIngo Molnar } 46386470930SIngo Molnar 464a41794cdSArnaldo Carvalho de Melo static int perf_config_from_file(config_fn_t fn, const char *filename, void *data) 46586470930SIngo Molnar { 46686470930SIngo Molnar int ret; 46786470930SIngo Molnar FILE *f = fopen(filename, "r"); 46886470930SIngo Molnar 46986470930SIngo Molnar ret = -1; 47086470930SIngo Molnar if (f) { 47186470930SIngo Molnar config_file = f; 47286470930SIngo Molnar config_file_name = filename; 47386470930SIngo Molnar config_linenr = 1; 47486470930SIngo Molnar config_file_eof = 0; 47586470930SIngo Molnar ret = perf_parse_file(fn, data); 47686470930SIngo Molnar fclose(f); 47786470930SIngo Molnar config_file_name = NULL; 47886470930SIngo Molnar } 47986470930SIngo Molnar return ret; 48086470930SIngo Molnar } 48186470930SIngo Molnar 482c7ac2417STaeung Song const char *perf_etc_perfconfig(void) 48386470930SIngo Molnar { 48486470930SIngo Molnar static const char *system_wide; 48586470930SIngo Molnar if (!system_wide) 48686470930SIngo Molnar system_wide = system_path(ETC_PERFCONFIG); 48786470930SIngo Molnar return system_wide; 48886470930SIngo Molnar } 48986470930SIngo Molnar 49086470930SIngo Molnar static int perf_env_bool(const char *k, int def) 49186470930SIngo Molnar { 49286470930SIngo Molnar const char *v = getenv(k); 49386470930SIngo Molnar return v ? perf_config_bool(k, v) : def; 49486470930SIngo Molnar } 49586470930SIngo Molnar 496a41794cdSArnaldo Carvalho de Melo static int perf_config_system(void) 49786470930SIngo Molnar { 49886470930SIngo Molnar return !perf_env_bool("PERF_CONFIG_NOSYSTEM", 0); 49986470930SIngo Molnar } 50086470930SIngo Molnar 501a41794cdSArnaldo Carvalho de Melo static int perf_config_global(void) 50286470930SIngo Molnar { 50386470930SIngo Molnar return !perf_env_bool("PERF_CONFIG_NOGLOBAL", 0); 50486470930SIngo Molnar } 50586470930SIngo Molnar 50620105ca1STaeung Song static struct perf_config_section *find_section(struct list_head *sections, 50720105ca1STaeung Song const char *section_name) 50820105ca1STaeung Song { 50920105ca1STaeung Song struct perf_config_section *section; 51020105ca1STaeung Song 51120105ca1STaeung Song list_for_each_entry(section, sections, node) 51220105ca1STaeung Song if (!strcmp(section->name, section_name)) 51320105ca1STaeung Song return section; 51420105ca1STaeung Song 51520105ca1STaeung Song return NULL; 51620105ca1STaeung Song } 51720105ca1STaeung Song 51820105ca1STaeung Song static struct perf_config_item *find_config_item(const char *name, 51920105ca1STaeung Song struct perf_config_section *section) 52020105ca1STaeung Song { 52120105ca1STaeung Song struct perf_config_item *item; 52220105ca1STaeung Song 52320105ca1STaeung Song list_for_each_entry(item, §ion->items, node) 52420105ca1STaeung Song if (!strcmp(item->name, name)) 52520105ca1STaeung Song return item; 52620105ca1STaeung Song 52720105ca1STaeung Song return NULL; 52820105ca1STaeung Song } 52920105ca1STaeung Song 53020105ca1STaeung Song static struct perf_config_section *add_section(struct list_head *sections, 53120105ca1STaeung Song const char *section_name) 53220105ca1STaeung Song { 53320105ca1STaeung Song struct perf_config_section *section = zalloc(sizeof(*section)); 53420105ca1STaeung Song 53520105ca1STaeung Song if (!section) 53620105ca1STaeung Song return NULL; 53720105ca1STaeung Song 53820105ca1STaeung Song INIT_LIST_HEAD(§ion->items); 53920105ca1STaeung Song section->name = strdup(section_name); 54020105ca1STaeung Song if (!section->name) { 54120105ca1STaeung Song pr_debug("%s: strdup failed\n", __func__); 54220105ca1STaeung Song free(section); 54320105ca1STaeung Song return NULL; 54420105ca1STaeung Song } 54520105ca1STaeung Song 54620105ca1STaeung Song list_add_tail(§ion->node, sections); 54720105ca1STaeung Song return section; 54820105ca1STaeung Song } 54920105ca1STaeung Song 55020105ca1STaeung Song static struct perf_config_item *add_config_item(struct perf_config_section *section, 55120105ca1STaeung Song const char *name) 55220105ca1STaeung Song { 55320105ca1STaeung Song struct perf_config_item *item = zalloc(sizeof(*item)); 55420105ca1STaeung Song 55520105ca1STaeung Song if (!item) 55620105ca1STaeung Song return NULL; 55720105ca1STaeung Song 55820105ca1STaeung Song item->name = strdup(name); 55920105ca1STaeung Song if (!item->name) { 56020105ca1STaeung Song pr_debug("%s: strdup failed\n", __func__); 56120105ca1STaeung Song free(item); 56220105ca1STaeung Song return NULL; 56320105ca1STaeung Song } 56420105ca1STaeung Song 56520105ca1STaeung Song list_add_tail(&item->node, §ion->items); 56620105ca1STaeung Song return item; 56720105ca1STaeung Song } 56820105ca1STaeung Song 56920105ca1STaeung Song static int set_value(struct perf_config_item *item, const char *value) 57020105ca1STaeung Song { 57120105ca1STaeung Song char *val = strdup(value); 57220105ca1STaeung Song 57320105ca1STaeung Song if (!val) 57420105ca1STaeung Song return -1; 57520105ca1STaeung Song 57620105ca1STaeung Song zfree(&item->value); 57720105ca1STaeung Song item->value = val; 57820105ca1STaeung Song return 0; 57920105ca1STaeung Song } 58020105ca1STaeung Song 58120105ca1STaeung Song static int collect_config(const char *var, const char *value, 58220105ca1STaeung Song void *perf_config_set) 58320105ca1STaeung Song { 58420105ca1STaeung Song int ret = -1; 58520105ca1STaeung Song char *ptr, *key; 58620105ca1STaeung Song char *section_name, *name; 58720105ca1STaeung Song struct perf_config_section *section = NULL; 58820105ca1STaeung Song struct perf_config_item *item = NULL; 58920105ca1STaeung Song struct perf_config_set *set = perf_config_set; 5907db91f25STaeung Song struct list_head *sections; 59120105ca1STaeung Song 5927db91f25STaeung Song if (set == NULL) 5937db91f25STaeung Song return -1; 5947db91f25STaeung Song 5957db91f25STaeung Song sections = &set->sections; 59620105ca1STaeung Song key = ptr = strdup(var); 59720105ca1STaeung Song if (!key) { 59820105ca1STaeung Song pr_debug("%s: strdup failed\n", __func__); 59920105ca1STaeung Song return -1; 60020105ca1STaeung Song } 60120105ca1STaeung Song 60220105ca1STaeung Song section_name = strsep(&ptr, "."); 60320105ca1STaeung Song name = ptr; 60420105ca1STaeung Song if (name == NULL || value == NULL) 60520105ca1STaeung Song goto out_free; 60620105ca1STaeung Song 60720105ca1STaeung Song section = find_section(sections, section_name); 60820105ca1STaeung Song if (!section) { 60920105ca1STaeung Song section = add_section(sections, section_name); 61020105ca1STaeung Song if (!section) 61120105ca1STaeung Song goto out_free; 61220105ca1STaeung Song } 61320105ca1STaeung Song 61420105ca1STaeung Song item = find_config_item(name, section); 61520105ca1STaeung Song if (!item) { 61620105ca1STaeung Song item = add_config_item(section, name); 61720105ca1STaeung Song if (!item) 61820105ca1STaeung Song goto out_free; 61920105ca1STaeung Song } 62020105ca1STaeung Song 62108d090cfSTaeung Song /* perf_config_set can contain both user and system config items. 62208d090cfSTaeung Song * So we should know where each value is from. 62308d090cfSTaeung Song * The classification would be needed when a particular config file 62408d090cfSTaeung Song * is overwrited by setting feature i.e. set_config(). 62508d090cfSTaeung Song */ 62608d090cfSTaeung Song if (strcmp(config_file_name, perf_etc_perfconfig()) == 0) { 62708d090cfSTaeung Song section->from_system_config = true; 62808d090cfSTaeung Song item->from_system_config = true; 62908d090cfSTaeung Song } else { 63008d090cfSTaeung Song section->from_system_config = false; 63108d090cfSTaeung Song item->from_system_config = false; 63208d090cfSTaeung Song } 63308d090cfSTaeung Song 63420105ca1STaeung Song ret = set_value(item, value); 63520105ca1STaeung Song return ret; 63620105ca1STaeung Song 63720105ca1STaeung Song out_free: 63820105ca1STaeung Song free(key); 63920105ca1STaeung Song return -1; 64020105ca1STaeung Song } 64120105ca1STaeung Song 64208d090cfSTaeung Song int perf_config_set__collect(struct perf_config_set *set, const char *file_name, 643c6fc018aSTaeung Song const char *var, const char *value) 644c6fc018aSTaeung Song { 64508d090cfSTaeung Song config_file_name = file_name; 646c6fc018aSTaeung Song return collect_config(var, value, set); 647c6fc018aSTaeung Song } 648c6fc018aSTaeung Song 6498beeb00fSTaeung Song static int perf_config_set__init(struct perf_config_set *set) 6508beeb00fSTaeung Song { 6518beeb00fSTaeung Song int ret = -1; 6528beeb00fSTaeung Song const char *home = NULL; 6533e00cbe8SJiri Olsa char *user_config; 6543e00cbe8SJiri Olsa struct stat st; 6558beeb00fSTaeung Song 6568beeb00fSTaeung Song /* Setting $PERF_CONFIG makes perf read _only_ the given config file. */ 6578beeb00fSTaeung Song if (config_exclusive_filename) 6588beeb00fSTaeung Song return perf_config_from_file(collect_config, config_exclusive_filename, set); 6598beeb00fSTaeung Song if (perf_config_system() && !access(perf_etc_perfconfig(), R_OK)) { 6608beeb00fSTaeung Song if (perf_config_from_file(collect_config, perf_etc_perfconfig(), set) < 0) 6618beeb00fSTaeung Song goto out; 6628beeb00fSTaeung Song } 6638beeb00fSTaeung Song 6648beeb00fSTaeung Song home = getenv("HOME"); 6658beeb00fSTaeung Song 6663e00cbe8SJiri Olsa /* 6673e00cbe8SJiri Olsa * Skip reading user config if: 6683e00cbe8SJiri Olsa * - there is no place to read it from (HOME) 6693e00cbe8SJiri Olsa * - we are asked not to (PERF_CONFIG_NOGLOBAL=1) 6703e00cbe8SJiri Olsa */ 6713e00cbe8SJiri Olsa if (!home || !*home || !perf_config_global()) 6723e00cbe8SJiri Olsa return 0; 6733e00cbe8SJiri Olsa 6743e00cbe8SJiri Olsa user_config = strdup(mkpath("%s/.perfconfig", home)); 6758beeb00fSTaeung Song if (user_config == NULL) { 6764cf134e7SArnaldo Carvalho de Melo pr_warning("Not enough memory to process %s/.perfconfig, ignoring it.", home); 6778beeb00fSTaeung Song goto out; 6788beeb00fSTaeung Song } 6798beeb00fSTaeung Song 680afc45cf5SArnaldo Carvalho de Melo if (stat(user_config, &st) < 0) { 681afc45cf5SArnaldo Carvalho de Melo if (errno == ENOENT) 682afc45cf5SArnaldo Carvalho de Melo ret = 0; 6838beeb00fSTaeung Song goto out_free; 684afc45cf5SArnaldo Carvalho de Melo } 685afc45cf5SArnaldo Carvalho de Melo 686afc45cf5SArnaldo Carvalho de Melo ret = 0; 6878beeb00fSTaeung Song 6888beeb00fSTaeung Song if (st.st_uid && (st.st_uid != geteuid())) { 6894cf134e7SArnaldo Carvalho de Melo pr_warning("File %s not owned by current user or root, ignoring it.", user_config); 6908beeb00fSTaeung Song goto out_free; 6918beeb00fSTaeung Song } 6928beeb00fSTaeung Song 693afc45cf5SArnaldo Carvalho de Melo if (st.st_size) 6948beeb00fSTaeung Song ret = perf_config_from_file(collect_config, user_config, set); 6953e00cbe8SJiri Olsa 6968beeb00fSTaeung Song out_free: 6978beeb00fSTaeung Song free(user_config); 6988beeb00fSTaeung Song out: 6998beeb00fSTaeung Song return ret; 7008beeb00fSTaeung Song } 7018beeb00fSTaeung Song 70220105ca1STaeung Song struct perf_config_set *perf_config_set__new(void) 70320105ca1STaeung Song { 70420105ca1STaeung Song struct perf_config_set *set = zalloc(sizeof(*set)); 70520105ca1STaeung Song 70620105ca1STaeung Song if (set) { 70720105ca1STaeung Song INIT_LIST_HEAD(&set->sections); 70855421b4fSTaeung Song perf_config_set__init(set); 70920105ca1STaeung Song } 71020105ca1STaeung Song 71120105ca1STaeung Song return set; 71220105ca1STaeung Song } 71320105ca1STaeung Song 714d01bd1acSArnaldo Carvalho de Melo static int perf_config__init(void) 715d01bd1acSArnaldo Carvalho de Melo { 716d01bd1acSArnaldo Carvalho de Melo if (config_set == NULL) 717d01bd1acSArnaldo Carvalho de Melo config_set = perf_config_set__new(); 718d01bd1acSArnaldo Carvalho de Melo 719d01bd1acSArnaldo Carvalho de Melo return config_set == NULL; 720d01bd1acSArnaldo Carvalho de Melo } 721d01bd1acSArnaldo Carvalho de Melo 7228a0a9c7eSTaeung Song int perf_config(config_fn_t fn, void *data) 7238a0a9c7eSTaeung Song { 7248a0a9c7eSTaeung Song int ret = 0; 7258a0a9c7eSTaeung Song char key[BUFSIZ]; 7268a0a9c7eSTaeung Song struct perf_config_section *section; 7278a0a9c7eSTaeung Song struct perf_config_item *item; 7288a0a9c7eSTaeung Song 729d01bd1acSArnaldo Carvalho de Melo if (config_set == NULL && perf_config__init()) 7308a0a9c7eSTaeung Song return -1; 7318a0a9c7eSTaeung Song 7328a0a9c7eSTaeung Song perf_config_set__for_each_entry(config_set, section, item) { 7338a0a9c7eSTaeung Song char *value = item->value; 7348a0a9c7eSTaeung Song 7358a0a9c7eSTaeung Song if (value) { 7368a0a9c7eSTaeung Song scnprintf(key, sizeof(key), "%s.%s", 7378a0a9c7eSTaeung Song section->name, item->name); 7388a0a9c7eSTaeung Song ret = fn(key, value, data); 7398a0a9c7eSTaeung Song if (ret < 0) { 7408a0a9c7eSTaeung Song pr_err("Error: wrong config key-value pair %s=%s\n", 7418a0a9c7eSTaeung Song key, value); 7428a0a9c7eSTaeung Song break; 7438a0a9c7eSTaeung Song } 7448a0a9c7eSTaeung Song } 7458a0a9c7eSTaeung Song } 7468a0a9c7eSTaeung Song 7478a0a9c7eSTaeung Song return ret; 7488a0a9c7eSTaeung Song } 7498a0a9c7eSTaeung Song 7508a0a9c7eSTaeung Song void perf_config__exit(void) 7518a0a9c7eSTaeung Song { 7528a0a9c7eSTaeung Song perf_config_set__delete(config_set); 7538a0a9c7eSTaeung Song config_set = NULL; 7548a0a9c7eSTaeung Song } 7558a0a9c7eSTaeung Song 7568a0a9c7eSTaeung Song void perf_config__refresh(void) 7578a0a9c7eSTaeung Song { 7588a0a9c7eSTaeung Song perf_config__exit(); 7598a0a9c7eSTaeung Song perf_config__init(); 7608a0a9c7eSTaeung Song } 7618a0a9c7eSTaeung Song 76220105ca1STaeung Song static void perf_config_item__delete(struct perf_config_item *item) 76320105ca1STaeung Song { 76420105ca1STaeung Song zfree(&item->name); 76520105ca1STaeung Song zfree(&item->value); 76620105ca1STaeung Song free(item); 76720105ca1STaeung Song } 76820105ca1STaeung Song 76920105ca1STaeung Song static void perf_config_section__purge(struct perf_config_section *section) 77020105ca1STaeung Song { 77120105ca1STaeung Song struct perf_config_item *item, *tmp; 77220105ca1STaeung Song 77320105ca1STaeung Song list_for_each_entry_safe(item, tmp, §ion->items, node) { 77420105ca1STaeung Song list_del_init(&item->node); 77520105ca1STaeung Song perf_config_item__delete(item); 77620105ca1STaeung Song } 77720105ca1STaeung Song } 77820105ca1STaeung Song 77920105ca1STaeung Song static void perf_config_section__delete(struct perf_config_section *section) 78020105ca1STaeung Song { 78120105ca1STaeung Song perf_config_section__purge(section); 78220105ca1STaeung Song zfree(§ion->name); 78320105ca1STaeung Song free(section); 78420105ca1STaeung Song } 78520105ca1STaeung Song 78620105ca1STaeung Song static void perf_config_set__purge(struct perf_config_set *set) 78720105ca1STaeung Song { 78820105ca1STaeung Song struct perf_config_section *section, *tmp; 78920105ca1STaeung Song 79020105ca1STaeung Song list_for_each_entry_safe(section, tmp, &set->sections, node) { 79120105ca1STaeung Song list_del_init(§ion->node); 79220105ca1STaeung Song perf_config_section__delete(section); 79320105ca1STaeung Song } 79420105ca1STaeung Song } 79520105ca1STaeung Song 79620105ca1STaeung Song void perf_config_set__delete(struct perf_config_set *set) 79720105ca1STaeung Song { 798826424ccSTaeung Song if (set == NULL) 799826424ccSTaeung Song return; 800826424ccSTaeung Song 80120105ca1STaeung Song perf_config_set__purge(set); 80220105ca1STaeung Song free(set); 80320105ca1STaeung Song } 80420105ca1STaeung Song 80586470930SIngo Molnar /* 80686470930SIngo Molnar * Call this to report error for your variable that should not 80786470930SIngo Molnar * get a boolean value (i.e. "[my] var" means "true"). 80886470930SIngo Molnar */ 80986470930SIngo Molnar int config_error_nonbool(const char *var) 81086470930SIngo Molnar { 81162d94b00SArnaldo Carvalho de Melo pr_err("Missing value for '%s'", var); 81262d94b00SArnaldo Carvalho de Melo return -1; 81386470930SIngo Molnar } 81445de34bbSStephane Eranian 81599ce8e9fSJiri Olsa void set_buildid_dir(const char *dir) 81645de34bbSStephane Eranian { 81799ce8e9fSJiri Olsa if (dir) 818*75c375c0SSihyeon Jang scnprintf(buildid_dir, MAXPATHLEN, "%s", dir); 81945de34bbSStephane Eranian 82045de34bbSStephane Eranian /* default to $HOME/.debug */ 82145de34bbSStephane Eranian if (buildid_dir[0] == '\0') { 82237194f44STaeung Song char *home = getenv("HOME"); 82337194f44STaeung Song 82437194f44STaeung Song if (home) { 825*75c375c0SSihyeon Jang snprintf(buildid_dir, MAXPATHLEN, "%s/%s", 82637194f44STaeung Song home, DEBUG_CACHE_DIR); 82745de34bbSStephane Eranian } else { 82845de34bbSStephane Eranian strncpy(buildid_dir, DEBUG_CACHE_DIR, MAXPATHLEN-1); 82945de34bbSStephane Eranian } 83045de34bbSStephane Eranian buildid_dir[MAXPATHLEN-1] = '\0'; 83145de34bbSStephane Eranian } 83245de34bbSStephane Eranian /* for communicating with external commands */ 83345de34bbSStephane Eranian setenv("PERF_BUILDID_DIR", buildid_dir, 1); 83445de34bbSStephane Eranian } 835