summaryrefslogtreecommitdiffstats
path: root/kernel/include/trace/events/workqueue.h
blob: bf0e18ba6cfb0d062b5c0b15a98dff854a5d2ad9 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
#undef TRACE_SYSTEM
#define TRACE_SYSTEM workqueue

#if !defined(_TRACE_WORKQUEUE_H) || defined(TRACE_HEADER_MULTI_READ)
#define _TRACE_WORKQUEUE_H

#include <linux/tracepoint.h>
#include <linux/workqueue.h>

DECLARE_EVENT_CLASS(workqueue_work,

	TP_PROTO(struct work_struct *work),

	TP_ARGS(work),

	TP_STRUCT__entry(
		__field( void *,	work	)
	),

	TP_fast_assign(
		__entry->work		= work;
	),

	TP_printk("work struct %p", __entry->work)
);

/**
 * workqueue_queue_work - called when a work gets queued
 * @req_cpu:	the requested cpu
 * @pwq:	pointer to struct pool_workqueue
 * @work:	pointer to struct work_struct
 *
 * This event occurs when a work is queued immediately or once a
 * delayed work is actually queued on a workqueue (ie: once the delay
 * has been reached).
 */
TRACE_EVENT(workqueue_queue_work,

	TP_PROTO(unsigned int req_cpu, struct pool_workqueue *pwq,
		 struct work_struct *work),

	TP_ARGS(req_cpu, pwq, work),

	TP_STRUCT__entry(
		__field( void *,	work	)
		__field( void *,	function)
		__field( void *,	workqueue)
		__field( unsigned int,	req_cpu	)
		__field( unsigned int,	cpu	)
	),

	TP_fast_assign(
		__entry->work		= work;
		__entry->function	= work->func;
		__entry->workqueue	= pwq->wq;
		__entry->req_cpu	= req_cpu;
		__entry->cpu		= pwq->pool->cpu;
	),

	TP_printk("work struct=%p function=%pf workqueue=%p req_cpu=%u cpu=%u",
		  __entry->work, __entry->function, __entry->workqueue,
		  __entry->req_cpu, __entry->cpu)
);

/**
 * workqueue_activate_work - called when a work gets activated
 * @work:	pointer to struct work_struct
 *
 * This event occurs when a queued work is put on the active queue,
 * which happens immediately after queueing unless @max_active limit
 * is reached.
 */
DEFINE_EVENT(workqueue_work, workqueue_activate_work,

	TP_PROTO(struct work_struct *work),

	TP_ARGS(work)
);

/**
 * workqueue_execute_start - called immediately before the workqueue callback
 * @work:	pointer to struct work_struct
 *
 * Allows to track workqueue execution.
 */
TRACE_EVENT(workqueue_execute_start,

	TP_PROTO(struct work_struct *work),

	TP_ARGS(work),

	TP_STRUCT__entry(
		__field( void *,	work	)
		__field( void *,	function)
	),

	TP_fast_assign(
		__entry->work		= work;
		__entry->function	= work->func;
	),

	TP_printk("work struct %p: function %pf", __entry->work, __entry->function)
);

/**
 * workqueue_execute_end - called immediately after the workqueue callback
 * @work:	pointer to struct work_struct
 *
 * Allows to track workqueue execution.
 */
DEFINE_EVENT(workqueue_work, workqueue_execute_end,

	TP_PROTO(struct work_struct *work),

	TP_ARGS(work)
);

#endif /*  _TRACE_WORKQUEUE_H */

