Em Thu, Dec 18, 2014 at 08:10:43AM -0700, David Ahern escreveu: > Adds helper for following kernel formats: > %pi4 print an IPv4 address with leading zeros > %pI4 print an IPv4 address without leading zeros > %pi6 print an IPv6 address without colons > %pI6 print an IPv6 address with colons > %pI6c print an IPv6 address with colons > %pISpc print an IP address from a sockaddr > > Allows these formats to be used in tracepoints. > > Quite a bit of this is adapted from code in lib/vsprintf.c.
Can't we try as much as possible use that code directly? Something like we do for lib/rbtree.c in tools/perf/, or like I did recently with tools/lib/util/find_next_bit.c, i.e. retain at least this kind of sharing: diff -u tools/lib/util/find_next_bit.c lib/find_next_bit.c Should show that only bits that we have not a need at the moment in tools/ were left behind. We want to either get fixes for free by directly reusing or notice it easily, using diff. - Arnaldo > v2: > - pass ptr+1 to print_ip_arg per Namhyung's comments > - added field length checks to sockaddr function > > Signed-off-by: David Ahern <dsah...@gmail.com> > Cc: Namhyung Kim <namhy...@kernel.org> > Cc: Jiri Olsa <jo...@kernel.org> > Cc: Steven Rostedt <rost...@goodmis.org> > --- > tools/lib/traceevent/event-parse.c | 326 > +++++++++++++++++++++++++++++++++++++ > 1 file changed, 326 insertions(+) > > diff --git a/tools/lib/traceevent/event-parse.c > b/tools/lib/traceevent/event-parse.c > index cf3a44bf1ec3..48d57c9fc476 100644 > --- a/tools/lib/traceevent/event-parse.c > +++ b/tools/lib/traceevent/event-parse.c > @@ -32,6 +32,7 @@ > #include <stdint.h> > #include <limits.h> > > +#include <netinet/ip6.h> > #include "event-parse.h" > #include "event-utils.h" > > @@ -4149,6 +4150,322 @@ static void print_mac_arg(struct trace_seq *s, int > mac, void *data, int size, > trace_seq_printf(s, fmt, buf[0], buf[1], buf[2], buf[3], buf[4], > buf[5]); > } > > +static void print_ip4_addr(struct trace_seq *s, char i, unsigned char *buf) > +{ > + const char *fmt; > + > + if (i == 'i') > + fmt = "%03d.%03d.%03d.%03d"; > + else > + fmt = "%d.%d.%d.%d"; > + > + trace_seq_printf(s, fmt, buf[0], buf[1], buf[2], buf[3]); > +} > + > +static inline bool ipv6_addr_v4mapped(const struct in6_addr *a) > +{ > + return ((unsigned long)(a->s6_addr32[0] | a->s6_addr32[1]) | > + (unsigned long)(a->s6_addr32[2] ^ htonl(0x0000ffff))) == 0UL; > +} > + > +static inline bool ipv6_addr_is_isatap(const struct in6_addr *addr) > +{ > + return (addr->s6_addr32[2] | htonl(0x02000000)) == htonl(0x02005EFE); > +} > + > +static void print_ip6c_addr(struct trace_seq *s, unsigned char *addr) > +{ > + int i, j, range; > + unsigned char zerolength[8]; > + int longest = 1; > + int colonpos = -1; > + uint16_t word; > + uint8_t hi, lo; > + bool needcolon = false; > + bool useIPv4; > + struct in6_addr in6; > + > + memcpy(&in6, addr, sizeof(struct in6_addr)); > + > + useIPv4 = ipv6_addr_v4mapped(&in6) || ipv6_addr_is_isatap(&in6); > + > + memset(zerolength, 0, sizeof(zerolength)); > + > + if (useIPv4) > + range = 6; > + else > + range = 8; > + > + /* find position of longest 0 run */ > + for (i = 0; i < range; i++) { > + for (j = i; j < range; j++) { > + if (in6.s6_addr16[j] != 0) > + break; > + zerolength[i]++; > + } > + } > + for (i = 0; i < range; i++) { > + if (zerolength[i] > longest) { > + longest = zerolength[i]; > + colonpos = i; > + } > + } > + if (longest == 1) /* don't compress a single 0 */ > + colonpos = -1; > + > + /* emit address */ > + for (i = 0; i < range; i++) { > + if (i == colonpos) { > + if (needcolon || i == 0) > + trace_seq_printf(s, ":"); > + trace_seq_printf(s, ":"); > + needcolon = false; > + i += longest - 1; > + continue; > + } > + if (needcolon) { > + trace_seq_printf(s, ":"); > + needcolon = false; > + } > + /* hex u16 without leading 0s */ > + word = ntohs(in6.s6_addr16[i]); > + hi = word >> 8; > + lo = word & 0xff; > + if (hi) > + trace_seq_printf(s, "%x%02x", hi, lo); > + else > + trace_seq_printf(s, "%x", lo); > + > + needcolon = true; > + } > + > + if (useIPv4) { > + if (needcolon) > + trace_seq_printf(s, ":"); > + print_ip4_addr(s, 'I', &in6.s6_addr[12]); > + } > + > + return; > +} > + > +static void print_ip6_addr(struct trace_seq *s, char i, unsigned char *buf) > +{ > + int j; > + > + for (j = 0; j < 16; j += 2) { > + trace_seq_printf(s, "%02x%02x", buf[j], buf[j+1]); > + if (i == 'I' && j < 14) > + trace_seq_printf(s, ":"); > + } > +} > + > +/* > + * %pi4 print an IPv4 address with leading zeros > + * %pI4 print an IPv4 address without leading zeros > + * %pi6 print an IPv6 address without colons > + * %pI6 print an IPv6 address with colons > + * %pI6c print an IPv6 address in compressed form with colons > + * %pISpc print an IP address based on sockaddr; p adds port. > + */ > +static int print_ipv4_arg(struct trace_seq *s, const char *ptr, char i, > + void *data, int size, struct event_format *event, > + struct print_arg *arg) > +{ > + unsigned char *buf; > + > + if (arg->type == PRINT_FUNC) { > + process_defined_func(s, data, size, event, arg); > + return 0; > + } > + > + if (arg->type != PRINT_FIELD) { > + trace_seq_printf(s, "ARG TYPE NOT FIELD BUT %d", arg->type); > + return 0; > + } > + > + if (!arg->field.field) { > + arg->field.field = > + pevent_find_any_field(event, arg->field.name); > + if (!arg->field.field) { > + do_warning("%s: field %s not found", > + __func__, arg->field.name); > + return 0; > + } > + } > + > + buf = data + arg->field.field->offset; > + > + if (arg->field.field->size != 4) { > + trace_seq_printf(s, "INVALIDIPv4"); > + return 0; > + } > + print_ip4_addr(s, i, buf); > + > + return 0; > +} > + > +static int print_ipv6_arg(struct trace_seq *s, const char *ptr, char i, > + void *data, int size, struct event_format *event, > + struct print_arg *arg) > +{ > + char have_c = 0; > + unsigned char *buf; > + int rc = 0; > + > + /* pI6c */ > + if (*ptr == 'c') { > + have_c = 1; > + ptr++; > + rc++; > + } > + > + if (arg->type == PRINT_FUNC) { > + process_defined_func(s, data, size, event, arg); > + return rc; > + } > + > + if (arg->type != PRINT_FIELD) { > + trace_seq_printf(s, "ARG TYPE NOT FIELD BUT %d", arg->type); > + return rc; > + } > + > + if (!arg->field.field) { > + arg->field.field = > + pevent_find_any_field(event, arg->field.name); > + if (!arg->field.field) { > + do_warning("%s: field %s not found", > + __func__, arg->field.name); > + return rc; > + } > + } > + > + buf = data + arg->field.field->offset; > + > + if (arg->field.field->size != 16) { > + trace_seq_printf(s, "INVALIDIPv6"); > + return rc; > + } > + > + if (have_c) > + print_ip6c_addr(s, buf); > + else > + print_ip6_addr(s, i, buf); > + > + return rc; > +} > + > +static int print_ipsa_arg(struct trace_seq *s, const char *ptr, char i, > + void *data, int size, struct event_format *event, > + struct print_arg *arg) > +{ > + char have_c = 0, have_p = 0; > + unsigned char *buf; > + struct sockaddr_storage *sa; > + int rc = 0; > + > + /* pISpc */ > + if (*ptr == 'p') { > + have_p = 1; > + ptr++; > + rc++; > + } > + if (*ptr == 'c') { > + have_c = 1; > + ptr++; > + rc++; > + } > + > + if (arg->type == PRINT_FUNC) { > + process_defined_func(s, data, size, event, arg); > + return rc; > + } > + > + if (arg->type != PRINT_FIELD) { > + trace_seq_printf(s, "ARG TYPE NOT FIELD BUT %d", arg->type); > + return rc; > + } > + > + if (!arg->field.field) { > + arg->field.field = > + pevent_find_any_field(event, arg->field.name); > + if (!arg->field.field) { > + do_warning("%s: field %s not found", > + __func__, arg->field.name); > + return rc; > + } > + } > + > + sa = (struct sockaddr_storage *) (data + arg->field.field->offset); > + > + if (sa->ss_family == AF_INET) { > + struct sockaddr_in *sa4 = (struct sockaddr_in *) sa; > + > + if (arg->field.field->size < sizeof(struct sockaddr_in)) { > + trace_seq_printf(s, "INVALIDIPv4"); > + return rc; > + } > + > + print_ip4_addr(s, i, (unsigned char *) &sa4->sin_addr); > + if (have_p) > + trace_seq_printf(s, ":%d", ntohs(sa4->sin_port)); > + > + > + } else if (sa->ss_family == AF_INET6) { > + struct sockaddr_in6 *sa6 = (struct sockaddr_in6 *) sa; > + > + if (arg->field.field->size < sizeof(struct sockaddr_in6)) { > + trace_seq_printf(s, "INVALIDIPv6"); > + return rc; > + } > + > + if (have_p) > + trace_seq_printf(s, "["); > + > + buf = (unsigned char *) &sa6->sin6_addr; > + if (have_c) > + print_ip6c_addr(s, buf); > + else > + print_ip6_addr(s, i, buf); > + > + if (have_p) > + trace_seq_printf(s, "]:%d", ntohs(sa6->sin6_port)); > + } > + > + return rc; > +} > + > +static int print_ip_arg(struct trace_seq *s, const char *ptr, > + void *data, int size, struct event_format *event, > + struct print_arg *arg) > +{ > + char i = *ptr; /* 'i' or 'I' */ > + char ver; > + int rc = 0; > + > + ptr++; > + rc++; > + > + ver = *ptr; > + ptr++; > + rc++; > + > + switch (ver) { > + case '4': > + rc += print_ipv4_arg(s, ptr, i, data, size, event, arg); > + break; > + case '6': > + rc += print_ipv6_arg(s, ptr, i, data, size, event, arg); > + break; > + case 'S': > + rc += print_ipsa_arg(s, ptr, i, data, size, event, arg); > + break; > + default: > + return 0; > + } > + > + return rc; > +} > + > static int is_printable_array(char *p, unsigned int len) > { > unsigned int i; > @@ -4337,6 +4654,15 @@ static void pretty_print(struct trace_seq *s, void > *data, int size, struct event > ptr++; > arg = arg->next; > break; > + } else if (*(ptr+1) == 'I' || *(ptr+1) == 'i') { > + int n; > + > + n = print_ip_arg(s, ptr+1, data, size, > event, arg); > + if (n > 0) { > + ptr += n; > + arg = arg->next; > + break; > + } > } > > /* fall through */ > -- > 1.9.3 (Apple Git-50) -- To unsubscribe from this list: send the line "unsubscribe linux-kernel" in the body of a message to majord...@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html Please read the FAQ at http://www.tux.org/lkml/