xref: /OK3568_Linux_fs/kernel/tools/perf/util/intel-pt-decoder/intel-pt-pkt-decoder.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0-only
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun  * intel_pt_pkt_decoder.c: Intel Processor Trace support
4*4882a593Smuzhiyun  * Copyright (c) 2013-2014, Intel Corporation.
5*4882a593Smuzhiyun  */
6*4882a593Smuzhiyun 
7*4882a593Smuzhiyun #include <stdio.h>
8*4882a593Smuzhiyun #include <string.h>
9*4882a593Smuzhiyun #include <endian.h>
10*4882a593Smuzhiyun #include <byteswap.h>
11*4882a593Smuzhiyun #include <linux/compiler.h>
12*4882a593Smuzhiyun 
13*4882a593Smuzhiyun #include "intel-pt-pkt-decoder.h"
14*4882a593Smuzhiyun 
15*4882a593Smuzhiyun #define BIT(n)		(1 << (n))
16*4882a593Smuzhiyun 
17*4882a593Smuzhiyun #define BIT63		((uint64_t)1 << 63)
18*4882a593Smuzhiyun 
19*4882a593Smuzhiyun #define NR_FLAG		BIT63
20*4882a593Smuzhiyun 
21*4882a593Smuzhiyun #if __BYTE_ORDER == __BIG_ENDIAN
22*4882a593Smuzhiyun #define le16_to_cpu bswap_16
23*4882a593Smuzhiyun #define le32_to_cpu bswap_32
24*4882a593Smuzhiyun #define le64_to_cpu bswap_64
25*4882a593Smuzhiyun #define memcpy_le64(d, s, n) do { \
26*4882a593Smuzhiyun 	memcpy((d), (s), (n));    \
27*4882a593Smuzhiyun 	*(d) = le64_to_cpu(*(d)); \
28*4882a593Smuzhiyun } while (0)
29*4882a593Smuzhiyun #else
30*4882a593Smuzhiyun #define le16_to_cpu
31*4882a593Smuzhiyun #define le32_to_cpu
32*4882a593Smuzhiyun #define le64_to_cpu
33*4882a593Smuzhiyun #define memcpy_le64 memcpy
34*4882a593Smuzhiyun #endif
35*4882a593Smuzhiyun 
36*4882a593Smuzhiyun static const char * const packet_name[] = {
37*4882a593Smuzhiyun 	[INTEL_PT_BAD]		= "Bad Packet!",
38*4882a593Smuzhiyun 	[INTEL_PT_PAD]		= "PAD",
39*4882a593Smuzhiyun 	[INTEL_PT_TNT]		= "TNT",
40*4882a593Smuzhiyun 	[INTEL_PT_TIP_PGD]	= "TIP.PGD",
41*4882a593Smuzhiyun 	[INTEL_PT_TIP_PGE]	= "TIP.PGE",
42*4882a593Smuzhiyun 	[INTEL_PT_TSC]		= "TSC",
43*4882a593Smuzhiyun 	[INTEL_PT_TMA]		= "TMA",
44*4882a593Smuzhiyun 	[INTEL_PT_MODE_EXEC]	= "MODE.Exec",
45*4882a593Smuzhiyun 	[INTEL_PT_MODE_TSX]	= "MODE.TSX",
46*4882a593Smuzhiyun 	[INTEL_PT_MTC]		= "MTC",
47*4882a593Smuzhiyun 	[INTEL_PT_TIP]		= "TIP",
48*4882a593Smuzhiyun 	[INTEL_PT_FUP]		= "FUP",
49*4882a593Smuzhiyun 	[INTEL_PT_CYC]		= "CYC",
50*4882a593Smuzhiyun 	[INTEL_PT_VMCS]		= "VMCS",
51*4882a593Smuzhiyun 	[INTEL_PT_PSB]		= "PSB",
52*4882a593Smuzhiyun 	[INTEL_PT_PSBEND]	= "PSBEND",
53*4882a593Smuzhiyun 	[INTEL_PT_CBR]		= "CBR",
54*4882a593Smuzhiyun 	[INTEL_PT_TRACESTOP]	= "TraceSTOP",
55*4882a593Smuzhiyun 	[INTEL_PT_PIP]		= "PIP",
56*4882a593Smuzhiyun 	[INTEL_PT_OVF]		= "OVF",
57*4882a593Smuzhiyun 	[INTEL_PT_MNT]		= "MNT",
58*4882a593Smuzhiyun 	[INTEL_PT_PTWRITE]	= "PTWRITE",
59*4882a593Smuzhiyun 	[INTEL_PT_PTWRITE_IP]	= "PTWRITE",
60*4882a593Smuzhiyun 	[INTEL_PT_EXSTOP]	= "EXSTOP",
61*4882a593Smuzhiyun 	[INTEL_PT_EXSTOP_IP]	= "EXSTOP",
62*4882a593Smuzhiyun 	[INTEL_PT_MWAIT]	= "MWAIT",
63*4882a593Smuzhiyun 	[INTEL_PT_PWRE]		= "PWRE",
64*4882a593Smuzhiyun 	[INTEL_PT_PWRX]		= "PWRX",
65*4882a593Smuzhiyun 	[INTEL_PT_BBP]		= "BBP",
66*4882a593Smuzhiyun 	[INTEL_PT_BIP]		= "BIP",
67*4882a593Smuzhiyun 	[INTEL_PT_BEP]		= "BEP",
68*4882a593Smuzhiyun 	[INTEL_PT_BEP_IP]	= "BEP",
69*4882a593Smuzhiyun };
70*4882a593Smuzhiyun 
intel_pt_pkt_name(enum intel_pt_pkt_type type)71*4882a593Smuzhiyun const char *intel_pt_pkt_name(enum intel_pt_pkt_type type)
72*4882a593Smuzhiyun {
73*4882a593Smuzhiyun 	return packet_name[type];
74*4882a593Smuzhiyun }
75*4882a593Smuzhiyun 
intel_pt_get_long_tnt(const unsigned char * buf,size_t len,struct intel_pt_pkt * packet)76*4882a593Smuzhiyun static int intel_pt_get_long_tnt(const unsigned char *buf, size_t len,
77*4882a593Smuzhiyun 				 struct intel_pt_pkt *packet)
78*4882a593Smuzhiyun {
79*4882a593Smuzhiyun 	uint64_t payload;
80*4882a593Smuzhiyun 	int count;
81*4882a593Smuzhiyun 
82*4882a593Smuzhiyun 	if (len < 8)
83*4882a593Smuzhiyun 		return INTEL_PT_NEED_MORE_BYTES;
84*4882a593Smuzhiyun 
85*4882a593Smuzhiyun 	payload = le64_to_cpu(*(uint64_t *)buf);
86*4882a593Smuzhiyun 
87*4882a593Smuzhiyun 	for (count = 47; count; count--) {
88*4882a593Smuzhiyun 		if (payload & BIT63)
89*4882a593Smuzhiyun 			break;
90*4882a593Smuzhiyun 		payload <<= 1;
91*4882a593Smuzhiyun 	}
92*4882a593Smuzhiyun 
93*4882a593Smuzhiyun 	packet->type = INTEL_PT_TNT;
94*4882a593Smuzhiyun 	packet->count = count;
95*4882a593Smuzhiyun 	packet->payload = payload << 1;
96*4882a593Smuzhiyun 	return 8;
97*4882a593Smuzhiyun }
98*4882a593Smuzhiyun 
intel_pt_get_pip(const unsigned char * buf,size_t len,struct intel_pt_pkt * packet)99*4882a593Smuzhiyun static int intel_pt_get_pip(const unsigned char *buf, size_t len,
100*4882a593Smuzhiyun 			    struct intel_pt_pkt *packet)
101*4882a593Smuzhiyun {
102*4882a593Smuzhiyun 	uint64_t payload = 0;
103*4882a593Smuzhiyun 
104*4882a593Smuzhiyun 	if (len < 8)
105*4882a593Smuzhiyun 		return INTEL_PT_NEED_MORE_BYTES;
106*4882a593Smuzhiyun 
107*4882a593Smuzhiyun 	packet->type = INTEL_PT_PIP;
108*4882a593Smuzhiyun 	memcpy_le64(&payload, buf + 2, 6);
109*4882a593Smuzhiyun 	packet->payload = payload >> 1;
110*4882a593Smuzhiyun 	if (payload & 1)
111*4882a593Smuzhiyun 		packet->payload |= NR_FLAG;
112*4882a593Smuzhiyun 
113*4882a593Smuzhiyun 	return 8;
114*4882a593Smuzhiyun }
115*4882a593Smuzhiyun 
intel_pt_get_tracestop(struct intel_pt_pkt * packet)116*4882a593Smuzhiyun static int intel_pt_get_tracestop(struct intel_pt_pkt *packet)
117*4882a593Smuzhiyun {
118*4882a593Smuzhiyun 	packet->type = INTEL_PT_TRACESTOP;
119*4882a593Smuzhiyun 	return 2;
120*4882a593Smuzhiyun }
121*4882a593Smuzhiyun 
intel_pt_get_cbr(const unsigned char * buf,size_t len,struct intel_pt_pkt * packet)122*4882a593Smuzhiyun static int intel_pt_get_cbr(const unsigned char *buf, size_t len,
123*4882a593Smuzhiyun 			    struct intel_pt_pkt *packet)
124*4882a593Smuzhiyun {
125*4882a593Smuzhiyun 	if (len < 4)
126*4882a593Smuzhiyun 		return INTEL_PT_NEED_MORE_BYTES;
127*4882a593Smuzhiyun 	packet->type = INTEL_PT_CBR;
128*4882a593Smuzhiyun 	packet->payload = le16_to_cpu(*(uint16_t *)(buf + 2));
129*4882a593Smuzhiyun 	return 4;
130*4882a593Smuzhiyun }
131*4882a593Smuzhiyun 
intel_pt_get_vmcs(const unsigned char * buf,size_t len,struct intel_pt_pkt * packet)132*4882a593Smuzhiyun static int intel_pt_get_vmcs(const unsigned char *buf, size_t len,
133*4882a593Smuzhiyun 			     struct intel_pt_pkt *packet)
134*4882a593Smuzhiyun {
135*4882a593Smuzhiyun 	unsigned int count = (52 - 5) >> 3;
136*4882a593Smuzhiyun 
137*4882a593Smuzhiyun 	if (count < 1 || count > 7)
138*4882a593Smuzhiyun 		return INTEL_PT_BAD_PACKET;
139*4882a593Smuzhiyun 
140*4882a593Smuzhiyun 	if (len < count + 2)
141*4882a593Smuzhiyun 		return INTEL_PT_NEED_MORE_BYTES;
142*4882a593Smuzhiyun 
143*4882a593Smuzhiyun 	packet->type = INTEL_PT_VMCS;
144*4882a593Smuzhiyun 	packet->count = count;
145*4882a593Smuzhiyun 	memcpy_le64(&packet->payload, buf + 2, count);
146*4882a593Smuzhiyun 
147*4882a593Smuzhiyun 	return count + 2;
148*4882a593Smuzhiyun }
149*4882a593Smuzhiyun 
intel_pt_get_ovf(struct intel_pt_pkt * packet)150*4882a593Smuzhiyun static int intel_pt_get_ovf(struct intel_pt_pkt *packet)
151*4882a593Smuzhiyun {
152*4882a593Smuzhiyun 	packet->type = INTEL_PT_OVF;
153*4882a593Smuzhiyun 	return 2;
154*4882a593Smuzhiyun }
155*4882a593Smuzhiyun 
intel_pt_get_psb(const unsigned char * buf,size_t len,struct intel_pt_pkt * packet)156*4882a593Smuzhiyun static int intel_pt_get_psb(const unsigned char *buf, size_t len,
157*4882a593Smuzhiyun 			    struct intel_pt_pkt *packet)
158*4882a593Smuzhiyun {
159*4882a593Smuzhiyun 	int i;
160*4882a593Smuzhiyun 
161*4882a593Smuzhiyun 	if (len < 16)
162*4882a593Smuzhiyun 		return INTEL_PT_NEED_MORE_BYTES;
163*4882a593Smuzhiyun 
164*4882a593Smuzhiyun 	for (i = 2; i < 16; i += 2) {
165*4882a593Smuzhiyun 		if (buf[i] != 2 || buf[i + 1] != 0x82)
166*4882a593Smuzhiyun 			return INTEL_PT_BAD_PACKET;
167*4882a593Smuzhiyun 	}
168*4882a593Smuzhiyun 
169*4882a593Smuzhiyun 	packet->type = INTEL_PT_PSB;
170*4882a593Smuzhiyun 	return 16;
171*4882a593Smuzhiyun }
172*4882a593Smuzhiyun 
intel_pt_get_psbend(struct intel_pt_pkt * packet)173*4882a593Smuzhiyun static int intel_pt_get_psbend(struct intel_pt_pkt *packet)
174*4882a593Smuzhiyun {
175*4882a593Smuzhiyun 	packet->type = INTEL_PT_PSBEND;
176*4882a593Smuzhiyun 	return 2;
177*4882a593Smuzhiyun }
178*4882a593Smuzhiyun 
intel_pt_get_tma(const unsigned char * buf,size_t len,struct intel_pt_pkt * packet)179*4882a593Smuzhiyun static int intel_pt_get_tma(const unsigned char *buf, size_t len,
180*4882a593Smuzhiyun 			    struct intel_pt_pkt *packet)
181*4882a593Smuzhiyun {
182*4882a593Smuzhiyun 	if (len < 7)
183*4882a593Smuzhiyun 		return INTEL_PT_NEED_MORE_BYTES;
184*4882a593Smuzhiyun 
185*4882a593Smuzhiyun 	packet->type = INTEL_PT_TMA;
186*4882a593Smuzhiyun 	packet->payload = buf[2] | (buf[3] << 8);
187*4882a593Smuzhiyun 	packet->count = buf[5] | ((buf[6] & BIT(0)) << 8);
188*4882a593Smuzhiyun 	return 7;
189*4882a593Smuzhiyun }
190*4882a593Smuzhiyun 
intel_pt_get_pad(struct intel_pt_pkt * packet)191*4882a593Smuzhiyun static int intel_pt_get_pad(struct intel_pt_pkt *packet)
192*4882a593Smuzhiyun {
193*4882a593Smuzhiyun 	packet->type = INTEL_PT_PAD;
194*4882a593Smuzhiyun 	return 1;
195*4882a593Smuzhiyun }
196*4882a593Smuzhiyun 
intel_pt_get_mnt(const unsigned char * buf,size_t len,struct intel_pt_pkt * packet)197*4882a593Smuzhiyun static int intel_pt_get_mnt(const unsigned char *buf, size_t len,
198*4882a593Smuzhiyun 			    struct intel_pt_pkt *packet)
199*4882a593Smuzhiyun {
200*4882a593Smuzhiyun 	if (len < 11)
201*4882a593Smuzhiyun 		return INTEL_PT_NEED_MORE_BYTES;
202*4882a593Smuzhiyun 	packet->type = INTEL_PT_MNT;
203*4882a593Smuzhiyun 	memcpy_le64(&packet->payload, buf + 3, 8);
204*4882a593Smuzhiyun 	return 11
205*4882a593Smuzhiyun ;
206*4882a593Smuzhiyun }
207*4882a593Smuzhiyun 
intel_pt_get_3byte(const unsigned char * buf,size_t len,struct intel_pt_pkt * packet)208*4882a593Smuzhiyun static int intel_pt_get_3byte(const unsigned char *buf, size_t len,
209*4882a593Smuzhiyun 			      struct intel_pt_pkt *packet)
210*4882a593Smuzhiyun {
211*4882a593Smuzhiyun 	if (len < 3)
212*4882a593Smuzhiyun 		return INTEL_PT_NEED_MORE_BYTES;
213*4882a593Smuzhiyun 
214*4882a593Smuzhiyun 	switch (buf[2]) {
215*4882a593Smuzhiyun 	case 0x88: /* MNT */
216*4882a593Smuzhiyun 		return intel_pt_get_mnt(buf, len, packet);
217*4882a593Smuzhiyun 	default:
218*4882a593Smuzhiyun 		return INTEL_PT_BAD_PACKET;
219*4882a593Smuzhiyun 	}
220*4882a593Smuzhiyun }
221*4882a593Smuzhiyun 
intel_pt_get_ptwrite(const unsigned char * buf,size_t len,struct intel_pt_pkt * packet)222*4882a593Smuzhiyun static int intel_pt_get_ptwrite(const unsigned char *buf, size_t len,
223*4882a593Smuzhiyun 				struct intel_pt_pkt *packet)
224*4882a593Smuzhiyun {
225*4882a593Smuzhiyun 	packet->count = (buf[1] >> 5) & 0x3;
226*4882a593Smuzhiyun 	packet->type = buf[1] & BIT(7) ? INTEL_PT_PTWRITE_IP :
227*4882a593Smuzhiyun 					 INTEL_PT_PTWRITE;
228*4882a593Smuzhiyun 
229*4882a593Smuzhiyun 	switch (packet->count) {
230*4882a593Smuzhiyun 	case 0:
231*4882a593Smuzhiyun 		if (len < 6)
232*4882a593Smuzhiyun 			return INTEL_PT_NEED_MORE_BYTES;
233*4882a593Smuzhiyun 		packet->payload = le32_to_cpu(*(uint32_t *)(buf + 2));
234*4882a593Smuzhiyun 		return 6;
235*4882a593Smuzhiyun 	case 1:
236*4882a593Smuzhiyun 		if (len < 10)
237*4882a593Smuzhiyun 			return INTEL_PT_NEED_MORE_BYTES;
238*4882a593Smuzhiyun 		packet->payload = le64_to_cpu(*(uint64_t *)(buf + 2));
239*4882a593Smuzhiyun 		return 10;
240*4882a593Smuzhiyun 	default:
241*4882a593Smuzhiyun 		return INTEL_PT_BAD_PACKET;
242*4882a593Smuzhiyun 	}
243*4882a593Smuzhiyun }
244*4882a593Smuzhiyun 
intel_pt_get_exstop(struct intel_pt_pkt * packet)245*4882a593Smuzhiyun static int intel_pt_get_exstop(struct intel_pt_pkt *packet)
246*4882a593Smuzhiyun {
247*4882a593Smuzhiyun 	packet->type = INTEL_PT_EXSTOP;
248*4882a593Smuzhiyun 	return 2;
249*4882a593Smuzhiyun }
250*4882a593Smuzhiyun 
intel_pt_get_exstop_ip(struct intel_pt_pkt * packet)251*4882a593Smuzhiyun static int intel_pt_get_exstop_ip(struct intel_pt_pkt *packet)
252*4882a593Smuzhiyun {
253*4882a593Smuzhiyun 	packet->type = INTEL_PT_EXSTOP_IP;
254*4882a593Smuzhiyun 	return 2;
255*4882a593Smuzhiyun }
256*4882a593Smuzhiyun 
intel_pt_get_mwait(const unsigned char * buf,size_t len,struct intel_pt_pkt * packet)257*4882a593Smuzhiyun static int intel_pt_get_mwait(const unsigned char *buf, size_t len,
258*4882a593Smuzhiyun 			      struct intel_pt_pkt *packet)
259*4882a593Smuzhiyun {
260*4882a593Smuzhiyun 	if (len < 10)
261*4882a593Smuzhiyun 		return INTEL_PT_NEED_MORE_BYTES;
262*4882a593Smuzhiyun 	packet->type = INTEL_PT_MWAIT;
263*4882a593Smuzhiyun 	packet->payload = le64_to_cpu(*(uint64_t *)(buf + 2));
264*4882a593Smuzhiyun 	return 10;
265*4882a593Smuzhiyun }
266*4882a593Smuzhiyun 
intel_pt_get_pwre(const unsigned char * buf,size_t len,struct intel_pt_pkt * packet)267*4882a593Smuzhiyun static int intel_pt_get_pwre(const unsigned char *buf, size_t len,
268*4882a593Smuzhiyun 			     struct intel_pt_pkt *packet)
269*4882a593Smuzhiyun {
270*4882a593Smuzhiyun 	if (len < 4)
271*4882a593Smuzhiyun 		return INTEL_PT_NEED_MORE_BYTES;
272*4882a593Smuzhiyun 	packet->type = INTEL_PT_PWRE;
273*4882a593Smuzhiyun 	memcpy_le64(&packet->payload, buf + 2, 2);
274*4882a593Smuzhiyun 	return 4;
275*4882a593Smuzhiyun }
276*4882a593Smuzhiyun 
intel_pt_get_pwrx(const unsigned char * buf,size_t len,struct intel_pt_pkt * packet)277*4882a593Smuzhiyun static int intel_pt_get_pwrx(const unsigned char *buf, size_t len,
278*4882a593Smuzhiyun 			     struct intel_pt_pkt *packet)
279*4882a593Smuzhiyun {
280*4882a593Smuzhiyun 	if (len < 7)
281*4882a593Smuzhiyun 		return INTEL_PT_NEED_MORE_BYTES;
282*4882a593Smuzhiyun 	packet->type = INTEL_PT_PWRX;
283*4882a593Smuzhiyun 	memcpy_le64(&packet->payload, buf + 2, 5);
284*4882a593Smuzhiyun 	return 7;
285*4882a593Smuzhiyun }
286*4882a593Smuzhiyun 
intel_pt_get_bbp(const unsigned char * buf,size_t len,struct intel_pt_pkt * packet)287*4882a593Smuzhiyun static int intel_pt_get_bbp(const unsigned char *buf, size_t len,
288*4882a593Smuzhiyun 			    struct intel_pt_pkt *packet)
289*4882a593Smuzhiyun {
290*4882a593Smuzhiyun 	if (len < 3)
291*4882a593Smuzhiyun 		return INTEL_PT_NEED_MORE_BYTES;
292*4882a593Smuzhiyun 	packet->type = INTEL_PT_BBP;
293*4882a593Smuzhiyun 	packet->count = buf[2] >> 7;
294*4882a593Smuzhiyun 	packet->payload = buf[2] & 0x1f;
295*4882a593Smuzhiyun 	return 3;
296*4882a593Smuzhiyun }
297*4882a593Smuzhiyun 
intel_pt_get_bip_4(const unsigned char * buf,size_t len,struct intel_pt_pkt * packet)298*4882a593Smuzhiyun static int intel_pt_get_bip_4(const unsigned char *buf, size_t len,
299*4882a593Smuzhiyun 			      struct intel_pt_pkt *packet)
300*4882a593Smuzhiyun {
301*4882a593Smuzhiyun 	if (len < 5)
302*4882a593Smuzhiyun 		return INTEL_PT_NEED_MORE_BYTES;
303*4882a593Smuzhiyun 	packet->type = INTEL_PT_BIP;
304*4882a593Smuzhiyun 	packet->count = buf[0] >> 3;
305*4882a593Smuzhiyun 	memcpy_le64(&packet->payload, buf + 1, 4);
306*4882a593Smuzhiyun 	return 5;
307*4882a593Smuzhiyun }
308*4882a593Smuzhiyun 
intel_pt_get_bip_8(const unsigned char * buf,size_t len,struct intel_pt_pkt * packet)309*4882a593Smuzhiyun static int intel_pt_get_bip_8(const unsigned char *buf, size_t len,
310*4882a593Smuzhiyun 			      struct intel_pt_pkt *packet)
311*4882a593Smuzhiyun {
312*4882a593Smuzhiyun 	if (len < 9)
313*4882a593Smuzhiyun 		return INTEL_PT_NEED_MORE_BYTES;
314*4882a593Smuzhiyun 	packet->type = INTEL_PT_BIP;
315*4882a593Smuzhiyun 	packet->count = buf[0] >> 3;
316*4882a593Smuzhiyun 	memcpy_le64(&packet->payload, buf + 1, 8);
317*4882a593Smuzhiyun 	return 9;
318*4882a593Smuzhiyun }
319*4882a593Smuzhiyun 
intel_pt_get_bep(size_t len,struct intel_pt_pkt * packet)320*4882a593Smuzhiyun static int intel_pt_get_bep(size_t len, struct intel_pt_pkt *packet)
321*4882a593Smuzhiyun {
322*4882a593Smuzhiyun 	if (len < 2)
323*4882a593Smuzhiyun 		return INTEL_PT_NEED_MORE_BYTES;
324*4882a593Smuzhiyun 	packet->type = INTEL_PT_BEP;
325*4882a593Smuzhiyun 	return 2;
326*4882a593Smuzhiyun }
327*4882a593Smuzhiyun 
intel_pt_get_bep_ip(size_t len,struct intel_pt_pkt * packet)328*4882a593Smuzhiyun static int intel_pt_get_bep_ip(size_t len, struct intel_pt_pkt *packet)
329*4882a593Smuzhiyun {
330*4882a593Smuzhiyun 	if (len < 2)
331*4882a593Smuzhiyun 		return INTEL_PT_NEED_MORE_BYTES;
332*4882a593Smuzhiyun 	packet->type = INTEL_PT_BEP_IP;
333*4882a593Smuzhiyun 	return 2;
334*4882a593Smuzhiyun }
335*4882a593Smuzhiyun 
intel_pt_get_ext(const unsigned char * buf,size_t len,struct intel_pt_pkt * packet)336*4882a593Smuzhiyun static int intel_pt_get_ext(const unsigned char *buf, size_t len,
337*4882a593Smuzhiyun 			    struct intel_pt_pkt *packet)
338*4882a593Smuzhiyun {
339*4882a593Smuzhiyun 	if (len < 2)
340*4882a593Smuzhiyun 		return INTEL_PT_NEED_MORE_BYTES;
341*4882a593Smuzhiyun 
342*4882a593Smuzhiyun 	if ((buf[1] & 0x1f) == 0x12)
343*4882a593Smuzhiyun 		return intel_pt_get_ptwrite(buf, len, packet);
344*4882a593Smuzhiyun 
345*4882a593Smuzhiyun 	switch (buf[1]) {
346*4882a593Smuzhiyun 	case 0xa3: /* Long TNT */
347*4882a593Smuzhiyun 		return intel_pt_get_long_tnt(buf, len, packet);
348*4882a593Smuzhiyun 	case 0x43: /* PIP */
349*4882a593Smuzhiyun 		return intel_pt_get_pip(buf, len, packet);
350*4882a593Smuzhiyun 	case 0x83: /* TraceStop */
351*4882a593Smuzhiyun 		return intel_pt_get_tracestop(packet);
352*4882a593Smuzhiyun 	case 0x03: /* CBR */
353*4882a593Smuzhiyun 		return intel_pt_get_cbr(buf, len, packet);
354*4882a593Smuzhiyun 	case 0xc8: /* VMCS */
355*4882a593Smuzhiyun 		return intel_pt_get_vmcs(buf, len, packet);
356*4882a593Smuzhiyun 	case 0xf3: /* OVF */
357*4882a593Smuzhiyun 		return intel_pt_get_ovf(packet);
358*4882a593Smuzhiyun 	case 0x82: /* PSB */
359*4882a593Smuzhiyun 		return intel_pt_get_psb(buf, len, packet);
360*4882a593Smuzhiyun 	case 0x23: /* PSBEND */
361*4882a593Smuzhiyun 		return intel_pt_get_psbend(packet);
362*4882a593Smuzhiyun 	case 0x73: /* TMA */
363*4882a593Smuzhiyun 		return intel_pt_get_tma(buf, len, packet);
364*4882a593Smuzhiyun 	case 0xC3: /* 3-byte header */
365*4882a593Smuzhiyun 		return intel_pt_get_3byte(buf, len, packet);
366*4882a593Smuzhiyun 	case 0x62: /* EXSTOP no IP */
367*4882a593Smuzhiyun 		return intel_pt_get_exstop(packet);
368*4882a593Smuzhiyun 	case 0xE2: /* EXSTOP with IP */
369*4882a593Smuzhiyun 		return intel_pt_get_exstop_ip(packet);
370*4882a593Smuzhiyun 	case 0xC2: /* MWAIT */
371*4882a593Smuzhiyun 		return intel_pt_get_mwait(buf, len, packet);
372*4882a593Smuzhiyun 	case 0x22: /* PWRE */
373*4882a593Smuzhiyun 		return intel_pt_get_pwre(buf, len, packet);
374*4882a593Smuzhiyun 	case 0xA2: /* PWRX */
375*4882a593Smuzhiyun 		return intel_pt_get_pwrx(buf, len, packet);
376*4882a593Smuzhiyun 	case 0x63: /* BBP */
377*4882a593Smuzhiyun 		return intel_pt_get_bbp(buf, len, packet);
378*4882a593Smuzhiyun 	case 0x33: /* BEP no IP */
379*4882a593Smuzhiyun 		return intel_pt_get_bep(len, packet);
380*4882a593Smuzhiyun 	case 0xb3: /* BEP with IP */
381*4882a593Smuzhiyun 		return intel_pt_get_bep_ip(len, packet);
382*4882a593Smuzhiyun 	default:
383*4882a593Smuzhiyun 		return INTEL_PT_BAD_PACKET;
384*4882a593Smuzhiyun 	}
385*4882a593Smuzhiyun }
386*4882a593Smuzhiyun 
intel_pt_get_short_tnt(unsigned int byte,struct intel_pt_pkt * packet)387*4882a593Smuzhiyun static int intel_pt_get_short_tnt(unsigned int byte,
388*4882a593Smuzhiyun 				  struct intel_pt_pkt *packet)
389*4882a593Smuzhiyun {
390*4882a593Smuzhiyun 	int count;
391*4882a593Smuzhiyun 
392*4882a593Smuzhiyun 	for (count = 6; count; count--) {
393*4882a593Smuzhiyun 		if (byte & BIT(7))
394*4882a593Smuzhiyun 			break;
395*4882a593Smuzhiyun 		byte <<= 1;
396*4882a593Smuzhiyun 	}
397*4882a593Smuzhiyun 
398*4882a593Smuzhiyun 	packet->type = INTEL_PT_TNT;
399*4882a593Smuzhiyun 	packet->count = count;
400*4882a593Smuzhiyun 	packet->payload = (uint64_t)byte << 57;
401*4882a593Smuzhiyun 
402*4882a593Smuzhiyun 	return 1;
403*4882a593Smuzhiyun }
404*4882a593Smuzhiyun 
intel_pt_get_cyc(unsigned int byte,const unsigned char * buf,size_t len,struct intel_pt_pkt * packet)405*4882a593Smuzhiyun static int intel_pt_get_cyc(unsigned int byte, const unsigned char *buf,
406*4882a593Smuzhiyun 			    size_t len, struct intel_pt_pkt *packet)
407*4882a593Smuzhiyun {
408*4882a593Smuzhiyun 	unsigned int offs = 1, shift;
409*4882a593Smuzhiyun 	uint64_t payload = byte >> 3;
410*4882a593Smuzhiyun 
411*4882a593Smuzhiyun 	byte >>= 2;
412*4882a593Smuzhiyun 	len -= 1;
413*4882a593Smuzhiyun 	for (shift = 5; byte & 1; shift += 7) {
414*4882a593Smuzhiyun 		if (offs > 9)
415*4882a593Smuzhiyun 			return INTEL_PT_BAD_PACKET;
416*4882a593Smuzhiyun 		if (len < offs)
417*4882a593Smuzhiyun 			return INTEL_PT_NEED_MORE_BYTES;
418*4882a593Smuzhiyun 		byte = buf[offs++];
419*4882a593Smuzhiyun 		payload |= ((uint64_t)byte >> 1) << shift;
420*4882a593Smuzhiyun 	}
421*4882a593Smuzhiyun 
422*4882a593Smuzhiyun 	packet->type = INTEL_PT_CYC;
423*4882a593Smuzhiyun 	packet->payload = payload;
424*4882a593Smuzhiyun 	return offs;
425*4882a593Smuzhiyun }
426*4882a593Smuzhiyun 
intel_pt_get_ip(enum intel_pt_pkt_type type,unsigned int byte,const unsigned char * buf,size_t len,struct intel_pt_pkt * packet)427*4882a593Smuzhiyun static int intel_pt_get_ip(enum intel_pt_pkt_type type, unsigned int byte,
428*4882a593Smuzhiyun 			   const unsigned char *buf, size_t len,
429*4882a593Smuzhiyun 			   struct intel_pt_pkt *packet)
430*4882a593Smuzhiyun {
431*4882a593Smuzhiyun 	int ip_len;
432*4882a593Smuzhiyun 
433*4882a593Smuzhiyun 	packet->count = byte >> 5;
434*4882a593Smuzhiyun 
435*4882a593Smuzhiyun 	switch (packet->count) {
436*4882a593Smuzhiyun 	case 0:
437*4882a593Smuzhiyun 		ip_len = 0;
438*4882a593Smuzhiyun 		break;
439*4882a593Smuzhiyun 	case 1:
440*4882a593Smuzhiyun 		if (len < 3)
441*4882a593Smuzhiyun 			return INTEL_PT_NEED_MORE_BYTES;
442*4882a593Smuzhiyun 		ip_len = 2;
443*4882a593Smuzhiyun 		packet->payload = le16_to_cpu(*(uint16_t *)(buf + 1));
444*4882a593Smuzhiyun 		break;
445*4882a593Smuzhiyun 	case 2:
446*4882a593Smuzhiyun 		if (len < 5)
447*4882a593Smuzhiyun 			return INTEL_PT_NEED_MORE_BYTES;
448*4882a593Smuzhiyun 		ip_len = 4;
449*4882a593Smuzhiyun 		packet->payload = le32_to_cpu(*(uint32_t *)(buf + 1));
450*4882a593Smuzhiyun 		break;
451*4882a593Smuzhiyun 	case 3:
452*4882a593Smuzhiyun 	case 4:
453*4882a593Smuzhiyun 		if (len < 7)
454*4882a593Smuzhiyun 			return INTEL_PT_NEED_MORE_BYTES;
455*4882a593Smuzhiyun 		ip_len = 6;
456*4882a593Smuzhiyun 		memcpy_le64(&packet->payload, buf + 1, 6);
457*4882a593Smuzhiyun 		break;
458*4882a593Smuzhiyun 	case 6:
459*4882a593Smuzhiyun 		if (len < 9)
460*4882a593Smuzhiyun 			return INTEL_PT_NEED_MORE_BYTES;
461*4882a593Smuzhiyun 		ip_len = 8;
462*4882a593Smuzhiyun 		packet->payload = le64_to_cpu(*(uint64_t *)(buf + 1));
463*4882a593Smuzhiyun 		break;
464*4882a593Smuzhiyun 	default:
465*4882a593Smuzhiyun 		return INTEL_PT_BAD_PACKET;
466*4882a593Smuzhiyun 	}
467*4882a593Smuzhiyun 
468*4882a593Smuzhiyun 	packet->type = type;
469*4882a593Smuzhiyun 
470*4882a593Smuzhiyun 	return ip_len + 1;
471*4882a593Smuzhiyun }
472*4882a593Smuzhiyun 
intel_pt_get_mode(const unsigned char * buf,size_t len,struct intel_pt_pkt * packet)473*4882a593Smuzhiyun static int intel_pt_get_mode(const unsigned char *buf, size_t len,
474*4882a593Smuzhiyun 			     struct intel_pt_pkt *packet)
475*4882a593Smuzhiyun {
476*4882a593Smuzhiyun 	if (len < 2)
477*4882a593Smuzhiyun 		return INTEL_PT_NEED_MORE_BYTES;
478*4882a593Smuzhiyun 
479*4882a593Smuzhiyun 	switch (buf[1] >> 5) {
480*4882a593Smuzhiyun 	case 0:
481*4882a593Smuzhiyun 		packet->type = INTEL_PT_MODE_EXEC;
482*4882a593Smuzhiyun 		switch (buf[1] & 3) {
483*4882a593Smuzhiyun 		case 0:
484*4882a593Smuzhiyun 			packet->payload = 16;
485*4882a593Smuzhiyun 			break;
486*4882a593Smuzhiyun 		case 1:
487*4882a593Smuzhiyun 			packet->payload = 64;
488*4882a593Smuzhiyun 			break;
489*4882a593Smuzhiyun 		case 2:
490*4882a593Smuzhiyun 			packet->payload = 32;
491*4882a593Smuzhiyun 			break;
492*4882a593Smuzhiyun 		default:
493*4882a593Smuzhiyun 			return INTEL_PT_BAD_PACKET;
494*4882a593Smuzhiyun 		}
495*4882a593Smuzhiyun 		break;
496*4882a593Smuzhiyun 	case 1:
497*4882a593Smuzhiyun 		packet->type = INTEL_PT_MODE_TSX;
498*4882a593Smuzhiyun 		if ((buf[1] & 3) == 3)
499*4882a593Smuzhiyun 			return INTEL_PT_BAD_PACKET;
500*4882a593Smuzhiyun 		packet->payload = buf[1] & 3;
501*4882a593Smuzhiyun 		break;
502*4882a593Smuzhiyun 	default:
503*4882a593Smuzhiyun 		return INTEL_PT_BAD_PACKET;
504*4882a593Smuzhiyun 	}
505*4882a593Smuzhiyun 
506*4882a593Smuzhiyun 	return 2;
507*4882a593Smuzhiyun }
508*4882a593Smuzhiyun 
intel_pt_get_tsc(const unsigned char * buf,size_t len,struct intel_pt_pkt * packet)509*4882a593Smuzhiyun static int intel_pt_get_tsc(const unsigned char *buf, size_t len,
510*4882a593Smuzhiyun 			    struct intel_pt_pkt *packet)
511*4882a593Smuzhiyun {
512*4882a593Smuzhiyun 	if (len < 8)
513*4882a593Smuzhiyun 		return INTEL_PT_NEED_MORE_BYTES;
514*4882a593Smuzhiyun 	packet->type = INTEL_PT_TSC;
515*4882a593Smuzhiyun 	memcpy_le64(&packet->payload, buf + 1, 7);
516*4882a593Smuzhiyun 	return 8;
517*4882a593Smuzhiyun }
518*4882a593Smuzhiyun 
intel_pt_get_mtc(const unsigned char * buf,size_t len,struct intel_pt_pkt * packet)519*4882a593Smuzhiyun static int intel_pt_get_mtc(const unsigned char *buf, size_t len,
520*4882a593Smuzhiyun 			    struct intel_pt_pkt *packet)
521*4882a593Smuzhiyun {
522*4882a593Smuzhiyun 	if (len < 2)
523*4882a593Smuzhiyun 		return INTEL_PT_NEED_MORE_BYTES;
524*4882a593Smuzhiyun 	packet->type = INTEL_PT_MTC;
525*4882a593Smuzhiyun 	packet->payload = buf[1];
526*4882a593Smuzhiyun 	return 2;
527*4882a593Smuzhiyun }
528*4882a593Smuzhiyun 
intel_pt_do_get_packet(const unsigned char * buf,size_t len,struct intel_pt_pkt * packet,enum intel_pt_pkt_ctx ctx)529*4882a593Smuzhiyun static int intel_pt_do_get_packet(const unsigned char *buf, size_t len,
530*4882a593Smuzhiyun 				  struct intel_pt_pkt *packet,
531*4882a593Smuzhiyun 				  enum intel_pt_pkt_ctx ctx)
532*4882a593Smuzhiyun {
533*4882a593Smuzhiyun 	unsigned int byte;
534*4882a593Smuzhiyun 
535*4882a593Smuzhiyun 	memset(packet, 0, sizeof(struct intel_pt_pkt));
536*4882a593Smuzhiyun 
537*4882a593Smuzhiyun 	if (!len)
538*4882a593Smuzhiyun 		return INTEL_PT_NEED_MORE_BYTES;
539*4882a593Smuzhiyun 
540*4882a593Smuzhiyun 	byte = buf[0];
541*4882a593Smuzhiyun 
542*4882a593Smuzhiyun 	switch (ctx) {
543*4882a593Smuzhiyun 	case INTEL_PT_NO_CTX:
544*4882a593Smuzhiyun 		break;
545*4882a593Smuzhiyun 	case INTEL_PT_BLK_4_CTX:
546*4882a593Smuzhiyun 		if ((byte & 0x7) == 4)
547*4882a593Smuzhiyun 			return intel_pt_get_bip_4(buf, len, packet);
548*4882a593Smuzhiyun 		break;
549*4882a593Smuzhiyun 	case INTEL_PT_BLK_8_CTX:
550*4882a593Smuzhiyun 		if ((byte & 0x7) == 4)
551*4882a593Smuzhiyun 			return intel_pt_get_bip_8(buf, len, packet);
552*4882a593Smuzhiyun 		break;
553*4882a593Smuzhiyun 	default:
554*4882a593Smuzhiyun 		break;
555*4882a593Smuzhiyun 	}
556*4882a593Smuzhiyun 
557*4882a593Smuzhiyun 	if (!(byte & BIT(0))) {
558*4882a593Smuzhiyun 		if (byte == 0)
559*4882a593Smuzhiyun 			return intel_pt_get_pad(packet);
560*4882a593Smuzhiyun 		if (byte == 2)
561*4882a593Smuzhiyun 			return intel_pt_get_ext(buf, len, packet);
562*4882a593Smuzhiyun 		return intel_pt_get_short_tnt(byte, packet);
563*4882a593Smuzhiyun 	}
564*4882a593Smuzhiyun 
565*4882a593Smuzhiyun 	if ((byte & 2))
566*4882a593Smuzhiyun 		return intel_pt_get_cyc(byte, buf, len, packet);
567*4882a593Smuzhiyun 
568*4882a593Smuzhiyun 	switch (byte & 0x1f) {
569*4882a593Smuzhiyun 	case 0x0D:
570*4882a593Smuzhiyun 		return intel_pt_get_ip(INTEL_PT_TIP, byte, buf, len, packet);
571*4882a593Smuzhiyun 	case 0x11:
572*4882a593Smuzhiyun 		return intel_pt_get_ip(INTEL_PT_TIP_PGE, byte, buf, len,
573*4882a593Smuzhiyun 				       packet);
574*4882a593Smuzhiyun 	case 0x01:
575*4882a593Smuzhiyun 		return intel_pt_get_ip(INTEL_PT_TIP_PGD, byte, buf, len,
576*4882a593Smuzhiyun 				       packet);
577*4882a593Smuzhiyun 	case 0x1D:
578*4882a593Smuzhiyun 		return intel_pt_get_ip(INTEL_PT_FUP, byte, buf, len, packet);
579*4882a593Smuzhiyun 	case 0x19:
580*4882a593Smuzhiyun 		switch (byte) {
581*4882a593Smuzhiyun 		case 0x99:
582*4882a593Smuzhiyun 			return intel_pt_get_mode(buf, len, packet);
583*4882a593Smuzhiyun 		case 0x19:
584*4882a593Smuzhiyun 			return intel_pt_get_tsc(buf, len, packet);
585*4882a593Smuzhiyun 		case 0x59:
586*4882a593Smuzhiyun 			return intel_pt_get_mtc(buf, len, packet);
587*4882a593Smuzhiyun 		default:
588*4882a593Smuzhiyun 			return INTEL_PT_BAD_PACKET;
589*4882a593Smuzhiyun 		}
590*4882a593Smuzhiyun 	default:
591*4882a593Smuzhiyun 		return INTEL_PT_BAD_PACKET;
592*4882a593Smuzhiyun 	}
593*4882a593Smuzhiyun }
594*4882a593Smuzhiyun 
intel_pt_upd_pkt_ctx(const struct intel_pt_pkt * packet,enum intel_pt_pkt_ctx * ctx)595*4882a593Smuzhiyun void intel_pt_upd_pkt_ctx(const struct intel_pt_pkt *packet,
596*4882a593Smuzhiyun 			  enum intel_pt_pkt_ctx *ctx)
597*4882a593Smuzhiyun {
598*4882a593Smuzhiyun 	switch (packet->type) {
599*4882a593Smuzhiyun 	case INTEL_PT_BAD:
600*4882a593Smuzhiyun 	case INTEL_PT_PAD:
601*4882a593Smuzhiyun 	case INTEL_PT_TSC:
602*4882a593Smuzhiyun 	case INTEL_PT_TMA:
603*4882a593Smuzhiyun 	case INTEL_PT_MTC:
604*4882a593Smuzhiyun 	case INTEL_PT_FUP:
605*4882a593Smuzhiyun 	case INTEL_PT_CYC:
606*4882a593Smuzhiyun 	case INTEL_PT_CBR:
607*4882a593Smuzhiyun 	case INTEL_PT_MNT:
608*4882a593Smuzhiyun 	case INTEL_PT_EXSTOP:
609*4882a593Smuzhiyun 	case INTEL_PT_EXSTOP_IP:
610*4882a593Smuzhiyun 	case INTEL_PT_PWRE:
611*4882a593Smuzhiyun 	case INTEL_PT_PWRX:
612*4882a593Smuzhiyun 	case INTEL_PT_BIP:
613*4882a593Smuzhiyun 		break;
614*4882a593Smuzhiyun 	case INTEL_PT_TNT:
615*4882a593Smuzhiyun 	case INTEL_PT_TIP:
616*4882a593Smuzhiyun 	case INTEL_PT_TIP_PGD:
617*4882a593Smuzhiyun 	case INTEL_PT_TIP_PGE:
618*4882a593Smuzhiyun 	case INTEL_PT_MODE_EXEC:
619*4882a593Smuzhiyun 	case INTEL_PT_MODE_TSX:
620*4882a593Smuzhiyun 	case INTEL_PT_PIP:
621*4882a593Smuzhiyun 	case INTEL_PT_OVF:
622*4882a593Smuzhiyun 	case INTEL_PT_VMCS:
623*4882a593Smuzhiyun 	case INTEL_PT_TRACESTOP:
624*4882a593Smuzhiyun 	case INTEL_PT_PSB:
625*4882a593Smuzhiyun 	case INTEL_PT_PSBEND:
626*4882a593Smuzhiyun 	case INTEL_PT_PTWRITE:
627*4882a593Smuzhiyun 	case INTEL_PT_PTWRITE_IP:
628*4882a593Smuzhiyun 	case INTEL_PT_MWAIT:
629*4882a593Smuzhiyun 	case INTEL_PT_BEP:
630*4882a593Smuzhiyun 	case INTEL_PT_BEP_IP:
631*4882a593Smuzhiyun 		*ctx = INTEL_PT_NO_CTX;
632*4882a593Smuzhiyun 		break;
633*4882a593Smuzhiyun 	case INTEL_PT_BBP:
634*4882a593Smuzhiyun 		if (packet->count)
635*4882a593Smuzhiyun 			*ctx = INTEL_PT_BLK_4_CTX;
636*4882a593Smuzhiyun 		else
637*4882a593Smuzhiyun 			*ctx = INTEL_PT_BLK_8_CTX;
638*4882a593Smuzhiyun 		break;
639*4882a593Smuzhiyun 	default:
640*4882a593Smuzhiyun 		break;
641*4882a593Smuzhiyun 	}
642*4882a593Smuzhiyun }
643*4882a593Smuzhiyun 
intel_pt_get_packet(const unsigned char * buf,size_t len,struct intel_pt_pkt * packet,enum intel_pt_pkt_ctx * ctx)644*4882a593Smuzhiyun int intel_pt_get_packet(const unsigned char *buf, size_t len,
645*4882a593Smuzhiyun 			struct intel_pt_pkt *packet, enum intel_pt_pkt_ctx *ctx)
646*4882a593Smuzhiyun {
647*4882a593Smuzhiyun 	int ret;
648*4882a593Smuzhiyun 
649*4882a593Smuzhiyun 	ret = intel_pt_do_get_packet(buf, len, packet, *ctx);
650*4882a593Smuzhiyun 	if (ret > 0) {
651*4882a593Smuzhiyun 		while (ret < 8 && len > (size_t)ret && !buf[ret])
652*4882a593Smuzhiyun 			ret += 1;
653*4882a593Smuzhiyun 		intel_pt_upd_pkt_ctx(packet, ctx);
654*4882a593Smuzhiyun 	}
655*4882a593Smuzhiyun 	return ret;
656*4882a593Smuzhiyun }
657*4882a593Smuzhiyun 
intel_pt_pkt_desc(const struct intel_pt_pkt * packet,char * buf,size_t buf_len)658*4882a593Smuzhiyun int intel_pt_pkt_desc(const struct intel_pt_pkt *packet, char *buf,
659*4882a593Smuzhiyun 		      size_t buf_len)
660*4882a593Smuzhiyun {
661*4882a593Smuzhiyun 	int ret, i, nr;
662*4882a593Smuzhiyun 	unsigned long long payload = packet->payload;
663*4882a593Smuzhiyun 	const char *name = intel_pt_pkt_name(packet->type);
664*4882a593Smuzhiyun 
665*4882a593Smuzhiyun 	switch (packet->type) {
666*4882a593Smuzhiyun 	case INTEL_PT_BAD:
667*4882a593Smuzhiyun 	case INTEL_PT_PAD:
668*4882a593Smuzhiyun 	case INTEL_PT_PSB:
669*4882a593Smuzhiyun 	case INTEL_PT_PSBEND:
670*4882a593Smuzhiyun 	case INTEL_PT_TRACESTOP:
671*4882a593Smuzhiyun 	case INTEL_PT_OVF:
672*4882a593Smuzhiyun 		return snprintf(buf, buf_len, "%s", name);
673*4882a593Smuzhiyun 	case INTEL_PT_TNT: {
674*4882a593Smuzhiyun 		size_t blen = buf_len;
675*4882a593Smuzhiyun 
676*4882a593Smuzhiyun 		ret = snprintf(buf, blen, "%s ", name);
677*4882a593Smuzhiyun 		if (ret < 0)
678*4882a593Smuzhiyun 			return ret;
679*4882a593Smuzhiyun 		buf += ret;
680*4882a593Smuzhiyun 		blen -= ret;
681*4882a593Smuzhiyun 		for (i = 0; i < packet->count; i++) {
682*4882a593Smuzhiyun 			if (payload & BIT63)
683*4882a593Smuzhiyun 				ret = snprintf(buf, blen, "T");
684*4882a593Smuzhiyun 			else
685*4882a593Smuzhiyun 				ret = snprintf(buf, blen, "N");
686*4882a593Smuzhiyun 			if (ret < 0)
687*4882a593Smuzhiyun 				return ret;
688*4882a593Smuzhiyun 			buf += ret;
689*4882a593Smuzhiyun 			blen -= ret;
690*4882a593Smuzhiyun 			payload <<= 1;
691*4882a593Smuzhiyun 		}
692*4882a593Smuzhiyun 		ret = snprintf(buf, blen, " (%d)", packet->count);
693*4882a593Smuzhiyun 		if (ret < 0)
694*4882a593Smuzhiyun 			return ret;
695*4882a593Smuzhiyun 		blen -= ret;
696*4882a593Smuzhiyun 		return buf_len - blen;
697*4882a593Smuzhiyun 	}
698*4882a593Smuzhiyun 	case INTEL_PT_TIP_PGD:
699*4882a593Smuzhiyun 	case INTEL_PT_TIP_PGE:
700*4882a593Smuzhiyun 	case INTEL_PT_TIP:
701*4882a593Smuzhiyun 	case INTEL_PT_FUP:
702*4882a593Smuzhiyun 		if (!(packet->count))
703*4882a593Smuzhiyun 			return snprintf(buf, buf_len, "%s no ip", name);
704*4882a593Smuzhiyun 		__fallthrough;
705*4882a593Smuzhiyun 	case INTEL_PT_CYC:
706*4882a593Smuzhiyun 	case INTEL_PT_VMCS:
707*4882a593Smuzhiyun 	case INTEL_PT_MTC:
708*4882a593Smuzhiyun 	case INTEL_PT_MNT:
709*4882a593Smuzhiyun 	case INTEL_PT_CBR:
710*4882a593Smuzhiyun 	case INTEL_PT_TSC:
711*4882a593Smuzhiyun 		return snprintf(buf, buf_len, "%s 0x%llx", name, payload);
712*4882a593Smuzhiyun 	case INTEL_PT_TMA:
713*4882a593Smuzhiyun 		return snprintf(buf, buf_len, "%s CTC 0x%x FC 0x%x", name,
714*4882a593Smuzhiyun 				(unsigned)payload, packet->count);
715*4882a593Smuzhiyun 	case INTEL_PT_MODE_EXEC:
716*4882a593Smuzhiyun 		return snprintf(buf, buf_len, "%s %lld", name, payload);
717*4882a593Smuzhiyun 	case INTEL_PT_MODE_TSX:
718*4882a593Smuzhiyun 		return snprintf(buf, buf_len, "%s TXAbort:%u InTX:%u",
719*4882a593Smuzhiyun 				name, (unsigned)(payload >> 1) & 1,
720*4882a593Smuzhiyun 				(unsigned)payload & 1);
721*4882a593Smuzhiyun 	case INTEL_PT_PIP:
722*4882a593Smuzhiyun 		nr = packet->payload & NR_FLAG ? 1 : 0;
723*4882a593Smuzhiyun 		payload &= ~NR_FLAG;
724*4882a593Smuzhiyun 		ret = snprintf(buf, buf_len, "%s 0x%llx (NR=%d)",
725*4882a593Smuzhiyun 			       name, payload, nr);
726*4882a593Smuzhiyun 		return ret;
727*4882a593Smuzhiyun 	case INTEL_PT_PTWRITE:
728*4882a593Smuzhiyun 		return snprintf(buf, buf_len, "%s 0x%llx IP:0", name, payload);
729*4882a593Smuzhiyun 	case INTEL_PT_PTWRITE_IP:
730*4882a593Smuzhiyun 		return snprintf(buf, buf_len, "%s 0x%llx IP:1", name, payload);
731*4882a593Smuzhiyun 	case INTEL_PT_BEP:
732*4882a593Smuzhiyun 	case INTEL_PT_EXSTOP:
733*4882a593Smuzhiyun 		return snprintf(buf, buf_len, "%s IP:0", name);
734*4882a593Smuzhiyun 	case INTEL_PT_BEP_IP:
735*4882a593Smuzhiyun 	case INTEL_PT_EXSTOP_IP:
736*4882a593Smuzhiyun 		return snprintf(buf, buf_len, "%s IP:1", name);
737*4882a593Smuzhiyun 	case INTEL_PT_MWAIT:
738*4882a593Smuzhiyun 		return snprintf(buf, buf_len, "%s 0x%llx Hints 0x%x Extensions 0x%x",
739*4882a593Smuzhiyun 				name, payload, (unsigned int)(payload & 0xff),
740*4882a593Smuzhiyun 				(unsigned int)((payload >> 32) & 0x3));
741*4882a593Smuzhiyun 	case INTEL_PT_PWRE:
742*4882a593Smuzhiyun 		return snprintf(buf, buf_len, "%s 0x%llx HW:%u CState:%u Sub-CState:%u",
743*4882a593Smuzhiyun 				name, payload, !!(payload & 0x80),
744*4882a593Smuzhiyun 				(unsigned int)((payload >> 12) & 0xf),
745*4882a593Smuzhiyun 				(unsigned int)((payload >> 8) & 0xf));
746*4882a593Smuzhiyun 	case INTEL_PT_PWRX:
747*4882a593Smuzhiyun 		return snprintf(buf, buf_len, "%s 0x%llx Last CState:%u Deepest CState:%u Wake Reason 0x%x",
748*4882a593Smuzhiyun 				name, payload,
749*4882a593Smuzhiyun 				(unsigned int)((payload >> 4) & 0xf),
750*4882a593Smuzhiyun 				(unsigned int)(payload & 0xf),
751*4882a593Smuzhiyun 				(unsigned int)((payload >> 8) & 0xf));
752*4882a593Smuzhiyun 	case INTEL_PT_BBP:
753*4882a593Smuzhiyun 		return snprintf(buf, buf_len, "%s SZ %s-byte Type 0x%llx",
754*4882a593Smuzhiyun 				name, packet->count ? "4" : "8", payload);
755*4882a593Smuzhiyun 	case INTEL_PT_BIP:
756*4882a593Smuzhiyun 		return snprintf(buf, buf_len, "%s ID 0x%02x Value 0x%llx",
757*4882a593Smuzhiyun 				name, packet->count, payload);
758*4882a593Smuzhiyun 	default:
759*4882a593Smuzhiyun 		break;
760*4882a593Smuzhiyun 	}
761*4882a593Smuzhiyun 	return snprintf(buf, buf_len, "%s 0x%llx (%d)",
762*4882a593Smuzhiyun 			name, payload, packet->count);
763*4882a593Smuzhiyun }
764