12f083884Ss.makeev_local // The MIT License (MIT)
22f083884Ss.makeev_local //
32f083884Ss.makeev_local // Copyright (c) 2015 Sergey Makeev, Vadim Slyusarev
42f083884Ss.makeev_local //
52f083884Ss.makeev_local // Permission is hereby granted, free of charge, to any person obtaining a copy
62f083884Ss.makeev_local // of this software and associated documentation files (the "Software"), to deal
72f083884Ss.makeev_local // in the Software without restriction, including without limitation the rights
82f083884Ss.makeev_local // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
92f083884Ss.makeev_local // copies of the Software, and to permit persons to whom the Software is
102f083884Ss.makeev_local // furnished to do so, subject to the following conditions:
112f083884Ss.makeev_local //
122f083884Ss.makeev_local // The above copyright notice and this permission notice shall be included in
132f083884Ss.makeev_local // all copies or substantial portions of the Software.
142f083884Ss.makeev_local //
152f083884Ss.makeev_local // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
162f083884Ss.makeev_local // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
172f083884Ss.makeev_local // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
182f083884Ss.makeev_local // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
192f083884Ss.makeev_local // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
202f083884Ss.makeev_local // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
212f083884Ss.makeev_local // THE SOFTWARE.
222f083884Ss.makeev_local
232f083884Ss.makeev_local #include "Tests.h"
242f083884Ss.makeev_local #include <UnitTest++.h>
252f083884Ss.makeev_local #include <MTScheduler.h>
262f083884Ss.makeev_local
SUITE(WaitTests)272f083884Ss.makeev_local SUITE(WaitTests)
282f083884Ss.makeev_local {
292f083884Ss.makeev_local ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
302f083884Ss.makeev_local namespace SimpleWaitFromSubtask
312f083884Ss.makeev_local {
322f083884Ss.makeev_local MT::Atomic32<int32> subTaskCount;
332f083884Ss.makeev_local MT::Atomic32<int32> taskCount;
342f083884Ss.makeev_local
352f083884Ss.makeev_local MT::TaskGroup testGroup;
362f083884Ss.makeev_local
372f083884Ss.makeev_local struct Subtask
382f083884Ss.makeev_local {
39b23bdf5aSs.makeev_local MT_DECLARE_TASK(Subtask, MT::StackRequirements::STANDARD, MT::TaskPriority::NORMAL, MT::Color::Blue);
402f083884Ss.makeev_local
412f083884Ss.makeev_local void Do(MT::FiberContext&)
422f083884Ss.makeev_local {
43*3cb1fd8eSs.makeev_local MT::SpinSleepMilliSeconds(2);
442f083884Ss.makeev_local subTaskCount.IncFetch();
452f083884Ss.makeev_local }
462f083884Ss.makeev_local };
472f083884Ss.makeev_local
482f083884Ss.makeev_local
492f083884Ss.makeev_local struct Task
502f083884Ss.makeev_local {
51b23bdf5aSs.makeev_local MT_DECLARE_TASK(Task, MT::StackRequirements::STANDARD, MT::TaskPriority::NORMAL, MT::Color::Blue);
522f083884Ss.makeev_local
532f083884Ss.makeev_local void Do(MT::FiberContext& ctx)
542f083884Ss.makeev_local {
552f083884Ss.makeev_local Subtask tasks[2];
569d94a278Ss.makeev_local ctx.RunSubtasksAndYield(testGroup, &tasks[0], MT_ARRAY_SIZE(tasks));
572f083884Ss.makeev_local
582f083884Ss.makeev_local taskCount.IncFetch();
592f083884Ss.makeev_local
602f083884Ss.makeev_local }
612f083884Ss.makeev_local };
622f083884Ss.makeev_local
632f083884Ss.makeev_local
642f083884Ss.makeev_local // Checks one simple task
652f083884Ss.makeev_local TEST(RunOneSimpleWaitTask)
662f083884Ss.makeev_local {
672f083884Ss.makeev_local taskCount.Store(0);
682f083884Ss.makeev_local subTaskCount.Store(0);
692f083884Ss.makeev_local
702f083884Ss.makeev_local MT::TaskScheduler scheduler;
712f083884Ss.makeev_local
722f083884Ss.makeev_local testGroup = scheduler.CreateGroup();
732f083884Ss.makeev_local
742f083884Ss.makeev_local Task tasks[16];
752f083884Ss.makeev_local scheduler.RunAsync(MT::TaskGroup::Default(), &tasks[0], MT_ARRAY_SIZE(tasks));
762f083884Ss.makeev_local
772f083884Ss.makeev_local CHECK(scheduler.WaitAll(2000));
782f083884Ss.makeev_local
792f083884Ss.makeev_local int subTaskCountFinisehd = subTaskCount.Load();
80f7a9bfc3Ss.makeev_local CHECK_EQUAL(MT_ARRAY_SIZE(tasks) * 2, (size_t)subTaskCountFinisehd);
812f083884Ss.makeev_local
822f083884Ss.makeev_local int taskCountFinished = taskCount.Load();
83f7a9bfc3Ss.makeev_local CHECK_EQUAL(MT_ARRAY_SIZE(tasks), (size_t)taskCountFinished);
842f083884Ss.makeev_local }
853d930776Ss.makeev_local
863d930776Ss.makeev_local ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
873d930776Ss.makeev_local
883d930776Ss.makeev_local struct LongTask
893d930776Ss.makeev_local {
903d930776Ss.makeev_local MT_DECLARE_TASK(LongTask, MT::StackRequirements::STANDARD, MT::TaskPriority::NORMAL, MT::Color::Blue);
913d930776Ss.makeev_local
923d930776Ss.makeev_local void Do(MT::FiberContext&)
933d930776Ss.makeev_local {
94e0717aaeSs.makeev_local MT::Thread::Sleep(1);
953d930776Ss.makeev_local }
963d930776Ss.makeev_local };
973d930776Ss.makeev_local
983d930776Ss.makeev_local TEST(TimeoutWaitAllTest)
993d930776Ss.makeev_local {
1003d930776Ss.makeev_local MT::TaskScheduler scheduler;
1013d930776Ss.makeev_local
1024dcd71e3Ss.makeev_local LongTask tasks[4000];
1033d930776Ss.makeev_local scheduler.RunAsync(MT::TaskGroup::Default(), &tasks[0], MT_ARRAY_SIZE(tasks));
1043d930776Ss.makeev_local
1053d930776Ss.makeev_local int64 startTime = MT::GetTimeMicroSeconds();
1063d930776Ss.makeev_local
107f7a9bfc3Ss.makeev_local CHECK_EQUAL(false, scheduler.WaitAll(33));
1083d930776Ss.makeev_local
1093d930776Ss.makeev_local int64 endTime = MT::GetTimeMicroSeconds();
1103d930776Ss.makeev_local int32 waitTime = (int32)(endTime - startTime);
1113d930776Ss.makeev_local printf("WaitAll(33) = %3.2f ms\n", waitTime / 1000.0f);
1123d930776Ss.makeev_local }
1133d930776Ss.makeev_local
1143d930776Ss.makeev_local TEST(TimeoutWaitGroupTest)
1153d930776Ss.makeev_local {
1163d930776Ss.makeev_local MT::TaskScheduler scheduler;
1173d930776Ss.makeev_local
1183d930776Ss.makeev_local MT::TaskGroup myGroup = scheduler.CreateGroup();
1193d930776Ss.makeev_local
1204dcd71e3Ss.makeev_local LongTask tasks[4000];
1213d930776Ss.makeev_local scheduler.RunAsync(myGroup, &tasks[0], MT_ARRAY_SIZE(tasks));
1223d930776Ss.makeev_local
1233d930776Ss.makeev_local int64 startTime = MT::GetTimeMicroSeconds();
1243d930776Ss.makeev_local
125f7a9bfc3Ss.makeev_local CHECK_EQUAL(false, scheduler.WaitGroup(myGroup, 33));
1263d930776Ss.makeev_local
1273d930776Ss.makeev_local int64 endTime = MT::GetTimeMicroSeconds();
1283d930776Ss.makeev_local int32 waitTime = (int32)(endTime - startTime);
1293d930776Ss.makeev_local printf("WaitGroup(33) = %3.2f ms\n", waitTime / 1000.0f);
1303d930776Ss.makeev_local }
1313d930776Ss.makeev_local
1323d930776Ss.makeev_local
1333d930776Ss.makeev_local ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
1343d930776Ss.makeev_local
1353d930776Ss.makeev_local
1363d930776Ss.makeev_local MT::Atomic32<uint32> finishedTaskCount;
1373d930776Ss.makeev_local
1383d930776Ss.makeev_local struct SecondaryTask
1393d930776Ss.makeev_local {
1403d930776Ss.makeev_local MT_DECLARE_TASK(SecondaryTask, MT::StackRequirements::STANDARD, MT::TaskPriority::NORMAL, MT::Color::Blue);
1413d930776Ss.makeev_local
1423d930776Ss.makeev_local void Do(MT::FiberContext&)
1433d930776Ss.makeev_local {
144*3cb1fd8eSs.makeev_local MT::SpinSleepMicroSeconds(20);
1453d930776Ss.makeev_local finishedTaskCount.IncFetch();
1463d930776Ss.makeev_local }
1473d930776Ss.makeev_local };
1483d930776Ss.makeev_local
1493d930776Ss.makeev_local
1503d930776Ss.makeev_local struct PrimaryTask
1513d930776Ss.makeev_local {
1523d930776Ss.makeev_local MT::TaskGroup secondaryGroup;
1533d930776Ss.makeev_local
1543d930776Ss.makeev_local MT_DECLARE_TASK(PrimaryTask, MT::StackRequirements::STANDARD, MT::TaskPriority::NORMAL, MT::Color::Blue);
1553d930776Ss.makeev_local
1563d930776Ss.makeev_local PrimaryTask(MT::TaskGroup _secondaryGroup)
1573d930776Ss.makeev_local : secondaryGroup(_secondaryGroup)
1583d930776Ss.makeev_local {
1593d930776Ss.makeev_local }
1603d930776Ss.makeev_local
1613d930776Ss.makeev_local void Do(MT::FiberContext& ctx)
1623d930776Ss.makeev_local {
1633d930776Ss.makeev_local SecondaryTask tasks[64];
1643d930776Ss.makeev_local ctx.RunAsync(secondaryGroup, &tasks[0], MT_ARRAY_SIZE(tasks));
1653d930776Ss.makeev_local
1663d930776Ss.makeev_local //CHECK(ctx.WaitGroup(secondaryGroup, 10000));
1673d930776Ss.makeev_local
1683d930776Ss.makeev_local finishedTaskCount.IncFetch();
1693d930776Ss.makeev_local }
1703d930776Ss.makeev_local };
1713d930776Ss.makeev_local
1723d930776Ss.makeev_local
1733d930776Ss.makeev_local TEST(RunOneSimpleWaitTaskFromTask)
1743d930776Ss.makeev_local {
1753d930776Ss.makeev_local finishedTaskCount.Store(0);
1763d930776Ss.makeev_local
1773d930776Ss.makeev_local MT::TaskScheduler scheduler;
1783d930776Ss.makeev_local
1793d930776Ss.makeev_local MT::TaskGroup mainGroup = scheduler.CreateGroup();
1803d930776Ss.makeev_local MT::TaskGroup secondaryGroup = scheduler.CreateGroup();
1813d930776Ss.makeev_local
1823d930776Ss.makeev_local PrimaryTask task(secondaryGroup);
1833d930776Ss.makeev_local scheduler.RunAsync(mainGroup, &task, 1);
1843d930776Ss.makeev_local
1853d930776Ss.makeev_local CHECK(scheduler.WaitGroup(mainGroup, 10000));
1863d930776Ss.makeev_local }
1873d930776Ss.makeev_local
1883d930776Ss.makeev_local
1892f083884Ss.makeev_local }
1902f083884Ss.makeev_local ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
1912f083884Ss.makeev_local }
1922f083884Ss.makeev_local
193