123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400 |
- #include "util.h"
- #include "linux/string.h"
- #define K 1024LL
- /*
- * perf_atoll()
- * Parse (\d+)(b|B|kb|KB|mb|MB|gb|GB|tb|TB) (e.g. "256MB")
- * and return its numeric value
- */
- s64 perf_atoll(const char *str)
- {
- s64 length;
- char *p;
- char c;
- if (!isdigit(str[0]))
- goto out_err;
- length = strtoll(str, &p, 10);
- switch (c = *p++) {
- case 'b': case 'B':
- if (*p)
- goto out_err;
- __fallthrough;
- case '\0':
- return length;
- default:
- goto out_err;
- /* two-letter suffices */
- case 'k': case 'K':
- length <<= 10;
- break;
- case 'm': case 'M':
- length <<= 20;
- break;
- case 'g': case 'G':
- length <<= 30;
- break;
- case 't': case 'T':
- length <<= 40;
- break;
- }
- /* we want the cases to match */
- if (islower(c)) {
- if (strcmp(p, "b") != 0)
- goto out_err;
- } else {
- if (strcmp(p, "B") != 0)
- goto out_err;
- }
- return length;
- out_err:
- return -1;
- }
- /*
- * Helper function for splitting a string into an argv-like array.
- * originally copied from lib/argv_split.c
- */
- static const char *skip_sep(const char *cp)
- {
- while (*cp && isspace(*cp))
- cp++;
- return cp;
- }
- static const char *skip_arg(const char *cp)
- {
- while (*cp && !isspace(*cp))
- cp++;
- return cp;
- }
- static int count_argc(const char *str)
- {
- int count = 0;
- while (*str) {
- str = skip_sep(str);
- if (*str) {
- count++;
- str = skip_arg(str);
- }
- }
- return count;
- }
- /**
- * argv_free - free an argv
- * @argv - the argument vector to be freed
- *
- * Frees an argv and the strings it points to.
- */
- void argv_free(char **argv)
- {
- char **p;
- for (p = argv; *p; p++)
- zfree(p);
- free(argv);
- }
- /**
- * argv_split - split a string at whitespace, returning an argv
- * @str: the string to be split
- * @argcp: returned argument count
- *
- * Returns an array of pointers to strings which are split out from
- * @str. This is performed by strictly splitting on white-space; no
- * quote processing is performed. Multiple whitespace characters are
- * considered to be a single argument separator. The returned array
- * is always NULL-terminated. Returns NULL on memory allocation
- * failure.
- */
- char **argv_split(const char *str, int *argcp)
- {
- int argc = count_argc(str);
- char **argv = zalloc(sizeof(*argv) * (argc+1));
- char **argvp;
- if (argv == NULL)
- goto out;
- if (argcp)
- *argcp = argc;
- argvp = argv;
- while (*str) {
- str = skip_sep(str);
- if (*str) {
- const char *p = str;
- char *t;
- str = skip_arg(str);
- t = strndup(p, str-p);
- if (t == NULL)
- goto fail;
- *argvp++ = t;
- }
- }
- *argvp = NULL;
- out:
- return argv;
- fail:
- argv_free(argv);
- return NULL;
- }
- /* Character class matching */
- static bool __match_charclass(const char *pat, char c, const char **npat)
- {
- bool complement = false, ret = true;
- if (*pat == '!') {
- complement = true;
- pat++;
- }
- if (*pat++ == c) /* First character is special */
- goto end;
- while (*pat && *pat != ']') { /* Matching */
- if (*pat == '-' && *(pat + 1) != ']') { /* Range */
- if (*(pat - 1) <= c && c <= *(pat + 1))
- goto end;
- if (*(pat - 1) > *(pat + 1))
- goto error;
- pat += 2;
- } else if (*pat++ == c)
- goto end;
- }
- if (!*pat)
- goto error;
- ret = false;
- end:
- while (*pat && *pat != ']') /* Searching closing */
- pat++;
- if (!*pat)
- goto error;
- *npat = pat + 1;
- return complement ? !ret : ret;
- error:
- return false;
- }
- /* Glob/lazy pattern matching */
- static bool __match_glob(const char *str, const char *pat, bool ignore_space)
- {
- while (*str && *pat && *pat != '*') {
- if (ignore_space) {
- /* Ignore spaces for lazy matching */
- if (isspace(*str)) {
- str++;
- continue;
- }
- if (isspace(*pat)) {
- pat++;
- continue;
- }
- }
- if (*pat == '?') { /* Matches any single character */
- str++;
- pat++;
- continue;
- } else if (*pat == '[') /* Character classes/Ranges */
- if (__match_charclass(pat + 1, *str, &pat)) {
- str++;
- continue;
- } else
- return false;
- else if (*pat == '\\') /* Escaped char match as normal char */
- pat++;
- if (*str++ != *pat++)
- return false;
- }
- /* Check wild card */
- if (*pat == '*') {
- while (*pat == '*')
- pat++;
- if (!*pat) /* Tail wild card matches all */
- return true;
- while (*str)
- if (__match_glob(str++, pat, ignore_space))
- return true;
- }
- return !*str && !*pat;
- }
- /**
- * strglobmatch - glob expression pattern matching
- * @str: the target string to match
- * @pat: the pattern string to match
- *
- * This returns true if the @str matches @pat. @pat can includes wildcards
- * ('*','?') and character classes ([CHARS], complementation and ranges are
- * also supported). Also, this supports escape character ('\') to use special
- * characters as normal character.
- *
- * Note: if @pat syntax is broken, this always returns false.
- */
- bool strglobmatch(const char *str, const char *pat)
- {
- return __match_glob(str, pat, false);
- }
- /**
- * strlazymatch - matching pattern strings lazily with glob pattern
- * @str: the target string to match
- * @pat: the pattern string to match
- *
- * This is similar to strglobmatch, except this ignores spaces in
- * the target string.
- */
- bool strlazymatch(const char *str, const char *pat)
- {
- return __match_glob(str, pat, true);
- }
- /**
- * strtailcmp - Compare the tail of two strings
- * @s1: 1st string to be compared
- * @s2: 2nd string to be compared
- *
- * Return 0 if whole of either string is same as another's tail part.
- */
- int strtailcmp(const char *s1, const char *s2)
- {
- int i1 = strlen(s1);
- int i2 = strlen(s2);
- while (--i1 >= 0 && --i2 >= 0) {
- if (s1[i1] != s2[i2])
- return s1[i1] - s2[i2];
- }
- return 0;
- }
- /**
- * strxfrchar - Locate and replace character in @s
- * @s: The string to be searched/changed.
- * @from: Source character to be replaced.
- * @to: Destination character.
- *
- * Return pointer to the changed string.
- */
- char *strxfrchar(char *s, char from, char to)
- {
- char *p = s;
- while ((p = strchr(p, from)) != NULL)
- *p++ = to;
- return s;
- }
- /**
- * ltrim - Removes leading whitespace from @s.
- * @s: The string to be stripped.
- *
- * Return pointer to the first non-whitespace character in @s.
- */
- char *ltrim(char *s)
- {
- int len = strlen(s);
- while (len && isspace(*s)) {
- len--;
- s++;
- }
- return s;
- }
- /**
- * rtrim - Removes trailing whitespace from @s.
- * @s: The string to be stripped.
- *
- * Note that the first trailing whitespace is replaced with a %NUL-terminator
- * in the given string @s. Returns @s.
- */
- char *rtrim(char *s)
- {
- size_t size = strlen(s);
- char *end;
- if (!size)
- return s;
- end = s + size - 1;
- while (end >= s && isspace(*end))
- end--;
- *(end + 1) = '\0';
- return s;
- }
- /**
- * memdup - duplicate region of memory
- * @src: memory region to duplicate
- * @len: memory region length
- */
- void *memdup(const void *src, size_t len)
- {
- void *p;
- p = malloc(len);
- if (p)
- memcpy(p, src, len);
- return p;
- }
- char *asprintf_expr_inout_ints(const char *var, bool in, size_t nints, int *ints)
- {
- /*
- * FIXME: replace this with an expression using log10() when we
- * find a suitable implementation, maybe the one in the dvb drivers...
- *
- * "%s == %d || " = log10(MAXINT) * 2 + 8 chars for the operators
- */
- size_t size = nints * 28 + 1; /* \0 */
- size_t i, printed = 0;
- char *expr = malloc(size);
- if (expr) {
- const char *or_and = "||", *eq_neq = "==";
- char *e = expr;
- if (!in) {
- or_and = "&&";
- eq_neq = "!=";
- }
- for (i = 0; i < nints; ++i) {
- if (printed == size)
- goto out_err_overflow;
- if (i > 0)
- printed += snprintf(e + printed, size - printed, " %s ", or_and);
- printed += scnprintf(e + printed, size - printed,
- "%s %s %d", var, eq_neq, ints[i]);
- }
- }
- return expr;
- out_err_overflow:
- free(expr);
- return NULL;
- }
|