/* This part must be outside protection */
#include <trace/define_trace.h>
class="kt">uint64_t a) { uint64_t m = 0x7f7f7f7f7f7f7f7fULL; uint64_t c = ~(((a & m) + m) | a | m); /* a.......b.......c.......d.......e.......f.......g.......h....... */ c |= c << 7; /* ab......bc......cd......de......ef......fg......gh......h....... */ c |= c << 14; /* abcd....bcde....cdef....defg....efgh....fgh.....gh......h....... */ c |= c << 28; /* abcdefghbcdefgh.cdefgh..defgh...efgh....fgh.....gh......h....... */ return c >> 56; } uint64_t helper_cmpbge(uint64_t a, uint64_t b) { uint64_t mask = 0x00ff00ff00ff00ffULL; uint64_t test = 0x0100010001000100ULL; uint64_t al, ah, bl, bh, cl, ch; /* Separate the bytes to avoid false positives. */ al = a & mask; bl = b & mask; ah = (a >> 8) & mask; bh = (b >> 8) & mask; /* "Compare". If a byte in B is greater than a byte in A, it will clear the test bit. */ cl = ((al | test) - bl) & test; ch = ((ah | test) - bh) & test; /* Fold all of the test bits into a contiguous set. */ /* ch=.......a...............c...............e...............g........ */ /* cl=.......b...............d...............f...............h........ */ cl += ch << 1; /* cl=......ab..............cd..............ef..............gh........ */ cl |= cl << 14; /* cl=......abcd............cdef............efgh............gh........ */ cl |= cl << 28; /* cl=......abcdefgh........cdefgh..........efgh............gh........ */ return cl >> 50; } uint64_t helper_minub8(uint64_t op1, uint64_t op2) { uint64_t res = 0; uint8_t opa, opb, opr; int i; for (i = 0; i < 8; ++i) { opa = op1 >> (i * 8); opb = op2 >> (i * 8); opr = opa < opb ? opa : opb; res |= (uint64_t)opr << (i * 8); } return res; } uint64_t helper_minsb8(uint64_t op1, uint64_t op2) { uint64_t res = 0; int8_t opa, opb; uint8_t opr; int i; for (i = 0; i < 8; ++i) { opa = op1 >> (i * 8); opb = op2 >> (i * 8); opr = opa < opb ? opa : opb; res |= (uint64_t)opr << (i * 8); } return res; } uint64_t helper_minuw4(uint64_t op1, uint64_t op2) { uint64_t res = 0; uint16_t opa, opb, opr; int i; for (i = 0; i < 4; ++i) { opa = op1 >> (i * 16); opb = op2 >> (i * 16); opr = opa < opb ? opa : opb; res |= (uint64_t)opr << (i * 16); } return res; } uint64_t helper_minsw4(uint64_t op1, uint64_t op2) { uint64_t res = 0; int16_t opa, opb; uint16_t opr; int i; for (i = 0; i < 4; ++i) { opa = op1 >> (i * 16); opb = op2 >> (i * 16); opr = opa < opb ? opa : opb; res |= (uint64_t)opr << (i * 16); } return res; } uint64_t helper_maxub8(uint64_t op1, uint64_t op2) { uint64_t res = 0; uint8_t opa, opb, opr; int i; for (i = 0; i < 8; ++i) { opa = op1 >> (i * 8); opb = op2 >> (i * 8); opr = opa > opb ? opa : opb; res |= (uint64_t)opr << (i * 8); } return res; } uint64_t helper_maxsb8(uint64_t op1, uint64_t op2) { uint64_t res = 0; int8_t opa, opb; uint8_t opr; int i; for (i = 0; i < 8; ++i) { opa = op1 >> (i * 8); opb = op2 >> (i * 8); opr = opa > opb ? opa : opb; res |= (uint64_t)opr << (i * 8); } return res; } uint64_t helper_maxuw4(uint64_t op1, uint64_t op2) { uint64_t res = 0; uint16_t opa, opb, opr; int i; for (i = 0; i < 4; ++i) { opa = op1 >> (i * 16); opb = op2 >> (i * 16); opr = opa > opb ? opa : opb; res |= (uint64_t)opr << (i * 16); } return res; } uint64_t helper_maxsw4(uint64_t op1, uint64_t op2) { uint64_t res = 0; int16_t opa, opb; uint16_t opr; int i; for (i = 0; i < 4; ++i) { opa = op1 >> (i * 16); opb = op2 >> (i * 16); opr = opa > opb ? opa : opb; res |= (uint64_t)opr << (i * 16); } return res; } uint64_t helper_perr(uint64_t op1, uint64_t op2) { uint64_t res = 0; uint8_t opa, opb, opr; int i; for (i = 0; i < 8; ++i) { opa = op1 >> (i * 8); opb = op2 >> (i * 8); if (opa >= opb) { opr = opa - opb; } else { opr = opb - opa; } res += opr; } return res; } uint64_t helper_pklb(uint64_t op1) { return (op1 & 0xff) | ((op1 >> 24) & 0xff00); } uint64_t helper_pkwb(uint64_t op1) { return ((op1 & 0xff) | ((op1 >> 8) & 0xff00) | ((op1 >> 16) & 0xff0000) | ((op1 >> 24) & 0xff000000)); } uint64_t helper_unpkbl(uint64_t op1) { return (op1 & 0xff) | ((op1 & 0xff00) << 24); } uint64_t helper_unpkbw(uint64_t op1) { return ((op1 & 0xff) | ((op1 & 0xff00) << 8) | ((op1 & 0xff0000) << 16) | ((op1 & 0xff000000) << 24)); } void helper_check_overflow(CPUAlphaState *env, uint64_t op1, uint64_t op2) { if (unlikely(op1 != op2)) { arith_excp(env, GETPC(), EXC_M_IOV, 0); } }