[lttng-dev] [PATCH lttng-modules v6 4/5] Extract the payload for epoll_ctl

Julien Desfossez jdesfossez at efficios.com
Sat Apr 30 14:54:53 UTC 2016


On 30-Apr-2016 02:19:24 PM, Mathieu Desnoyers wrote:
> ----- On Apr 30, 2016, at 10:17 AM, Mathieu Desnoyers mathieu.desnoyers at efficios.com wrote:
> 
> > ----- On Apr 29, 2016, at 6:53 PM, Julien Desfossez jdesfossez at efficios.com
> > wrote:
> > 
> >> Map the operation to its name (EPOLL_CTL_*), extract the standard event
> >> flags (EPOLL*) and output the data in two different formats: FD as an
> >> int in decimal, and u64 in hex. The less standard event flags are not
> >> extracted yet, but we extract the raw value in hex for more advanced
> >> analyses.
> >> 
> >> Here is an example output:
> >> syscall_entry_epoll_ctl: {
> >>  epfd = 4, op_enum = ( "EPOLL_CTL_ADD" : container = 1 ),
> >>  fd = 0, event = { raw_events = 0x80000003,
> >>  events = { EPOLLIN = 1, EPOLLPRI = 1, EPOLLOUT = 0, EPOLLERR = 0,
> >>    padding = 0 },
> >>  data_union = { u64 = 0x0, fd = 0 } }
> >> }
> >> 
> >> syscall_exit_epoll_ctl: { ret = 0 }
> >> 
> >> Signed-off-by: Julien Desfossez <jdesfossez at efficios.com>
> >> ---
> >> .../syscalls/headers/syscalls_pointers_override.h  | 140 +++++++++++++++++++++
> >> 1 file changed, 140 insertions(+)
> >> 
> >> diff --git a/instrumentation/syscalls/headers/syscalls_pointers_override.h
> >> b/instrumentation/syscalls/headers/syscalls_pointers_override.h
> >> index 11d5f3d..59931df 100644
> >> --- a/instrumentation/syscalls/headers/syscalls_pointers_override.h
> >> +++ b/instrumentation/syscalls/headers/syscalls_pointers_override.h
> >> @@ -555,4 +555,144 @@ SC_LTTNG_TRACEPOINT_EVENT_CODE(ppoll,
> >> )
> >> #endif /* defined(CONFIG_X86_32) || defined(CONFIG_X86_64) ||
> >> defined(CONFIG_ARM64) || defined(CONFIG_ARM) */
> >> 
> >> +#include <linux/eventpoll.h>
> >> +
> >> +SC_LTTNG_TRACEPOINT_ENUM(lttng_epoll_op,
> >> +	TP_ENUM_VALUES(
> >> +		ctf_enum_value("EPOLL_CTL_ADD", EPOLL_CTL_ADD)
> >> +		ctf_enum_value("EPOLL_CTL_DEL", EPOLL_CTL_DEL)
> >> +		ctf_enum_value("EPOLL_CTL_MOD", EPOLL_CTL_MOD)
> >> +	)
> >> +)
> >> +
> >> +#ifndef ONCE_LTTNG_TRACE_EPOLL_CTL_H
> >> +#define ONCE_LTTNG_TRACE_EPOLL_CTL_H
> >> +
> >> +#define LTTNG_EPOLL_NRFLAGS (POLLHUP + 1)
> >> +#define EPOLL_FLAGS_PADDING_SIZE (sizeof(uint8_t) * BITS_PER_BYTE) - \
> >> +	ilog2(LTTNG_EPOLL_NRFLAGS - 1)
> >> +
> >> +/*
> >> + * Only extract the values specified by iBCS2 for now.
> >> + */
> >> +static struct lttng_event_field lttng_epoll_ctl_events_fields[] = {
> >> +	/* 0x0001 */
> >> +	[ilog2(POLLIN)] = {
> >> +		.name = "EPOLLIN",
> >> +		.type = __type_integer(int, 1, 1, 0, __LITTLE_ENDIAN, 10, none),
> >> +	},
> >> +	/* 0x0002 */
> >> +	[ilog2(POLLPRI)] = {
> >> +		.name = "EPOLLPRI",
> >> +		.type = __type_integer(int, 1, 1, 0, __LITTLE_ENDIAN, 10, none),
> >> +	},
> >> +	/* 0x0004 */
> >> +	[ilog2(POLLOUT)] = {
> >> +		.name = "EPOLLOUT",
> >> +		.type = __type_integer(int, 1, 1, 0, __LITTLE_ENDIAN, 10, none),
> >> +	},
> >> +	/* 0x0008 */
> >> +	[ilog2(POLLERR)] = {
> >> +		.name = "EPOLLERR",
> >> +		.type = __type_integer(int, 1, 1, 0, __LITTLE_ENDIAN, 10, none),
> >> +	},
> >> +	/* 0x0010 */
> >> +	[ilog2(POLLHUP)] = {
> >> +		.name = "EPOLLHUP",
> >> +		.type = __type_integer(int, 1, 1, 0, __LITTLE_ENDIAN, 10, none),
> >> +	},
> >> +	[ilog2(LTTNG_EPOLL_NRFLAGS)] = {
> >> +		.name = "padding",
> >> +		.type = __type_integer(int, EPOLL_FLAGS_PADDING_SIZE, 1, 0,
> >> +				__LITTLE_ENDIAN, 10, none),
> >> +	},
> >> +
> >> +};
> >> +
> >> +static struct lttng_event_field lttng_epoll_data_fields[] = {
> >> +	[0] = {
> >> +		.name = "u64",
> >> +		.type = __type_integer(uint64_t, 0, 0, 0, __BYTE_ORDER, 16, none),
> >> +	},
> >> +	[1] = {
> >> +		.name = "fd",
> >> +		.type = __type_integer(int, 0, 0, 0, __BYTE_ORDER, 10, none),
> >> +	},
> >> +};
> >> +
> >> +static struct lttng_event_field epoll_ctl_fields[] = {
> >> +	[0] = {
> >> +		.name = "data_union",
> >> +		.type = {
> >> +			.atype = atype_struct,
> >> +			.u._struct.nr_fields = ARRAY_SIZE(lttng_epoll_data_fields),
> >> +			.u._struct.fields = lttng_epoll_data_fields,
> >> +		}
> >> +	},
> >> +	[1] = {
> >> +		.name = "raw_events",
> >> +		.type = __type_integer(uint32_t, 0, 0, 0, __BYTE_ORDER, 16, none),
> >> +	},
> >> +	[2] = {
> >> +		.name = "events",
> >> +		.type = {
> >> +			.atype = atype_struct,
> >> +			.u._struct.nr_fields = ARRAY_SIZE(lttng_epoll_ctl_events_fields),
> >> +			.u._struct.fields = lttng_epoll_ctl_events_fields,
> >> +		}
> >> +	},
> >> +};
> >> +#endif /* ONCE_LTTNG_TRACE_EPOLL_CTL_H */
> >> +
> >> +#if defined(CONFIG_X86_32) || defined(CONFIG_X86_64) || defined(CONFIG_ARM64)
> >> || defined(CONFIG_ARM)
> >> +#define OVERRIDE_32_epoll_ctl
> >> +#define OVERRIDE_64_epoll_ctl
> >> +SC_LTTNG_TRACEPOINT_EVENT_CODE(epoll_ctl,
> >> +	TP_PROTO(sc_exit(long ret,) int epfd, int op, int fd,
> >> +		struct epoll_event __user * uevent),
> >> +	TP_ARGS(sc_exit(ret,) epfd, op, fd, uevent),
> >> +	TP_locvar(
> >> +		struct epoll_event event;
> >> +		int err;
> >> +	),
> >> +	TP_code_pre(
> >> +		tp_locvar->err = lib_ring_buffer_copy_from_user_check_nofault(
> >> +			&tp_locvar->event, uevent, sizeof(struct epoll_event));
> >> +		),
> >> +	TP_FIELDS(
> >> +		sc_exit(ctf_integer(long, ret, ret))
> >> +		sc_in(ctf_integer(int, epfd, epfd))
> >> +		sc_in(ctf_enum(lttng_epoll_op, int, op_enum, op))
> >> +		sc_in(ctf_integer(int, fd, fd))
> >> +		sc_in(
> >> +			ctf_custom_field(
> >> +				ctf_custom_type(
> >> +					.atype = atype_struct,
> >> +					.u._struct.nr_fields = ARRAY_SIZE(epoll_ctl_fields),
> >> +					.u._struct.fields = epoll_ctl_fields,
> >> +				),
> >> +				event,
> >> +				ctf_custom_code(
> >> +					ctf_align(uint64_t)
> >> +					if (!tp_locvar->err) {
> >> +						ctf_integer_type(uint64_t, tp_locvar->event.data)
> >> +						ctf_integer_type(int, tp_locvar->event.data)
> >> +						ctf_integer_bitfield_type(uint32_t,
> >> +							cpu_to_le32(tp_locvar->event.events))
> > 
> > not sure why we reverse the endianness here ? It does not
> > match the "__BYTE_ORDER" type description, and differs from
> > the way raw_events fields is handled in the poll/ppoll instrumentation.
> 
> also, why use a 32-bit integer rather than 16-bit as in poll/ppoll here ?

Poll uses a short for the "events" field, but this field in struct
epoll_event is a uint32_t, so we have to use the full 32 bits for the
raw value in epoll_ctl and epoll_wait.

Julien


More information about the lttng-dev mailing list