123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662 |
- // SPDX-License-Identifier: GPL-2.0
- /*
- * Common code for probe-based Dynamic events.
- *
- * This code was copied from kernel/trace/trace_kprobe.c written by
- * Masami Hiramatsu <masami.hiramatsu.pt@hitachi.com>
- *
- * Updates to make this generic:
- * Copyright (C) IBM Corporation, 2010-2011
- * Author: Srikar Dronamraju
- */
- #define pr_fmt(fmt) "trace_probe: " fmt
- #include "trace_probe.h"
- const char *reserved_field_names[] = {
- "common_type",
- "common_flags",
- "common_preempt_count",
- "common_pid",
- "common_tgid",
- FIELD_STRING_IP,
- FIELD_STRING_RETIP,
- FIELD_STRING_FUNC,
- };
- /* Printing in basic type function template */
- #define DEFINE_BASIC_PRINT_TYPE_FUNC(tname, type, fmt) \
- int PRINT_TYPE_FUNC_NAME(tname)(struct trace_seq *s, void *data, void *ent)\
- { \
- trace_seq_printf(s, fmt, *(type *)data); \
- return !trace_seq_has_overflowed(s); \
- } \
- const char PRINT_TYPE_FMT_NAME(tname)[] = fmt;
- DEFINE_BASIC_PRINT_TYPE_FUNC(u8, u8, "%u")
- DEFINE_BASIC_PRINT_TYPE_FUNC(u16, u16, "%u")
- DEFINE_BASIC_PRINT_TYPE_FUNC(u32, u32, "%u")
- DEFINE_BASIC_PRINT_TYPE_FUNC(u64, u64, "%Lu")
- DEFINE_BASIC_PRINT_TYPE_FUNC(s8, s8, "%d")
- DEFINE_BASIC_PRINT_TYPE_FUNC(s16, s16, "%d")
- DEFINE_BASIC_PRINT_TYPE_FUNC(s32, s32, "%d")
- DEFINE_BASIC_PRINT_TYPE_FUNC(s64, s64, "%Ld")
- DEFINE_BASIC_PRINT_TYPE_FUNC(x8, u8, "0x%x")
- DEFINE_BASIC_PRINT_TYPE_FUNC(x16, u16, "0x%x")
- DEFINE_BASIC_PRINT_TYPE_FUNC(x32, u32, "0x%x")
- DEFINE_BASIC_PRINT_TYPE_FUNC(x64, u64, "0x%Lx")
- int PRINT_TYPE_FUNC_NAME(symbol)(struct trace_seq *s, void *data, void *ent)
- {
- trace_seq_printf(s, "%pS", (void *)*(unsigned long *)data);
- return !trace_seq_has_overflowed(s);
- }
- const char PRINT_TYPE_FMT_NAME(symbol)[] = "%pS";
- /* Print type function for string type */
- int PRINT_TYPE_FUNC_NAME(string)(struct trace_seq *s, void *data, void *ent)
- {
- int len = *(u32 *)data >> 16;
- if (!len)
- trace_seq_puts(s, "(fault)");
- else
- trace_seq_printf(s, "\"%s\"",
- (const char *)get_loc_data(data, ent));
- return !trace_seq_has_overflowed(s);
- }
- const char PRINT_TYPE_FMT_NAME(string)[] = "\\\"%s\\\"";
- /* Fetch type information table */
- static const struct fetch_type probe_fetch_types[] = {
- /* Special types */
- __ASSIGN_FETCH_TYPE("string", string, string, sizeof(u32), 1,
- "__data_loc char[]"),
- /* Basic types */
- ASSIGN_FETCH_TYPE(u8, u8, 0),
- ASSIGN_FETCH_TYPE(u16, u16, 0),
- ASSIGN_FETCH_TYPE(u32, u32, 0),
- ASSIGN_FETCH_TYPE(u64, u64, 0),
- ASSIGN_FETCH_TYPE(s8, u8, 1),
- ASSIGN_FETCH_TYPE(s16, u16, 1),
- ASSIGN_FETCH_TYPE(s32, u32, 1),
- ASSIGN_FETCH_TYPE(s64, u64, 1),
- ASSIGN_FETCH_TYPE_ALIAS(x8, u8, u8, 0),
- ASSIGN_FETCH_TYPE_ALIAS(x16, u16, u16, 0),
- ASSIGN_FETCH_TYPE_ALIAS(x32, u32, u32, 0),
- ASSIGN_FETCH_TYPE_ALIAS(x64, u64, u64, 0),
- ASSIGN_FETCH_TYPE_ALIAS(symbol, ADDR_FETCH_TYPE, ADDR_FETCH_TYPE, 0),
- ASSIGN_FETCH_TYPE_END
- };
- static const struct fetch_type *find_fetch_type(const char *type)
- {
- int i;
- if (!type)
- type = DEFAULT_FETCH_TYPE_STR;
- /* Special case: bitfield */
- if (*type == 'b') {
- unsigned long bs;
- type = strchr(type, '/');
- if (!type)
- goto fail;
- type++;
- if (kstrtoul(type, 0, &bs))
- goto fail;
- switch (bs) {
- case 8:
- return find_fetch_type("u8");
- case 16:
- return find_fetch_type("u16");
- case 32:
- return find_fetch_type("u32");
- case 64:
- return find_fetch_type("u64");
- default:
- goto fail;
- }
- }
- for (i = 0; probe_fetch_types[i].name; i++) {
- if (strcmp(type, probe_fetch_types[i].name) == 0)
- return &probe_fetch_types[i];
- }
- fail:
- return NULL;
- }
- /* Split symbol and offset. */
- int traceprobe_split_symbol_offset(char *symbol, long *offset)
- {
- char *tmp;
- int ret;
- if (!offset)
- return -EINVAL;
- tmp = strpbrk(symbol, "+-");
- if (tmp) {
- ret = kstrtol(tmp, 0, offset);
- if (ret)
- return ret;
- *tmp = '\0';
- } else
- *offset = 0;
- return 0;
- }
- #define PARAM_MAX_STACK (THREAD_SIZE / sizeof(unsigned long))
- static int parse_probe_vars(char *arg, const struct fetch_type *t,
- struct fetch_insn *code, unsigned int flags)
- {
- int ret = 0;
- unsigned long param;
- if (strcmp(arg, "retval") == 0) {
- if (flags & TPARG_FL_RETURN)
- code->op = FETCH_OP_RETVAL;
- else
- ret = -EINVAL;
- } else if (strncmp(arg, "stack", 5) == 0) {
- if (arg[5] == '\0') {
- code->op = FETCH_OP_STACKP;
- } else if (isdigit(arg[5])) {
- ret = kstrtoul(arg + 5, 10, ¶m);
- if (ret || ((flags & TPARG_FL_KERNEL) &&
- param > PARAM_MAX_STACK))
- ret = -EINVAL;
- else {
- code->op = FETCH_OP_STACK;
- code->param = (unsigned int)param;
- }
- } else
- ret = -EINVAL;
- } else if (strcmp(arg, "comm") == 0) {
- code->op = FETCH_OP_COMM;
- #ifdef CONFIG_HAVE_FUNCTION_ARG_ACCESS_API
- } else if (((flags & TPARG_FL_MASK) ==
- (TPARG_FL_KERNEL | TPARG_FL_FENTRY)) &&
- strncmp(arg, "arg", 3) == 0) {
- if (!isdigit(arg[3]))
- return -EINVAL;
- ret = kstrtoul(arg + 3, 10, ¶m);
- if (ret || !param || param > PARAM_MAX_STACK)
- return -EINVAL;
- code->op = FETCH_OP_ARG;
- code->param = (unsigned int)param - 1;
- #endif
- } else
- ret = -EINVAL;
- return ret;
- }
- /* Recursive argument parser */
- static int
- parse_probe_arg(char *arg, const struct fetch_type *type,
- struct fetch_insn **pcode, struct fetch_insn *end,
- unsigned int flags)
- {
- struct fetch_insn *code = *pcode;
- unsigned long param;
- long offset = 0;
- char *tmp;
- int ret = 0;
- switch (arg[0]) {
- case '$':
- ret = parse_probe_vars(arg + 1, type, code, flags);
- break;
- case '%': /* named register */
- ret = regs_query_register_offset(arg + 1);
- if (ret >= 0) {
- code->op = FETCH_OP_REG;
- code->param = (unsigned int)ret;
- ret = 0;
- }
- break;
- case '@': /* memory, file-offset or symbol */
- if (isdigit(arg[1])) {
- ret = kstrtoul(arg + 1, 0, ¶m);
- if (ret)
- break;
- /* load address */
- code->op = FETCH_OP_IMM;
- code->immediate = param;
- } else if (arg[1] == '+') {
- /* kprobes don't support file offsets */
- if (flags & TPARG_FL_KERNEL)
- return -EINVAL;
- ret = kstrtol(arg + 2, 0, &offset);
- if (ret)
- break;
- code->op = FETCH_OP_FOFFS;
- code->immediate = (unsigned long)offset; // imm64?
- } else {
- /* uprobes don't support symbols */
- if (!(flags & TPARG_FL_KERNEL))
- return -EINVAL;
- /* Preserve symbol for updating */
- code->op = FETCH_NOP_SYMBOL;
- code->data = kstrdup(arg + 1, GFP_KERNEL);
- if (!code->data)
- return -ENOMEM;
- if (++code == end)
- return -E2BIG;
- code->op = FETCH_OP_IMM;
- code->immediate = 0;
- }
- /* These are fetching from memory */
- if (++code == end)
- return -E2BIG;
- *pcode = code;
- code->op = FETCH_OP_DEREF;
- code->offset = offset;
- break;
- case '+': /* deref memory */
- arg++; /* Skip '+', because kstrtol() rejects it. */
- case '-':
- tmp = strchr(arg, '(');
- if (!tmp)
- return -EINVAL;
- *tmp = '\0';
- ret = kstrtol(arg, 0, &offset);
- if (ret)
- break;
- arg = tmp + 1;
- tmp = strrchr(arg, ')');
- if (tmp) {
- const struct fetch_type *t2 = find_fetch_type(NULL);
- *tmp = '\0';
- ret = parse_probe_arg(arg, t2, &code, end, flags);
- if (ret)
- break;
- if (code->op == FETCH_OP_COMM)
- return -EINVAL;
- if (++code == end)
- return -E2BIG;
- *pcode = code;
- code->op = FETCH_OP_DEREF;
- code->offset = offset;
- }
- break;
- }
- if (!ret && code->op == FETCH_OP_NOP) {
- /* Parsed, but do not find fetch method */
- ret = -EINVAL;
- }
- return ret;
- }
- #define BYTES_TO_BITS(nb) ((BITS_PER_LONG * (nb)) / sizeof(long))
- /* Bitfield type needs to be parsed into a fetch function */
- static int __parse_bitfield_probe_arg(const char *bf,
- const struct fetch_type *t,
- struct fetch_insn **pcode)
- {
- struct fetch_insn *code = *pcode;
- unsigned long bw, bo;
- char *tail;
- if (*bf != 'b')
- return 0;
- bw = simple_strtoul(bf + 1, &tail, 0); /* Use simple one */
- if (bw == 0 || *tail != '@')
- return -EINVAL;
- bf = tail + 1;
- bo = simple_strtoul(bf, &tail, 0);
- if (tail == bf || *tail != '/')
- return -EINVAL;
- code++;
- if (code->op != FETCH_OP_NOP)
- return -E2BIG;
- *pcode = code;
- code->op = FETCH_OP_MOD_BF;
- code->lshift = BYTES_TO_BITS(t->size) - (bw + bo);
- code->rshift = BYTES_TO_BITS(t->size) - bw;
- code->basesize = t->size;
- return (BYTES_TO_BITS(t->size) < (bw + bo)) ? -EINVAL : 0;
- }
- /* String length checking wrapper */
- int traceprobe_parse_probe_arg(char *arg, ssize_t *size,
- struct probe_arg *parg, unsigned int flags)
- {
- struct fetch_insn *code, *scode, *tmp = NULL;
- char *t, *t2;
- int ret, len;
- if (strlen(arg) > MAX_ARGSTR_LEN) {
- pr_info("Argument is too long.: %s\n", arg);
- return -ENOSPC;
- }
- parg->comm = kstrdup(arg, GFP_KERNEL);
- if (!parg->comm) {
- pr_info("Failed to allocate memory for command '%s'.\n", arg);
- return -ENOMEM;
- }
- t = strchr(arg, ':');
- if (t) {
- *t = '\0';
- t2 = strchr(++t, '[');
- if (t2) {
- *t2 = '\0';
- parg->count = simple_strtoul(t2 + 1, &t2, 0);
- if (strcmp(t2, "]") || parg->count == 0)
- return -EINVAL;
- if (parg->count > MAX_ARRAY_LEN)
- return -E2BIG;
- }
- }
- /*
- * The default type of $comm should be "string", and it can't be
- * dereferenced.
- */
- if (!t && strcmp(arg, "$comm") == 0)
- parg->type = find_fetch_type("string");
- else
- parg->type = find_fetch_type(t);
- if (!parg->type) {
- pr_info("Unsupported type: %s\n", t);
- return -EINVAL;
- }
- parg->offset = *size;
- *size += parg->type->size * (parg->count ?: 1);
- if (parg->count) {
- len = strlen(parg->type->fmttype) + 6;
- parg->fmt = kmalloc(len, GFP_KERNEL);
- if (!parg->fmt)
- return -ENOMEM;
- snprintf(parg->fmt, len, "%s[%d]", parg->type->fmttype,
- parg->count);
- }
- code = tmp = kzalloc(sizeof(*code) * FETCH_INSN_MAX, GFP_KERNEL);
- if (!code)
- return -ENOMEM;
- code[FETCH_INSN_MAX - 1].op = FETCH_OP_END;
- ret = parse_probe_arg(arg, parg->type, &code, &code[FETCH_INSN_MAX - 1],
- flags);
- if (ret)
- goto fail;
- /* Store operation */
- if (!strcmp(parg->type->name, "string")) {
- if (code->op != FETCH_OP_DEREF && code->op != FETCH_OP_IMM &&
- code->op != FETCH_OP_COMM) {
- pr_info("string only accepts memory or address.\n");
- ret = -EINVAL;
- goto fail;
- }
- if (code->op != FETCH_OP_DEREF || parg->count) {
- /*
- * IMM and COMM is pointing actual address, those must
- * be kept, and if parg->count != 0, this is an array
- * of string pointers instead of string address itself.
- */
- code++;
- if (code->op != FETCH_OP_NOP) {
- ret = -E2BIG;
- goto fail;
- }
- }
- code->op = FETCH_OP_ST_STRING; /* In DEREF case, replace it */
- code->size = parg->type->size;
- parg->dynamic = true;
- } else if (code->op == FETCH_OP_DEREF) {
- code->op = FETCH_OP_ST_MEM;
- code->size = parg->type->size;
- } else {
- code++;
- if (code->op != FETCH_OP_NOP) {
- ret = -E2BIG;
- goto fail;
- }
- code->op = FETCH_OP_ST_RAW;
- code->size = parg->type->size;
- }
- scode = code;
- /* Modify operation */
- if (t != NULL) {
- ret = __parse_bitfield_probe_arg(t, parg->type, &code);
- if (ret)
- goto fail;
- }
- /* Loop(Array) operation */
- if (parg->count) {
- if (scode->op != FETCH_OP_ST_MEM &&
- scode->op != FETCH_OP_ST_STRING) {
- pr_info("array only accepts memory or address\n");
- ret = -EINVAL;
- goto fail;
- }
- code++;
- if (code->op != FETCH_OP_NOP) {
- ret = -E2BIG;
- goto fail;
- }
- code->op = FETCH_OP_LP_ARRAY;
- code->param = parg->count;
- }
- code++;
- code->op = FETCH_OP_END;
- /* Shrink down the code buffer */
- parg->code = kzalloc(sizeof(*code) * (code - tmp + 1), GFP_KERNEL);
- if (!parg->code)
- ret = -ENOMEM;
- else
- memcpy(parg->code, tmp, sizeof(*code) * (code - tmp + 1));
- fail:
- if (ret) {
- for (code = tmp; code < tmp + FETCH_INSN_MAX; code++)
- if (code->op == FETCH_NOP_SYMBOL)
- kfree(code->data);
- }
- kfree(tmp);
- return ret;
- }
- /* Return 1 if name is reserved or already used by another argument */
- int traceprobe_conflict_field_name(const char *name,
- struct probe_arg *args, int narg)
- {
- int i;
- for (i = 0; i < ARRAY_SIZE(reserved_field_names); i++)
- if (strcmp(reserved_field_names[i], name) == 0)
- return 1;
- for (i = 0; i < narg; i++)
- if (strcmp(args[i].name, name) == 0)
- return 1;
- return 0;
- }
- void traceprobe_free_probe_arg(struct probe_arg *arg)
- {
- struct fetch_insn *code = arg->code;
- while (code && code->op != FETCH_OP_END) {
- if (code->op == FETCH_NOP_SYMBOL)
- kfree(code->data);
- code++;
- }
- kfree(arg->code);
- kfree(arg->name);
- kfree(arg->comm);
- kfree(arg->fmt);
- }
- int traceprobe_update_arg(struct probe_arg *arg)
- {
- struct fetch_insn *code = arg->code;
- long offset;
- char *tmp;
- char c;
- int ret = 0;
- while (code && code->op != FETCH_OP_END) {
- if (code->op == FETCH_NOP_SYMBOL) {
- if (code[1].op != FETCH_OP_IMM)
- return -EINVAL;
- tmp = strpbrk("+-", code->data);
- if (tmp)
- c = *tmp;
- ret = traceprobe_split_symbol_offset(code->data,
- &offset);
- if (ret)
- return ret;
- code[1].immediate =
- (unsigned long)kallsyms_lookup_name(code->data);
- if (tmp)
- *tmp = c;
- if (!code[1].immediate)
- return -ENOENT;
- code[1].immediate += offset;
- }
- code++;
- }
- return 0;
- }
- /* When len=0, we just calculate the needed length */
- #define LEN_OR_ZERO (len ? len - pos : 0)
- static int __set_print_fmt(struct trace_probe *tp, char *buf, int len,
- bool is_return)
- {
- struct probe_arg *parg;
- int i, j;
- int pos = 0;
- const char *fmt, *arg;
- if (!is_return) {
- fmt = "(%lx)";
- arg = "REC->" FIELD_STRING_IP;
- } else {
- fmt = "(%lx <- %lx)";
- arg = "REC->" FIELD_STRING_FUNC ", REC->" FIELD_STRING_RETIP;
- }
- pos += snprintf(buf + pos, LEN_OR_ZERO, "\"%s", fmt);
- for (i = 0; i < tp->nr_args; i++) {
- parg = tp->args + i;
- pos += snprintf(buf + pos, LEN_OR_ZERO, " %s=", parg->name);
- if (parg->count) {
- pos += snprintf(buf + pos, LEN_OR_ZERO, "{%s",
- parg->type->fmt);
- for (j = 1; j < parg->count; j++)
- pos += snprintf(buf + pos, LEN_OR_ZERO, ",%s",
- parg->type->fmt);
- pos += snprintf(buf + pos, LEN_OR_ZERO, "}");
- } else
- pos += snprintf(buf + pos, LEN_OR_ZERO, "%s",
- parg->type->fmt);
- }
- pos += snprintf(buf + pos, LEN_OR_ZERO, "\", %s", arg);
- for (i = 0; i < tp->nr_args; i++) {
- parg = tp->args + i;
- if (parg->count) {
- if (strcmp(parg->type->name, "string") == 0)
- fmt = ", __get_str(%s[%d])";
- else
- fmt = ", REC->%s[%d]";
- for (j = 0; j < parg->count; j++)
- pos += snprintf(buf + pos, LEN_OR_ZERO,
- fmt, parg->name, j);
- } else {
- if (strcmp(parg->type->name, "string") == 0)
- fmt = ", __get_str(%s)";
- else
- fmt = ", REC->%s";
- pos += snprintf(buf + pos, LEN_OR_ZERO,
- fmt, parg->name);
- }
- }
- /* return the length of print_fmt */
- return pos;
- }
- #undef LEN_OR_ZERO
- int traceprobe_set_print_fmt(struct trace_probe *tp, bool is_return)
- {
- int len;
- char *print_fmt;
- /* First: called with 0 length to calculate the needed length */
- len = __set_print_fmt(tp, NULL, 0, is_return);
- print_fmt = kmalloc(len + 1, GFP_KERNEL);
- if (!print_fmt)
- return -ENOMEM;
- /* Second: actually write the @print_fmt */
- __set_print_fmt(tp, print_fmt, len + 1, is_return);
- tp->call.print_fmt = print_fmt;
- return 0;
- }
- int traceprobe_define_arg_fields(struct trace_event_call *event_call,
- size_t offset, struct trace_probe *tp)
- {
- int ret, i;
- /* Set argument names as fields */
- for (i = 0; i < tp->nr_args; i++) {
- struct probe_arg *parg = &tp->args[i];
- const char *fmt = parg->type->fmttype;
- int size = parg->type->size;
- if (parg->fmt)
- fmt = parg->fmt;
- if (parg->count)
- size *= parg->count;
- ret = trace_define_field(event_call, fmt, parg->name,
- offset + parg->offset, size,
- parg->type->is_signed,
- FILTER_OTHER);
- if (ret)
- return ret;
- }
- return 0;
- }
|