[PATCH v3 07/10] pert tools: Add queue management functionality
Robert Walker
robert.walker at arm.com
Mon Jan 22 09:25:07 PST 2018
On 01/17/2018 05:52 PM, Mathieu Poirier wrote:
> Add functionatlity to setup trace queues so that traces associated with
> CoreSight auxtrace events found in the perf.data file can be classified
> properly. The decoder and memory callback associated with each queue are
> then used to decode the traces that have been assigned to that queue.
>
> Co-authored-by: Tor Jeremiassen <tor at ti.com>
> Signed-off-by: Mathieu Poirier <mathieu.poirier at linaro.org>
> ---
> tools/perf/util/cs-etm.c | 208 ++++++++++++++++++++++++++++++++++++++++++++++-
> 1 file changed, 204 insertions(+), 4 deletions(-)
>
> diff --git a/tools/perf/util/cs-etm.c b/tools/perf/util/cs-etm.c
> index cad429ce3c00..83eb676274b5 100644
> --- a/tools/perf/util/cs-etm.c
> +++ b/tools/perf/util/cs-etm.c
> @@ -196,15 +196,215 @@ static void cs_etm__free(struct perf_session *session)
> zfree(&aux);
> }
>
> +static u32 cs_etm__mem_access(struct cs_etm_queue *etmq, u64 address,
> + size_t size, u8 *buffer)
> +{
> + u8 cpumode;
> + u64 offset;
> + int len;
> + struct thread *thread;
> + struct machine *machine;
> + struct addr_location al;
> +
> + if (!etmq)
> + return -1;
> +
> + machine = etmq->etm->machine;
> + if (address >= etmq->etm->kernel_start)
> + cpumode = PERF_RECORD_MISC_KERNEL;
> + else
> + cpumode = PERF_RECORD_MISC_USER;
> +
> + thread = etmq->thread;
> + if (!thread) {
> + if (cpumode != PERF_RECORD_MISC_KERNEL)
> + return -EINVAL;
> + thread = etmq->etm->unknown_thread;
> + }
> +
> + thread__find_addr_map(thread, cpumode, MAP__FUNCTION, address, &al);
> +
> + if (!al.map || !al.map->dso)
> + return 0;
> +
> + if (al.map->dso->data.status == DSO_DATA_STATUS_ERROR &&
> + dso__data_status_seen(al.map->dso, DSO_DATA_STATUS_SEEN_ITRACE))
> + return 0;
> +
> + offset = al.map->map_ip(al.map, address);
> +
> + map__load(al.map);
> +
> + len = dso__data_read_offset(al.map->dso, machine, offset, buffer, size);
> +
> + if (len <= 0)
> + return 0;
> +
> + return len;
> +}
> +
> +static struct cs_etm_queue *cs_etm__alloc_queue(struct cs_etm_auxtrace *etm,
> + unsigned int queue_nr)
> +{
> + int i;
> + struct cs_etm_decoder_params d_params;
> + struct cs_etm_trace_params *t_params;
> + struct cs_etm_queue *etmq;
> +
> + etmq = zalloc(sizeof(*etmq));
> + if (!etmq)
> + return NULL;
> +
> + etmq->event_buf = malloc(PERF_SAMPLE_MAX_SIZE);
Should this and the other members of etmq alloc'd in this function be
free'd in cs_etm__free_queue() as they were in the original version at
https://github.com/Linaro/perf-opencsd/ ? I can't see them getting
freed anywhere else.
> + if (!etmq->event_buf)
> + goto out_free;
> +
> + etmq->etm = etm;
> + etmq->queue_nr = queue_nr;
> + etmq->pid = -1;
> + etmq->tid = -1;
> + etmq->cpu = -1;
> +
> + /* Use metadata to fill in trace parameters for trace decoder */
> + t_params = zalloc(sizeof(*t_params) * etm->num_cpu);
> +
> + if (!t_params)
> + goto out_free;
> +
> + for (i = 0; i < etm->num_cpu; i++) {
> + t_params[i].protocol = CS_ETM_PROTO_ETMV4i;
> + t_params[i].etmv4.reg_idr0 = etm->metadata[i][CS_ETMV4_TRCIDR0];
> + t_params[i].etmv4.reg_idr1 = etm->metadata[i][CS_ETMV4_TRCIDR1];
> + t_params[i].etmv4.reg_idr2 = etm->metadata[i][CS_ETMV4_TRCIDR2];
> + t_params[i].etmv4.reg_idr8 = etm->metadata[i][CS_ETMV4_TRCIDR8];
> + t_params[i].etmv4.reg_configr =
> + etm->metadata[i][CS_ETMV4_TRCCONFIGR];
> + t_params[i].etmv4.reg_traceidr =
> + etm->metadata[i][CS_ETMV4_TRCTRACEIDR];
> + }
> +
> + /* Set decoder parameters to simply print the trace packets */
> + d_params.packet_printer = cs_etm__packet_dump;
> + d_params.operation = CS_ETM_OPERATION_DECODE;
> + d_params.formatted = true;
> + d_params.fsyncs = false;
> + d_params.hsyncs = false;
> + d_params.frame_aligned = true;
> + d_params.data = etmq;
> +
> + etmq->decoder = cs_etm_decoder__new(etm->num_cpu, &d_params, t_params);
> +
> + zfree(&t_params);
> +
> + if (!etmq->decoder)
> + goto out_free;
> +
> + /*
> + * Register a function to handle all memory accesses required by
> + * the trace decoder library.
> + */
> + if (cs_etm_decoder__add_mem_access_cb(etmq->decoder,
> + 0x0L, ((u64) -1L),
> + cs_etm__mem_access))
> + goto out_free_decoder;
> +
> + etmq->offset = 0;
> +
> + return etmq;
> +
> +out_free_decoder:
> + cs_etm_decoder__free(etmq->decoder);
> +out_free:
> + zfree(&etmq->event_buf);
> + free(etmq);
> +
> + return NULL;
> +}
> +
> +static int cs_etm__setup_queue(struct cs_etm_auxtrace *etm,
> + struct auxtrace_queue *queue,
> + unsigned int queue_nr)
> +{
> + struct cs_etm_queue *etmq = queue->priv;
> +
> + if (list_empty(&queue->head) || etmq)
> + return 0;
> +
> + etmq = cs_etm__alloc_queue(etm, queue_nr);
> +
> + if (!etmq)
> + return -ENOMEM;
> +
> + queue->priv = etmq;
> +
> + if (queue->cpu != -1)
> + etmq->cpu = queue->cpu;
> +
> + etmq->tid = queue->tid;
> +
> + return 0;
> +}
> +
> +static int cs_etm__setup_queues(struct cs_etm_auxtrace *etm)
> +{
> + unsigned int i;
> + int ret;
> +
> + for (i = 0; i < etm->queues.nr_queues; i++) {
> + ret = cs_etm__setup_queue(etm, &etm->queues.queue_array[i], i);
> + if (ret)
> + return ret;
> + }
> +
> + return 0;
> +}
> +
> +static int cs_etm__update_queues(struct cs_etm_auxtrace *etm)
> +{
> + if (etm->queues.new_data) {
> + etm->queues.new_data = false;
> + return cs_etm__setup_queues(etm);
> + }
> +
> + return 0;
> +}
> +
> static int cs_etm__process_event(struct perf_session *session,
> union perf_event *event,
> struct perf_sample *sample,
> struct perf_tool *tool)
> {
> - (void) session;
> - (void) event;
> - (void) sample;
> - (void) tool;
> + int err = 0;
> + u64 timestamp;
> + struct cs_etm_auxtrace *etm = container_of(session->auxtrace,
> + struct cs_etm_auxtrace,
> + auxtrace);
> +
> + /* Keep compiler happy */
> + (void)event;
> +
> + if (dump_trace)
> + return 0;
> +
> + if (!tool->ordered_events) {
> + pr_err("CoreSight ETM Trace requires ordered events\n");
> + return -EINVAL;
> + }
> +
> + if (!etm->timeless_decoding)
> + return -EINVAL;
> +
> + if (sample->time && (sample->time != (u64) -1))
> + timestamp = sample->time;
> + else
> + timestamp = 0;
> +
> + if (timestamp || etm->timeless_decoding) {
> + err = cs_etm__update_queues(etm);
> + if (err)
> + return err;
> + }
> +
> return 0;
> }
>
>
More information about the linux-arm-kernel
mailing list