2018-04-19 01:05:18 +03:00
/* SPDX-License-Identifier: GPL-2.0 */
2016-09-16 18:50:00 +03:00
/*
* Copyright ( C ) 2015 Linaro Limited . All rights reserved .
* Author : Mathieu Poirier < mathieu . poirier @ linaro . org >
*/
# ifndef INCLUDE__UTIL_PERF_CS_ETM_H__
# define INCLUDE__UTIL_PERF_CS_ETM_H__
2018-01-17 20:52:11 +03:00
# include "util/event.h"
2019-06-07 21:14:27 +03:00
# include <linux/bits.h>
2018-01-17 20:52:11 +03:00
2019-08-30 20:45:20 +03:00
struct perf_session ;
2021-03-23 19:09:15 +03:00
/*
* Versioning header in case things need to change in the future . That way
2016-09-16 18:50:00 +03:00
* decoding of old snapshot is still possible .
*/
enum {
/* Starting with 0x0 */
2021-02-24 19:48:30 +03:00
CS_HEADER_VERSION ,
2016-09-16 18:50:00 +03:00
/* PMU->type (32 bit), total # of CPUs (32 bit) */
CS_PMU_TYPE_CPUS ,
CS_ETM_SNAPSHOT ,
2021-02-24 19:48:30 +03:00
CS_HEADER_VERSION_MAX ,
2016-09-16 18:50:00 +03:00
} ;
2021-02-24 19:48:30 +03:00
/*
* Update the version for new format .
*
* New version 1 format adds a param count to the per cpu metadata .
* This allows easy adding of new metadata parameters .
* Requires that new params always added after current ones .
* Also allows client reader to handle file versions that are different by
* checking the number of params in the file vs the number expected .
*/
# define CS_HEADER_CURRENT_VERSION 1
2016-09-16 18:50:00 +03:00
/* Beginning of header common to both ETMv3 and V4 */
enum {
CS_ETM_MAGIC ,
CS_ETM_CPU ,
2021-02-24 19:48:30 +03:00
/* Number of trace config params in following ETM specific block */
CS_ETM_NR_TRC_PARAMS ,
CS_ETM_COMMON_BLK_MAX_V1 ,
2016-09-16 18:50:00 +03:00
} ;
/* ETMv3/PTM metadata */
enum {
/* Dynamic, configurable parameters */
2021-02-24 19:48:30 +03:00
CS_ETM_ETMCR = CS_ETM_COMMON_BLK_MAX_V1 ,
2016-09-16 18:50:00 +03:00
CS_ETM_ETMTRACEIDR ,
/* RO, taken from sysFS */
CS_ETM_ETMCCER ,
CS_ETM_ETMIDR ,
CS_ETM_PRIV_MAX ,
} ;
2021-02-24 19:48:30 +03:00
/* define fixed version 0 length - allow new format reader to read old files. */
# define CS_ETM_NR_TRC_PARAMS_V0 (CS_ETM_ETMIDR - CS_ETM_ETMCR + 1)
2016-09-16 18:50:00 +03:00
/* ETMv4 metadata */
enum {
/* Dynamic, configurable parameters */
2021-02-24 19:48:30 +03:00
CS_ETMV4_TRCCONFIGR = CS_ETM_COMMON_BLK_MAX_V1 ,
2016-09-16 18:50:00 +03:00
CS_ETMV4_TRCTRACEIDR ,
/* RO, taken from sysFS */
CS_ETMV4_TRCIDR0 ,
CS_ETMV4_TRCIDR1 ,
CS_ETMV4_TRCIDR2 ,
CS_ETMV4_TRCIDR8 ,
CS_ETMV4_TRCAUTHSTATUS ,
CS_ETMV4_PRIV_MAX ,
} ;
2021-02-24 19:48:30 +03:00
/* define fixed version 0 length - allow new format reader to read old files. */
# define CS_ETMV4_NR_TRC_PARAMS_V0 (CS_ETMV4_TRCAUTHSTATUS - CS_ETMV4_TRCCONFIGR + 1)
2021-08-06 16:41:04 +03:00
/*
* ETE metadata is ETMv4 plus TRCDEVARCH register and doesn ' t support header V0 since it was
* added in header V1
*/
enum {
CS_ETE_TRCDEVARCH = CS_ETMV4_PRIV_MAX ,
CS_ETE_PRIV_MAX
} ;
2019-01-29 15:28:41 +03:00
/*
* ETMv3 exception encoding number :
2021-03-23 19:09:15 +03:00
* See Embedded Trace Macrocell specification ( ARM IHI 0014 Q )
2019-01-29 15:28:41 +03:00
* table 7 - 12 Encoding of Exception [ 3 : 0 ] for non - ARMv7 - M processors .
*/
enum {
CS_ETMV3_EXC_NONE = 0 ,
CS_ETMV3_EXC_DEBUG_HALT = 1 ,
CS_ETMV3_EXC_SMC = 2 ,
CS_ETMV3_EXC_HYP = 3 ,
CS_ETMV3_EXC_ASYNC_DATA_ABORT = 4 ,
CS_ETMV3_EXC_JAZELLE_THUMBEE = 5 ,
CS_ETMV3_EXC_PE_RESET = 8 ,
CS_ETMV3_EXC_UNDEFINED_INSTR = 9 ,
CS_ETMV3_EXC_SVC = 10 ,
CS_ETMV3_EXC_PREFETCH_ABORT = 11 ,
CS_ETMV3_EXC_DATA_FAULT = 12 ,
CS_ETMV3_EXC_GENERIC = 13 ,
CS_ETMV3_EXC_IRQ = 14 ,
CS_ETMV3_EXC_FIQ = 15 ,
} ;
/*
* ETMv4 exception encoding number :
* See ARM Embedded Trace Macrocell Architecture Specification ( ARM IHI 0064 D )
* table 6 - 12 Possible values for the TYPE field in an Exception instruction
* trace packet , for ARMv7 - A / R and ARMv8 - A / R PEs .
*/
enum {
CS_ETMV4_EXC_RESET = 0 ,
CS_ETMV4_EXC_DEBUG_HALT = 1 ,
CS_ETMV4_EXC_CALL = 2 ,
CS_ETMV4_EXC_TRAP = 3 ,
CS_ETMV4_EXC_SYSTEM_ERROR = 4 ,
CS_ETMV4_EXC_INST_DEBUG = 6 ,
CS_ETMV4_EXC_DATA_DEBUG = 7 ,
CS_ETMV4_EXC_ALIGNMENT = 10 ,
CS_ETMV4_EXC_INST_FAULT = 11 ,
CS_ETMV4_EXC_DATA_FAULT = 12 ,
CS_ETMV4_EXC_IRQ = 14 ,
CS_ETMV4_EXC_FIQ = 15 ,
CS_ETMV4_EXC_END = 31 ,
} ;
2019-05-24 20:34:58 +03:00
enum cs_etm_sample_type {
CS_ETM_EMPTY ,
CS_ETM_RANGE ,
CS_ETM_DISCONTINUITY ,
CS_ETM_EXCEPTION ,
CS_ETM_EXCEPTION_RET ,
} ;
enum cs_etm_isa {
CS_ETM_ISA_UNKNOWN ,
CS_ETM_ISA_A64 ,
CS_ETM_ISA_A32 ,
CS_ETM_ISA_T32 ,
} ;
struct cs_etm_queue ;
struct cs_etm_packet {
enum cs_etm_sample_type sample_type ;
enum cs_etm_isa isa ;
u64 start_addr ;
u64 end_addr ;
u32 instr_count ;
u32 last_instr_type ;
u32 last_instr_subtype ;
u32 flags ;
u32 exception_number ;
u8 last_instr_cond ;
u8 last_instr_taken_branch ;
u8 last_instr_size ;
u8 trace_chan_id ;
int cpu ;
} ;
# define CS_ETM_PACKET_MAX_BUFFER 1024
2019-05-24 20:35:00 +03:00
/*
* When working with per - thread scenarios the process under trace can
* be scheduled on any CPU and as such , more than one traceID may be
* associated with the same process . Since a traceID of ' 0 ' is illegal
* as per the CoreSight architecture , use that specific value to
* identify the queue where all packets ( with any traceID ) are
* aggregated .
*/
# define CS_ETM_PER_THREAD_TRACEID 0
2019-05-24 20:34:58 +03:00
struct cs_etm_packet_queue {
u32 packet_count ;
u32 head ;
u32 tail ;
2019-05-24 20:35:07 +03:00
u32 instr_count ;
2021-05-10 17:32:47 +03:00
u64 cs_timestamp ;
u64 next_cs_timestamp ;
2019-05-24 20:34:58 +03:00
struct cs_etm_packet packet_buffer [ CS_ETM_PACKET_MAX_BUFFER ] ;
} ;
2016-09-16 18:50:00 +03:00
# define KiB(x) ((x) * 1024)
# define MiB(x) ((x) * 1024 * 1024)
2019-05-24 20:34:58 +03:00
# define CS_ETM_INVAL_ADDR 0xdeadbeefdeadbeefUL
2019-05-24 20:34:52 +03:00
# define BMVAL(val, lsb, msb) ((val & GENMASK(msb, lsb)) >> lsb)
2021-02-24 19:48:30 +03:00
# define CS_ETM_HEADER_SIZE (CS_HEADER_VERSION_MAX * sizeof(u64))
2016-09-16 18:50:00 +03:00
2019-02-12 20:16:11 +03:00
# define __perf_cs_etmv3_magic 0x3030303030303030ULL
# define __perf_cs_etmv4_magic 0x4040404040404040ULL
2021-08-06 16:41:04 +03:00
# define __perf_cs_ete_magic 0x5050505050505050ULL
2016-09-16 18:50:00 +03:00
# define CS_ETMV3_PRIV_SIZE (CS_ETM_PRIV_MAX * sizeof(u64))
# define CS_ETMV4_PRIV_SIZE (CS_ETMV4_PRIV_MAX * sizeof(u64))
2021-08-06 16:41:04 +03:00
# define CS_ETE_PRIV_SIZE (CS_ETE_PRIV_MAX * sizeof(u64))
2016-09-16 18:50:00 +03:00
2018-01-17 20:52:11 +03:00
# ifdef HAVE_CSTRACE_SUPPORT
int cs_etm__process_auxtrace_info ( union perf_event * event ,
struct perf_session * session ) ;
2019-01-29 15:28:39 +03:00
int cs_etm__get_cpu ( u8 trace_chan_id , int * cpu ) ;
2021-02-24 19:48:34 +03:00
int cs_etm__get_pid_fmt ( u8 trace_chan_id , u64 * pid_fmt ) ;
2019-05-24 20:35:06 +03:00
int cs_etm__etmq_set_tid ( struct cs_etm_queue * etmq ,
pid_t tid , u8 trace_chan_id ) ;
2019-05-24 20:35:07 +03:00
bool cs_etm__etmq_is_timeless ( struct cs_etm_queue * etmq ) ;
void cs_etm__etmq_set_traceid_queue_timestamp ( struct cs_etm_queue * etmq ,
u8 trace_chan_id ) ;
2019-05-24 20:34:58 +03:00
struct cs_etm_packet_queue
2019-05-24 20:35:00 +03:00
* cs_etm__etmq_get_packet_queue ( struct cs_etm_queue * etmq , u8 trace_chan_id ) ;
2018-01-17 20:52:11 +03:00
# else
static inline int
cs_etm__process_auxtrace_info ( union perf_event * event __maybe_unused ,
struct perf_session * session __maybe_unused )
{
return - 1 ;
}
2019-01-29 15:28:39 +03:00
static inline int cs_etm__get_cpu ( u8 trace_chan_id __maybe_unused ,
int * cpu __maybe_unused )
{
return - 1 ;
}
2019-05-24 20:34:58 +03:00
2019-05-24 20:35:06 +03:00
static inline int cs_etm__etmq_set_tid (
struct cs_etm_queue * etmq __maybe_unused ,
pid_t tid __maybe_unused ,
u8 trace_chan_id __maybe_unused )
{
return - 1 ;
}
2019-05-24 20:35:07 +03:00
static inline bool cs_etm__etmq_is_timeless (
struct cs_etm_queue * etmq __maybe_unused )
{
/* What else to return? */
return true ;
}
static inline void cs_etm__etmq_set_traceid_queue_timestamp (
struct cs_etm_queue * etmq __maybe_unused ,
u8 trace_chan_id __maybe_unused ) { }
2019-05-24 20:34:58 +03:00
static inline struct cs_etm_packet_queue * cs_etm__etmq_get_packet_queue (
2019-05-24 20:35:00 +03:00
struct cs_etm_queue * etmq __maybe_unused ,
u8 trace_chan_id __maybe_unused )
2019-05-24 20:34:58 +03:00
{
return NULL ;
}
2018-01-17 20:52:11 +03:00
# endif
2016-09-16 18:50:00 +03:00
# endif