Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
godotengine
GitHub Repository: godotengine/godot
Path: blob/master/tests/core/templates/test_command_queue.h
10278 views
1
/**************************************************************************/
2
/* test_command_queue.h */
3
/**************************************************************************/
4
/* This file is part of: */
5
/* GODOT ENGINE */
6
/* https://godotengine.org */
7
/**************************************************************************/
8
/* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */
9
/* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur. */
10
/* */
11
/* Permission is hereby granted, free of charge, to any person obtaining */
12
/* a copy of this software and associated documentation files (the */
13
/* "Software"), to deal in the Software without restriction, including */
14
/* without limitation the rights to use, copy, modify, merge, publish, */
15
/* distribute, sublicense, and/or sell copies of the Software, and to */
16
/* permit persons to whom the Software is furnished to do so, subject to */
17
/* the following conditions: */
18
/* */
19
/* The above copyright notice and this permission notice shall be */
20
/* included in all copies or substantial portions of the Software. */
21
/* */
22
/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
23
/* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
24
/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. */
25
/* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
26
/* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
27
/* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
28
/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
29
/**************************************************************************/
30
31
#pragma once
32
33
#include "core/config/project_settings.h"
34
#include "core/math/random_number_generator.h"
35
#include "core/object/worker_thread_pool.h"
36
#include "core/os/os.h"
37
#include "core/os/thread.h"
38
#include "core/templates/command_queue_mt.h"
39
#include "tests/test_macros.h"
40
41
namespace TestCommandQueue {
42
43
class ThreadWork {
44
Semaphore thread_sem;
45
Semaphore main_sem;
46
Mutex mut;
47
int threading_errors = 0;
48
enum State {
49
MAIN_START,
50
MAIN_DONE,
51
THREAD_START,
52
THREAD_DONE,
53
} state;
54
55
public:
56
ThreadWork() {
57
mut.lock();
58
state = MAIN_START;
59
}
60
~ThreadWork() {
61
CHECK_MESSAGE(threading_errors == 0, "threads did not lock/unlock correctly");
62
}
63
void thread_wait_for_work() {
64
thread_sem.wait();
65
mut.lock();
66
if (state != MAIN_DONE) {
67
threading_errors++;
68
}
69
state = THREAD_START;
70
}
71
void thread_done_work() {
72
if (state != THREAD_START) {
73
threading_errors++;
74
}
75
state = THREAD_DONE;
76
mut.unlock();
77
main_sem.post();
78
}
79
80
void main_wait_for_done() {
81
main_sem.wait();
82
mut.lock();
83
if (state != THREAD_DONE) {
84
threading_errors++;
85
}
86
state = MAIN_START;
87
}
88
void main_start_work() {
89
if (state != MAIN_START) {
90
threading_errors++;
91
}
92
state = MAIN_DONE;
93
mut.unlock();
94
thread_sem.post();
95
}
96
};
97
98
class SharedThreadState {
99
public:
100
ThreadWork reader_threadwork;
101
ThreadWork writer_threadwork;
102
103
CommandQueueMT command_queue;
104
105
enum TestMsgType {
106
TEST_MSG_FUNC1_TRANSFORM,
107
TEST_MSG_FUNC2_TRANSFORM_FLOAT,
108
TEST_MSG_FUNC3_TRANSFORMx6,
109
TEST_MSGSYNC_FUNC1_TRANSFORM,
110
TEST_MSGSYNC_FUNC2_TRANSFORM_FLOAT,
111
TEST_MSGRET_FUNC1_TRANSFORM,
112
TEST_MSGRET_FUNC2_TRANSFORM_FLOAT,
113
TEST_MSG_MAX
114
};
115
116
Vector<TestMsgType> message_types_to_write;
117
bool during_writing = false;
118
int message_count_to_read = 0;
119
bool exit_threads = false;
120
121
Thread reader_thread;
122
WorkerThreadPool::TaskID reader_task_id = WorkerThreadPool::INVALID_TASK_ID;
123
Thread writer_thread;
124
125
int func1_count = 0;
126
127
void func1(Transform3D t) {
128
func1_count++;
129
}
130
void func2(Transform3D t, float f) {
131
func1_count++;
132
}
133
void func3(Transform3D t1, Transform3D t2, Transform3D t3, Transform3D t4, Transform3D t5, Transform3D t6) {
134
func1_count++;
135
}
136
Transform3D func1r(Transform3D t) {
137
func1_count++;
138
return t;
139
}
140
Transform3D func2r(Transform3D t, float f) {
141
func1_count++;
142
return t;
143
}
144
145
void add_msg_to_write(TestMsgType type) {
146
message_types_to_write.push_back(type);
147
}
148
149
void reader_thread_loop() {
150
reader_threadwork.thread_wait_for_work();
151
while (!exit_threads) {
152
if (reader_task_id == WorkerThreadPool::INVALID_TASK_ID) {
153
command_queue.flush_all();
154
} else {
155
if (message_count_to_read < 0) {
156
command_queue.flush_all();
157
}
158
for (int i = 0; i < message_count_to_read; i++) {
159
WorkerThreadPool::get_singleton()->yield();
160
command_queue.wait_and_flush();
161
}
162
}
163
message_count_to_read = 0;
164
165
reader_threadwork.thread_done_work();
166
reader_threadwork.thread_wait_for_work();
167
}
168
command_queue.flush_all();
169
reader_threadwork.thread_done_work();
170
}
171
static void static_reader_thread_loop(void *stsvoid) {
172
SharedThreadState *sts = static_cast<SharedThreadState *>(stsvoid);
173
sts->reader_thread_loop();
174
}
175
176
void writer_thread_loop() {
177
during_writing = false;
178
writer_threadwork.thread_wait_for_work();
179
while (!exit_threads) {
180
Transform3D tr;
181
Transform3D otr;
182
float f = 1;
183
during_writing = true;
184
for (int i = 0; i < message_types_to_write.size(); i++) {
185
TestMsgType msg_type = message_types_to_write[i];
186
switch (msg_type) {
187
case TEST_MSG_FUNC1_TRANSFORM:
188
command_queue.push(this, &SharedThreadState::func1, tr);
189
break;
190
case TEST_MSG_FUNC2_TRANSFORM_FLOAT:
191
command_queue.push(this, &SharedThreadState::func2, tr, f);
192
break;
193
case TEST_MSG_FUNC3_TRANSFORMx6:
194
command_queue.push(this, &SharedThreadState::func3, tr, tr, tr, tr, tr, tr);
195
break;
196
case TEST_MSGSYNC_FUNC1_TRANSFORM:
197
command_queue.push_and_sync(this, &SharedThreadState::func1, tr);
198
break;
199
case TEST_MSGSYNC_FUNC2_TRANSFORM_FLOAT:
200
command_queue.push_and_sync(this, &SharedThreadState::func2, tr, f);
201
break;
202
case TEST_MSGRET_FUNC1_TRANSFORM:
203
command_queue.push_and_ret(this, &SharedThreadState::func1r, &otr, tr);
204
break;
205
case TEST_MSGRET_FUNC2_TRANSFORM_FLOAT:
206
command_queue.push_and_ret(this, &SharedThreadState::func2r, &otr, tr, f);
207
break;
208
default:
209
break;
210
}
211
}
212
message_types_to_write.clear();
213
during_writing = false;
214
215
writer_threadwork.thread_done_work();
216
writer_threadwork.thread_wait_for_work();
217
}
218
writer_threadwork.thread_done_work();
219
}
220
static void static_writer_thread_loop(void *stsvoid) {
221
SharedThreadState *sts = static_cast<SharedThreadState *>(stsvoid);
222
sts->writer_thread_loop();
223
}
224
225
void init_threads(bool p_use_thread_pool_sync = false) {
226
if (p_use_thread_pool_sync) {
227
reader_task_id = WorkerThreadPool::get_singleton()->add_native_task(&SharedThreadState::static_reader_thread_loop, this, true);
228
command_queue.set_pump_task_id(reader_task_id);
229
} else {
230
reader_thread.start(&SharedThreadState::static_reader_thread_loop, this);
231
}
232
writer_thread.start(&SharedThreadState::static_writer_thread_loop, this);
233
}
234
void destroy_threads() {
235
exit_threads = true;
236
reader_threadwork.main_start_work();
237
writer_threadwork.main_start_work();
238
239
if (reader_task_id != WorkerThreadPool::INVALID_TASK_ID) {
240
WorkerThreadPool::get_singleton()->wait_for_task_completion(reader_task_id);
241
} else {
242
reader_thread.wait_to_finish();
243
}
244
writer_thread.wait_to_finish();
245
}
246
247
struct CopyMoveTestType {
248
inline static int copy_count;
249
inline static int move_count;
250
int value = 0;
251
252
CopyMoveTestType(int p_value = 0) :
253
value(p_value) {}
254
255
CopyMoveTestType(const CopyMoveTestType &p_other) :
256
value(p_other.value) {
257
copy_count++;
258
}
259
260
CopyMoveTestType(CopyMoveTestType &&p_other) :
261
value(p_other.value) {
262
move_count++;
263
}
264
265
CopyMoveTestType &operator=(const CopyMoveTestType &p_other) {
266
value = p_other.value;
267
copy_count++;
268
return *this;
269
}
270
271
CopyMoveTestType &operator=(CopyMoveTestType &&p_other) {
272
value = p_other.value;
273
move_count++;
274
return *this;
275
}
276
};
277
278
void copy_move_test_copy(CopyMoveTestType p_test_type) {
279
}
280
void copy_move_test_ref(const CopyMoveTestType &p_test_type) {
281
}
282
void copy_move_test_move(CopyMoveTestType &&p_test_type) {
283
}
284
};
285
286
static void test_command_queue_basic(bool p_use_thread_pool_sync) {
287
const char *COMMAND_QUEUE_SETTING = "memory/limits/command_queue/multithreading_queue_size_kb";
288
ProjectSettings::get_singleton()->set_setting(COMMAND_QUEUE_SETTING, 1);
289
SharedThreadState sts;
290
sts.init_threads(p_use_thread_pool_sync);
291
292
sts.add_msg_to_write(SharedThreadState::TEST_MSG_FUNC1_TRANSFORM);
293
sts.writer_threadwork.main_start_work();
294
sts.writer_threadwork.main_wait_for_done();
295
CHECK_MESSAGE(sts.func1_count == 0,
296
"Control: no messages read before reader has run.");
297
298
sts.message_count_to_read = 1;
299
sts.reader_threadwork.main_start_work();
300
sts.reader_threadwork.main_wait_for_done();
301
CHECK_MESSAGE(sts.func1_count == 1,
302
"Reader should have read one message");
303
304
sts.message_count_to_read = -1;
305
sts.reader_threadwork.main_start_work();
306
sts.reader_threadwork.main_wait_for_done();
307
CHECK_MESSAGE(sts.func1_count == 1,
308
"Reader should have read no additional messages from flush_all");
309
310
sts.add_msg_to_write(SharedThreadState::TEST_MSG_FUNC1_TRANSFORM);
311
sts.writer_threadwork.main_start_work();
312
sts.writer_threadwork.main_wait_for_done();
313
314
sts.message_count_to_read = -1;
315
sts.reader_threadwork.main_start_work();
316
sts.reader_threadwork.main_wait_for_done();
317
CHECK_MESSAGE(sts.func1_count == 2,
318
"Reader should have read one additional message from flush_all");
319
320
sts.destroy_threads();
321
322
CHECK_MESSAGE(sts.func1_count == 2,
323
"Reader should have read no additional messages after join");
324
ProjectSettings::get_singleton()->set_setting(COMMAND_QUEUE_SETTING,
325
ProjectSettings::get_singleton()->property_get_revert(COMMAND_QUEUE_SETTING));
326
}
327
328
TEST_CASE("[CommandQueue] Test Queue Basics") {
329
test_command_queue_basic(false);
330
}
331
332
TEST_CASE("[CommandQueue] Test Queue Basics with WorkerThreadPool sync.") {
333
test_command_queue_basic(true);
334
}
335
336
TEST_CASE("[CommandQueue] Test Queue Wrapping to same spot.") {
337
const char *COMMAND_QUEUE_SETTING = "memory/limits/command_queue/multithreading_queue_size_kb";
338
ProjectSettings::get_singleton()->set_setting(COMMAND_QUEUE_SETTING, 1);
339
SharedThreadState sts;
340
sts.init_threads();
341
342
sts.add_msg_to_write(SharedThreadState::TEST_MSG_FUNC3_TRANSFORMx6);
343
sts.add_msg_to_write(SharedThreadState::TEST_MSG_FUNC3_TRANSFORMx6);
344
sts.add_msg_to_write(SharedThreadState::TEST_MSG_FUNC1_TRANSFORM);
345
sts.writer_threadwork.main_start_work();
346
sts.writer_threadwork.main_wait_for_done();
347
348
sts.message_count_to_read = -1;
349
sts.reader_threadwork.main_start_work();
350
sts.reader_threadwork.main_wait_for_done();
351
CHECK_MESSAGE(sts.func1_count == 3,
352
"Reader should have read at least three messages");
353
354
sts.add_msg_to_write(SharedThreadState::TEST_MSG_FUNC3_TRANSFORMx6);
355
sts.writer_threadwork.main_start_work();
356
sts.writer_threadwork.main_wait_for_done();
357
sts.add_msg_to_write(SharedThreadState::TEST_MSG_FUNC1_TRANSFORM);
358
sts.add_msg_to_write(SharedThreadState::TEST_MSG_FUNC3_TRANSFORMx6);
359
sts.writer_threadwork.main_start_work();
360
OS::get_singleton()->delay_usec(1000);
361
362
sts.message_count_to_read = -1;
363
sts.reader_threadwork.main_start_work();
364
OS::get_singleton()->delay_usec(1000);
365
366
sts.writer_threadwork.main_wait_for_done();
367
sts.reader_threadwork.main_wait_for_done();
368
CHECK_MESSAGE(sts.func1_count >= 3,
369
"Reader should have read at least three messages");
370
371
sts.message_count_to_read = 6 - sts.func1_count;
372
sts.reader_threadwork.main_start_work();
373
374
// The following will fail immediately.
375
// The reason it hangs indefinitely in engine, is all subsequent calls to
376
// CommandQueue.wait_and_flush_one will also fail.
377
sts.reader_threadwork.main_wait_for_done();
378
379
// Because looping around uses an extra message, easiest to consume all.
380
sts.message_count_to_read = -1;
381
sts.reader_threadwork.main_start_work();
382
sts.reader_threadwork.main_wait_for_done();
383
CHECK_MESSAGE(sts.func1_count == 6,
384
"Reader should have read both message sets");
385
386
sts.destroy_threads();
387
388
CHECK_MESSAGE(sts.func1_count == 6,
389
"Reader should have read no additional messages after join");
390
ProjectSettings::get_singleton()->set_setting(COMMAND_QUEUE_SETTING,
391
ProjectSettings::get_singleton()->property_get_revert(COMMAND_QUEUE_SETTING));
392
}
393
394
TEST_CASE("[CommandQueue] Test Queue Lapping") {
395
const char *COMMAND_QUEUE_SETTING = "memory/limits/command_queue/multithreading_queue_size_kb";
396
ProjectSettings::get_singleton()->set_setting(COMMAND_QUEUE_SETTING, 1);
397
SharedThreadState sts;
398
sts.init_threads();
399
400
sts.add_msg_to_write(SharedThreadState::TEST_MSG_FUNC1_TRANSFORM);
401
sts.add_msg_to_write(SharedThreadState::TEST_MSG_FUNC3_TRANSFORMx6);
402
sts.add_msg_to_write(SharedThreadState::TEST_MSG_FUNC3_TRANSFORMx6);
403
sts.writer_threadwork.main_start_work();
404
sts.writer_threadwork.main_wait_for_done();
405
406
// We need to read an extra message so that it triggers the dealloc logic once.
407
// Otherwise, the queue will be considered full.
408
sts.message_count_to_read = 3;
409
sts.reader_threadwork.main_start_work();
410
sts.reader_threadwork.main_wait_for_done();
411
CHECK_MESSAGE(sts.func1_count == 3,
412
"Reader should have read first set of messages");
413
414
sts.add_msg_to_write(SharedThreadState::TEST_MSG_FUNC3_TRANSFORMx6);
415
sts.add_msg_to_write(SharedThreadState::TEST_MSG_FUNC3_TRANSFORMx6);
416
sts.writer_threadwork.main_start_work();
417
// Don't wait for these, because the queue isn't big enough.
418
sts.writer_threadwork.main_wait_for_done();
419
420
sts.add_msg_to_write(SharedThreadState::TEST_MSG_FUNC2_TRANSFORM_FLOAT);
421
sts.writer_threadwork.main_start_work();
422
OS::get_singleton()->delay_usec(1000);
423
424
sts.message_count_to_read = 3;
425
sts.reader_threadwork.main_start_work();
426
sts.reader_threadwork.main_wait_for_done();
427
428
sts.writer_threadwork.main_wait_for_done();
429
430
sts.message_count_to_read = -1;
431
sts.reader_threadwork.main_start_work();
432
sts.reader_threadwork.main_wait_for_done();
433
434
CHECK_MESSAGE(sts.func1_count == 6,
435
"Reader should have read rest of the messages after lapping writers.");
436
437
sts.destroy_threads();
438
439
CHECK_MESSAGE(sts.func1_count == 6,
440
"Reader should have read no additional messages after join");
441
ProjectSettings::get_singleton()->set_setting(COMMAND_QUEUE_SETTING,
442
ProjectSettings::get_singleton()->property_get_revert(COMMAND_QUEUE_SETTING));
443
}
444
445
TEST_CASE("[Stress][CommandQueue] Stress test command queue") {
446
const char *COMMAND_QUEUE_SETTING = "memory/limits/command_queue/multithreading_queue_size_kb";
447
ProjectSettings::get_singleton()->set_setting(COMMAND_QUEUE_SETTING, 1);
448
SharedThreadState sts;
449
sts.init_threads();
450
451
RandomNumberGenerator rng;
452
453
rng.set_seed(1837267);
454
455
int msgs_to_add = 2048;
456
457
for (int i = 0; i < msgs_to_add; i++) {
458
// randi_range is inclusive, so allow any enum value except MAX.
459
sts.add_msg_to_write((SharedThreadState::TestMsgType)rng.randi_range(0, SharedThreadState::TEST_MSG_MAX - 1));
460
}
461
sts.writer_threadwork.main_start_work();
462
463
int max_loop_iters = msgs_to_add * 2;
464
int loop_iters = 0;
465
while (sts.func1_count < msgs_to_add && loop_iters < max_loop_iters) {
466
int remaining = (msgs_to_add - sts.func1_count);
467
sts.message_count_to_read = rng.randi_range(1, remaining < 128 ? remaining : 128);
468
if (loop_iters % 3 == 0) {
469
sts.message_count_to_read = -1;
470
}
471
sts.reader_threadwork.main_start_work();
472
sts.reader_threadwork.main_wait_for_done();
473
loop_iters++;
474
}
475
CHECK_MESSAGE(loop_iters < max_loop_iters,
476
"Reader needed too many iterations to read messages!");
477
sts.writer_threadwork.main_wait_for_done();
478
479
sts.destroy_threads();
480
481
CHECK_MESSAGE(sts.func1_count == msgs_to_add,
482
"Reader should have read no additional messages after join");
483
ProjectSettings::get_singleton()->set_setting(COMMAND_QUEUE_SETTING,
484
ProjectSettings::get_singleton()->property_get_revert(COMMAND_QUEUE_SETTING));
485
}
486
487
TEST_CASE("[CommandQueue] Test Parameter Passing Semantics") {
488
SharedThreadState sts;
489
sts.init_threads();
490
491
SUBCASE("Testing with lvalue") {
492
SharedThreadState::CopyMoveTestType::copy_count = 0;
493
SharedThreadState::CopyMoveTestType::move_count = 0;
494
495
SharedThreadState::CopyMoveTestType lvalue(42);
496
497
SUBCASE("Pass by copy") {
498
sts.command_queue.push(&sts, &SharedThreadState::copy_move_test_copy, lvalue);
499
500
sts.message_count_to_read = -1;
501
sts.reader_threadwork.main_start_work();
502
sts.reader_threadwork.main_wait_for_done();
503
504
CHECK(SharedThreadState::CopyMoveTestType::copy_count == 1);
505
CHECK(SharedThreadState::CopyMoveTestType::move_count == 1);
506
}
507
508
SUBCASE("Pass by reference") {
509
sts.command_queue.push(&sts, &SharedThreadState::copy_move_test_ref, lvalue);
510
511
sts.message_count_to_read = -1;
512
sts.reader_threadwork.main_start_work();
513
sts.reader_threadwork.main_wait_for_done();
514
515
CHECK(SharedThreadState::CopyMoveTestType::copy_count == 1);
516
CHECK(SharedThreadState::CopyMoveTestType::move_count == 0);
517
}
518
}
519
520
SUBCASE("Testing with rvalue") {
521
SharedThreadState::CopyMoveTestType::copy_count = 0;
522
SharedThreadState::CopyMoveTestType::move_count = 0;
523
524
SUBCASE("Pass by copy") {
525
sts.command_queue.push(&sts, &SharedThreadState::copy_move_test_copy,
526
SharedThreadState::CopyMoveTestType(43));
527
528
sts.message_count_to_read = -1;
529
sts.reader_threadwork.main_start_work();
530
sts.reader_threadwork.main_wait_for_done();
531
532
CHECK(SharedThreadState::CopyMoveTestType::copy_count == 0);
533
CHECK(SharedThreadState::CopyMoveTestType::move_count == 2);
534
}
535
536
SUBCASE("Pass by reference") {
537
sts.command_queue.push(&sts, &SharedThreadState::copy_move_test_ref,
538
SharedThreadState::CopyMoveTestType(43));
539
540
sts.message_count_to_read = -1;
541
sts.reader_threadwork.main_start_work();
542
sts.reader_threadwork.main_wait_for_done();
543
544
CHECK(SharedThreadState::CopyMoveTestType::copy_count == 0);
545
CHECK(SharedThreadState::CopyMoveTestType::move_count == 1);
546
}
547
548
SUBCASE("Pass by rvalue reference") {
549
sts.command_queue.push(&sts, &SharedThreadState::copy_move_test_move,
550
SharedThreadState::CopyMoveTestType(43));
551
552
sts.message_count_to_read = -1;
553
sts.reader_threadwork.main_start_work();
554
sts.reader_threadwork.main_wait_for_done();
555
556
CHECK(SharedThreadState::CopyMoveTestType::copy_count == 0);
557
CHECK(SharedThreadState::CopyMoveTestType::move_count == 1);
558
}
559
}
560
561
sts.destroy_threads();
562
}
563
} // namespace TestCommandQueue
564
565