-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathtest_prb.c
412 lines (330 loc) · 9.39 KB
/
test_prb.c
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
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
// SPDX-License-Identifier: GPL-2.0
#include <linux/init.h>
#include <linux/module.h>
#include <linux/kthread.h>
#include <linux/delay.h>
#include <linux/random.h>
#include <linux/slab.h>
#include <linux/wait.h>
#include <linux/sched/clock.h>
#include "printk_ringbuffer.h"
/*
* This is a test module that starts "num_online_cpus()-1" writer threads
* that each write data of varying length. They do this as fast as
* they can. There is also 1 reader thread reading as fast as it can.
*
* Because the threads are running in such tight loops, they will call
* schedule() from time to time so the system stays alive.
*
* If the writers or reader encounter an error, the test is aborted. Test
* results are recorded to the ftrace buffers, with some additional
* information also provided via printk. The test can be aborted manually
* by removing the module. (Ideally the test should never abort on its own.)
*/
/* used by writers to signal reader of new records */
DECLARE_WAIT_QUEUE_HEAD(test_wait);
/* test data structure */
struct rbdata {
int len;
char text[0];
};
static char *test_running;
static int halt_test;
/* dump text data to the trace buffers */
static void print_record(const char *name, struct rbdata *dat, u64 seq, size_t text_len)
{
char buf[160];
snprintf(buf, sizeof(buf), "%s", dat->text);
buf[sizeof(buf) - 1] = 0;
trace_printk("seq=%llu len=%d %s%sval=%s\n",
seq, dat->len,
text_len > dat->len + sizeof(struct rbdata) + 1 ? "EXT " : "",
name, dat->len < sizeof(buf) ? buf : "<invalid>");
}
static bool check_data(struct rbdata *dat, u64 seq, unsigned long num)
{
static int allow_errors = 1;
int len;
len = strnlen(dat->text, 160);
if (len != dat->len || len >= 160) {
if (--allow_errors == 0)
WRITE_ONCE(halt_test, 1);
trace_printk("reader%lu invalid len for %llu (%d<->%d/0x%x)\n",
num, seq, len, dat->len, dat->len);
return false;
}
while (len) {
len--;
if (dat->text[len] != dat->text[0]) {
if (--allow_errors == 0)
WRITE_ONCE(halt_test, 1);
trace_printk("reader%lu bad data\n", num);
return false;
}
}
return true;
}
static bool printable(char *data)
{
char c;
int i;
c = data[0];
for (i = 0; i < 8; i++) {
if (data[i] < 'A' || data[i] > 'Z')
return false;
if (data[i] != c)
return false;
}
return true;
}
static void raw_dump(struct printk_ringbuffer *rb)
{
struct prb_desc_ring *der = &rb->desc_ring;
struct prb_data_ring *dar = &rb->text_data_ring;
struct prb_desc *d;
unsigned long *l;
char *data;
int i;
trace_printk("BEGIN raw dump\n");
trace_printk("BEGIN desc_ring\n");
for (i = 0; i < (1 << der->count_bits); i++) {
d = &der->descs[i];
trace_printk("%05d: sv=%016lx begin=%016lx next=%016lx\n",
i, atomic_long_read(&d->state_var),
d->text_blk_lpos.begin,
d->text_blk_lpos.next);
}
trace_printk("END desc_ring\n");
trace_printk("BEGIN text_data_ring\n");
for (i = 0; i < (1 << dar->size_bits); i += 8) {
data = &dar->data[i];
if (printable(data)) {
trace_printk("%04x: %c%c%c%c%c%c%c%c (%02x)\n", i,
data[0], data[1], data[2], data[3],
data[4], data[5], data[6], data[7],
data[0]);
} else {
l = (unsigned long *)data;
trace_printk("%04x: %016lx\n", i, *l);
}
}
trace_printk("END text_data_ring\n");
trace_printk("END raw dump\n");
}
/*
* sequentially dump all the valid records in the ringbuffer
* (used to verify memory integrity)
*
* Since there is no reader interface, the internal members are
* directly accessed. This function is called after all writers
* are finished so there is no need for any memory barriers.
*/
static void dump_rb(struct printk_ringbuffer *rb)
{
struct printk_info info;
struct printk_record r;
char text_buf[200];
u64 seq = 0;
prb_rec_init_rd(&r, &info, &text_buf[0], sizeof(text_buf));
trace_printk("BEGIN full dump\n");
while (prb_read_valid(rb, seq, &r)) {
/* check/track the sequence */
if (info.seq != seq)
trace_printk("DROPPED %llu\n", info.seq - seq);
if (!check_data((struct rbdata *)&r.text_buf[0], info.seq, 7))
trace_printk("*** BAD ***\n");
print_record("TEXT", (struct rbdata *)&r.text_buf[0],
info.seq, info.text_len);
seq = info.seq + 1;
}
trace_printk("END full dump\n");
raw_dump(rb);
}
DEFINE_PRINTKRB(test_rb, 10, 5);
static int prbtest_writer(void *data)
{
unsigned long num = (unsigned long)data;
struct prb_reserved_entry e;
char text_id = 'A' + num;
unsigned long count = 0;
struct printk_record r;
u64 min_ns = (u64)-1;
struct rbdata *dat;
u64 total_ns = 0;
u64 max_ns = 0;
u64 post_ns;
u64 pre_ns;
u64 seq;
int len;
set_cpus_allowed_ptr(current, cpumask_of(num));
pr_err("prbtest: start thread %03lu (writer)\n", num);
for (;;) {
len = sizeof(struct rbdata) + (prandom_u32() & 0x7f) + 2;
/* specify the text size for reservation */
prb_rec_init_wr(&r, len);
pre_ns = local_clock();
if (prb_reserve(&e, &test_rb, &r)) {
dat = (struct rbdata *)&r.text_buf[0];
dat->len = len - sizeof(struct rbdata) - 1;
memset(&dat->text[0], text_id, dat->len);
dat->text[dat->len] = 0;
r.info->text_len = len;
r.info->caller_id = num + 1048576;
seq = r.info->seq;
prb_commit(&e);
post_ns = local_clock();
/* append another struct */
prb_rec_init_wr(&r, len);
if (prb_reserve_in_last(&e, &test_rb, &r, num + 1048576)) {
if (r.info->seq != seq) {
trace_printk("writer%lu (%c) unexpected seq: %llu != %llu\n",
num, text_id, r.info->seq, seq);
}
if (r.info->text_len != len) {
trace_printk("writer%lu (%c) unexpected text_len: %u != %u\n",
num, text_id, r.info->text_len, len);
}
dat = (struct rbdata *)&r.text_buf[len];
dat->len = len - sizeof(struct rbdata) - 1;
memset(&dat->text[0], text_id, dat->len);
dat->text[dat->len] = 0;
r.info->text_len += len;
prb_commit(&e);
}
wake_up_interruptible(&test_wait);
post_ns -= pre_ns;
if (post_ns < min_ns)
min_ns = post_ns;
if (post_ns > max_ns)
max_ns = post_ns;
total_ns += post_ns;
}
if ((count++ & 0x3fff) == 0)
schedule();
if (READ_ONCE(halt_test) == 1)
break;
}
/* change @total_ns to average */
do_div(total_ns, count);
pr_err("prbtest: end thread %03lu (wrote %lu, max/avg/min %llu/%llu/%llu)\n",
num, count, max_ns, total_ns, min_ns);
test_running[num] = 0;
return 0;
}
static int prbtest_reader(void *data)
{
unsigned long num = (unsigned long)data;
unsigned long total_lost = 0;
unsigned long max_lost = 0;
unsigned long count = 0;
struct printk_info info;
struct printk_record r;
struct rbdata *dat;
char text_buf[400];
int did_sched = 1;
u64 seq = 0;
set_cpus_allowed_ptr(current, cpumask_of(num));
prb_rec_init_rd(&r, &info, &text_buf[0], sizeof(text_buf));
pr_err("prbtest: start thread %03lu (reader)\n", num);
while (!wait_event_interruptible(test_wait,
kthread_should_stop() ||
prb_read_valid(&test_rb, seq, &r))) {
if (kthread_should_stop())
break;
/* check/track the sequence */
if (info.seq < seq) {
WRITE_ONCE(halt_test, 1);
trace_printk("reader%lu invalid seq %llu -> %llu\n",
num, seq, info.seq);
} else if (info.seq != seq && !did_sched) {
total_lost += info.seq - seq;
if (max_lost < info.seq - seq)
max_lost = info.seq - seq;
}
dat = (struct rbdata *)&r.text_buf[0];
if (!check_data(dat, info.seq, num))
trace_printk("text error\n");
if (info.text_len > dat->len + sizeof(struct rbdata) + 1) {
dat = (struct rbdata *)&r.text_buf[dat->len + sizeof(struct rbdata) + 1];
if (!check_data(dat, info.seq, num))
trace_printk("text extension error\n");
}
did_sched = 0;
if ((count++ & 0x3fff) == 0) {
did_sched = 1;
schedule();
}
if (READ_ONCE(halt_test) == 1)
break;
seq = info.seq + 1;
}
pr_err(
"reader%lu: total_lost=%lu max_lost=%lu total_read=%lu seq=%llu\n",
num, total_lost, max_lost, count, info.seq);
pr_err("prbtest: end thread %03lu (reader)\n", num);
while (!kthread_should_stop())
msleep(1000);
test_running[num] = 0;
return 0;
}
static int module_test_running;
static struct task_struct *reader_thread;
static int start_test(void *arg)
{
struct task_struct *thread;
unsigned long i;
int num_cpus;
num_cpus = num_online_cpus();
test_running = kzalloc(num_cpus, GFP_KERNEL);
if (!test_running)
return -ENOMEM;
module_test_running = 1;
pr_err("prbtest: starting test\n");
for (i = 0; i < num_cpus; i++) {
test_running[i] = 1;
if (i < num_cpus - 1) {
thread = kthread_run(prbtest_writer, (void *)i,
"prbtest writer");
} else {
thread = kthread_run(prbtest_reader, (void *)i,
"prbtest reader");
reader_thread = thread;
}
if (IS_ERR(thread)) {
pr_err("prbtest: unable to create thread %lu\n", i);
test_running[i] = 0;
}
}
for (;;) {
msleep(1000);
for (i = 0; i < num_cpus; i++) {
if (test_running[i] == 1)
break;
}
if (i == num_cpus)
break;
}
pr_err("prbtest: completed test\n");
dump_rb(&test_rb);
module_test_running = 0;
return 0;
}
static int prbtest_init(void)
{
kthread_run(start_test, NULL, "prbtest");
return 0;
}
static void prbtest_exit(void)
{
if (reader_thread && !IS_ERR(reader_thread))
kthread_stop(reader_thread);
WRITE_ONCE(halt_test, 1);
while (module_test_running)
msleep(1000);
kfree(test_running);
}
module_init(prbtest_init);
module_exit(prbtest_exit);
MODULE_AUTHOR("John Ogness <john.ogness@linutronix.de>");
MODULE_DESCRIPTION("printk ringbuffer test");
MODULE_LICENSE("GPL v2");