335 lines
8.2 KiB
C
335 lines
8.2 KiB
C
|
/*
|
||
|
* SPDX-License-Identifier: GPL-2.0
|
||
|
*
|
||
|
* Copyright(C) 2015-2018 Linaro Limited.
|
||
|
*
|
||
|
* Author: Tor Jeremiassen <tor@ti.com>
|
||
|
* Author: Mathieu Poirier <mathieu.poirier@linaro.org>
|
||
|
*/
|
||
|
|
||
|
#include <linux/err.h>
|
||
|
#include <linux/list.h>
|
||
|
#include <stdlib.h>
|
||
|
#include <opencsd/c_api/opencsd_c_api.h>
|
||
|
#include <opencsd/etmv4/trc_pkt_types_etmv4.h>
|
||
|
#include <opencsd/ocsd_if_types.h>
|
||
|
|
||
|
#include "cs-etm.h"
|
||
|
#include "cs-etm-decoder.h"
|
||
|
#include "intlist.h"
|
||
|
#include "util.h"
|
||
|
|
||
|
#define MAX_BUFFER 1024
|
||
|
|
||
|
/* use raw logging */
|
||
|
#ifdef CS_DEBUG_RAW
|
||
|
#define CS_LOG_RAW_FRAMES
|
||
|
#ifdef CS_RAW_PACKED
|
||
|
#define CS_RAW_DEBUG_FLAGS (OCSD_DFRMTR_UNPACKED_RAW_OUT | \
|
||
|
OCSD_DFRMTR_PACKED_RAW_OUT)
|
||
|
#else
|
||
|
#define CS_RAW_DEBUG_FLAGS (OCSD_DFRMTR_UNPACKED_RAW_OUT)
|
||
|
#endif
|
||
|
#endif
|
||
|
|
||
|
struct cs_etm_decoder {
|
||
|
void *data;
|
||
|
void (*packet_printer)(const char *msg);
|
||
|
bool trace_on;
|
||
|
dcd_tree_handle_t dcd_tree;
|
||
|
cs_etm_mem_cb_type mem_access;
|
||
|
ocsd_datapath_resp_t prev_return;
|
||
|
u32 packet_count;
|
||
|
u32 head;
|
||
|
u32 tail;
|
||
|
struct cs_etm_packet packet_buffer[MAX_BUFFER];
|
||
|
};
|
||
|
|
||
|
static void cs_etm_decoder__gen_etmv4_config(struct cs_etm_trace_params *params,
|
||
|
ocsd_etmv4_cfg *config)
|
||
|
{
|
||
|
config->reg_configr = params->etmv4.reg_configr;
|
||
|
config->reg_traceidr = params->etmv4.reg_traceidr;
|
||
|
config->reg_idr0 = params->etmv4.reg_idr0;
|
||
|
config->reg_idr1 = params->etmv4.reg_idr1;
|
||
|
config->reg_idr2 = params->etmv4.reg_idr2;
|
||
|
config->reg_idr8 = params->etmv4.reg_idr8;
|
||
|
config->reg_idr9 = 0;
|
||
|
config->reg_idr10 = 0;
|
||
|
config->reg_idr11 = 0;
|
||
|
config->reg_idr12 = 0;
|
||
|
config->reg_idr13 = 0;
|
||
|
config->arch_ver = ARCH_V8;
|
||
|
config->core_prof = profile_CortexA;
|
||
|
}
|
||
|
|
||
|
static void cs_etm_decoder__print_str_cb(const void *p_context,
|
||
|
const char *msg,
|
||
|
const int str_len)
|
||
|
{
|
||
|
if (p_context && str_len)
|
||
|
((struct cs_etm_decoder *)p_context)->packet_printer(msg);
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
cs_etm_decoder__init_def_logger_printing(struct cs_etm_decoder_params *d_params,
|
||
|
struct cs_etm_decoder *decoder)
|
||
|
{
|
||
|
int ret = 0;
|
||
|
|
||
|
if (d_params->packet_printer == NULL)
|
||
|
return -1;
|
||
|
|
||
|
decoder->packet_printer = d_params->packet_printer;
|
||
|
|
||
|
/*
|
||
|
* Set up a library default logger to process any printers
|
||
|
* (packet/raw frame) we add later.
|
||
|
*/
|
||
|
ret = ocsd_def_errlog_init(OCSD_ERR_SEV_ERROR, 1);
|
||
|
if (ret != 0)
|
||
|
return -1;
|
||
|
|
||
|
/* no stdout / err / file output */
|
||
|
ret = ocsd_def_errlog_config_output(C_API_MSGLOGOUT_FLG_NONE, NULL);
|
||
|
if (ret != 0)
|
||
|
return -1;
|
||
|
|
||
|
/*
|
||
|
* Set the string CB for the default logger, passes strings to
|
||
|
* perf print logger.
|
||
|
*/
|
||
|
ret = ocsd_def_errlog_set_strprint_cb(decoder->dcd_tree,
|
||
|
(void *)decoder,
|
||
|
cs_etm_decoder__print_str_cb);
|
||
|
if (ret != 0)
|
||
|
ret = -1;
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
#ifdef CS_LOG_RAW_FRAMES
|
||
|
static void
|
||
|
cs_etm_decoder__init_raw_frame_logging(struct cs_etm_decoder_params *d_params,
|
||
|
struct cs_etm_decoder *decoder)
|
||
|
{
|
||
|
/* Only log these during a --dump operation */
|
||
|
if (d_params->operation == CS_ETM_OPERATION_PRINT) {
|
||
|
/* set up a library default logger to process the
|
||
|
* raw frame printer we add later
|
||
|
*/
|
||
|
ocsd_def_errlog_init(OCSD_ERR_SEV_ERROR, 1);
|
||
|
|
||
|
/* no stdout / err / file output */
|
||
|
ocsd_def_errlog_config_output(C_API_MSGLOGOUT_FLG_NONE, NULL);
|
||
|
|
||
|
/* set the string CB for the default logger,
|
||
|
* passes strings to perf print logger.
|
||
|
*/
|
||
|
ocsd_def_errlog_set_strprint_cb(decoder->dcd_tree,
|
||
|
(void *)decoder,
|
||
|
cs_etm_decoder__print_str_cb);
|
||
|
|
||
|
/* use the built in library printer for the raw frames */
|
||
|
ocsd_dt_set_raw_frame_printer(decoder->dcd_tree,
|
||
|
CS_RAW_DEBUG_FLAGS);
|
||
|
}
|
||
|
}
|
||
|
#else
|
||
|
static void
|
||
|
cs_etm_decoder__init_raw_frame_logging(
|
||
|
struct cs_etm_decoder_params *d_params __maybe_unused,
|
||
|
struct cs_etm_decoder *decoder __maybe_unused)
|
||
|
{
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
static int cs_etm_decoder__create_packet_printer(struct cs_etm_decoder *decoder,
|
||
|
const char *decoder_name,
|
||
|
void *trace_config)
|
||
|
{
|
||
|
u8 csid;
|
||
|
|
||
|
if (ocsd_dt_create_decoder(decoder->dcd_tree, decoder_name,
|
||
|
OCSD_CREATE_FLG_PACKET_PROC,
|
||
|
trace_config, &csid))
|
||
|
return -1;
|
||
|
|
||
|
if (ocsd_dt_set_pkt_protocol_printer(decoder->dcd_tree, csid, 0))
|
||
|
return -1;
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
cs_etm_decoder__create_etm_packet_printer(struct cs_etm_trace_params *t_params,
|
||
|
struct cs_etm_decoder *decoder)
|
||
|
{
|
||
|
const char *decoder_name;
|
||
|
ocsd_etmv4_cfg trace_config_etmv4;
|
||
|
void *trace_config;
|
||
|
|
||
|
switch (t_params->protocol) {
|
||
|
case CS_ETM_PROTO_ETMV4i:
|
||
|
cs_etm_decoder__gen_etmv4_config(t_params, &trace_config_etmv4);
|
||
|
decoder_name = OCSD_BUILTIN_DCD_ETMV4I;
|
||
|
trace_config = &trace_config_etmv4;
|
||
|
break;
|
||
|
default:
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
return cs_etm_decoder__create_packet_printer(decoder,
|
||
|
decoder_name,
|
||
|
trace_config);
|
||
|
}
|
||
|
|
||
|
static void cs_etm_decoder__clear_buffer(struct cs_etm_decoder *decoder)
|
||
|
{
|
||
|
int i;
|
||
|
|
||
|
decoder->head = 0;
|
||
|
decoder->tail = 0;
|
||
|
decoder->packet_count = 0;
|
||
|
for (i = 0; i < MAX_BUFFER; i++) {
|
||
|
decoder->packet_buffer[i].start_addr = 0xdeadbeefdeadbeefUL;
|
||
|
decoder->packet_buffer[i].end_addr = 0xdeadbeefdeadbeefUL;
|
||
|
decoder->packet_buffer[i].exc = false;
|
||
|
decoder->packet_buffer[i].exc_ret = false;
|
||
|
decoder->packet_buffer[i].cpu = INT_MIN;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
cs_etm_decoder__create_etm_decoder(struct cs_etm_decoder_params *d_params,
|
||
|
struct cs_etm_trace_params *t_params,
|
||
|
struct cs_etm_decoder *decoder)
|
||
|
{
|
||
|
if (d_params->operation == CS_ETM_OPERATION_PRINT)
|
||
|
return cs_etm_decoder__create_etm_packet_printer(t_params,
|
||
|
decoder);
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
struct cs_etm_decoder *
|
||
|
cs_etm_decoder__new(int num_cpu, struct cs_etm_decoder_params *d_params,
|
||
|
struct cs_etm_trace_params t_params[])
|
||
|
{
|
||
|
struct cs_etm_decoder *decoder;
|
||
|
ocsd_dcd_tree_src_t format;
|
||
|
u32 flags;
|
||
|
int i, ret;
|
||
|
|
||
|
if ((!t_params) || (!d_params))
|
||
|
return NULL;
|
||
|
|
||
|
decoder = zalloc(sizeof(*decoder));
|
||
|
|
||
|
if (!decoder)
|
||
|
return NULL;
|
||
|
|
||
|
decoder->data = d_params->data;
|
||
|
decoder->prev_return = OCSD_RESP_CONT;
|
||
|
cs_etm_decoder__clear_buffer(decoder);
|
||
|
format = (d_params->formatted ? OCSD_TRC_SRC_FRAME_FORMATTED :
|
||
|
OCSD_TRC_SRC_SINGLE);
|
||
|
flags = 0;
|
||
|
flags |= (d_params->fsyncs ? OCSD_DFRMTR_HAS_FSYNCS : 0);
|
||
|
flags |= (d_params->hsyncs ? OCSD_DFRMTR_HAS_HSYNCS : 0);
|
||
|
flags |= (d_params->frame_aligned ? OCSD_DFRMTR_FRAME_MEM_ALIGN : 0);
|
||
|
|
||
|
/*
|
||
|
* Drivers may add barrier frames when used with perf, set up to
|
||
|
* handle this. Barriers const of FSYNC packet repeated 4 times.
|
||
|
*/
|
||
|
flags |= OCSD_DFRMTR_RESET_ON_4X_FSYNC;
|
||
|
|
||
|
/* Create decode tree for the data source */
|
||
|
decoder->dcd_tree = ocsd_create_dcd_tree(format, flags);
|
||
|
|
||
|
if (decoder->dcd_tree == 0)
|
||
|
goto err_free_decoder;
|
||
|
|
||
|
/* init library print logging support */
|
||
|
ret = cs_etm_decoder__init_def_logger_printing(d_params, decoder);
|
||
|
if (ret != 0)
|
||
|
goto err_free_decoder_tree;
|
||
|
|
||
|
/* init raw frame logging if required */
|
||
|
cs_etm_decoder__init_raw_frame_logging(d_params, decoder);
|
||
|
|
||
|
for (i = 0; i < num_cpu; i++) {
|
||
|
ret = cs_etm_decoder__create_etm_decoder(d_params,
|
||
|
&t_params[i],
|
||
|
decoder);
|
||
|
if (ret != 0)
|
||
|
goto err_free_decoder_tree;
|
||
|
}
|
||
|
|
||
|
return decoder;
|
||
|
|
||
|
err_free_decoder_tree:
|
||
|
ocsd_destroy_dcd_tree(decoder->dcd_tree);
|
||
|
err_free_decoder:
|
||
|
free(decoder);
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
int cs_etm_decoder__process_data_block(struct cs_etm_decoder *decoder,
|
||
|
u64 indx, const u8 *buf,
|
||
|
size_t len, size_t *consumed)
|
||
|
{
|
||
|
int ret = 0;
|
||
|
ocsd_datapath_resp_t cur = OCSD_RESP_CONT;
|
||
|
ocsd_datapath_resp_t prev_return = decoder->prev_return;
|
||
|
size_t processed = 0;
|
||
|
u32 count;
|
||
|
|
||
|
while (processed < len) {
|
||
|
if (OCSD_DATA_RESP_IS_WAIT(prev_return)) {
|
||
|
cur = ocsd_dt_process_data(decoder->dcd_tree,
|
||
|
OCSD_OP_FLUSH,
|
||
|
0,
|
||
|
0,
|
||
|
NULL,
|
||
|
NULL);
|
||
|
} else if (OCSD_DATA_RESP_IS_CONT(prev_return)) {
|
||
|
cur = ocsd_dt_process_data(decoder->dcd_tree,
|
||
|
OCSD_OP_DATA,
|
||
|
indx + processed,
|
||
|
len - processed,
|
||
|
&buf[processed],
|
||
|
&count);
|
||
|
processed += count;
|
||
|
} else {
|
||
|
ret = -EINVAL;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Return to the input code if the packet buffer is full.
|
||
|
* Flushing will get done once the packet buffer has been
|
||
|
* processed.
|
||
|
*/
|
||
|
if (OCSD_DATA_RESP_IS_WAIT(cur))
|
||
|
break;
|
||
|
|
||
|
prev_return = cur;
|
||
|
}
|
||
|
|
||
|
decoder->prev_return = cur;
|
||
|
*consumed = processed;
|
||
|
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
void cs_etm_decoder__free(struct cs_etm_decoder *decoder)
|
||
|
{
|
||
|
if (!decoder)
|
||
|
return;
|
||
|
|
||
|
ocsd_destroy_dcd_tree(decoder->dcd_tree);
|
||
|
decoder->dcd_tree = NULL;
|
||
|
free(decoder);
|
||
|
}
|