| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273 |
- // Copyright (C) 2009-2021, Panagiotis Christopoulos Charitos and contributors.
- // All rights reserved.
- // Code licensed under the BSD License.
- // http://www.anki3d.org/LICENSE
- #include <Tests/Framework/Framework.h>
- #include <anki/resource/AsyncLoader.h>
- #include <anki/util/HighRezTimer.h>
- #include <anki/util/Atomic.h>
- #include <anki/util/Functions.h>
- namespace anki
- {
- class Task : public AsyncLoaderTask
- {
- public:
- F32 m_sleepTime = 0.0;
- Barrier* m_barrier = nullptr;
- Atomic<U32>* m_count = nullptr;
- I32 m_id = -1;
- Bool m_pause;
- Bool m_resubmit;
- Task(F32 time, Barrier* barrier, Atomic<U32>* count, I32 id = -1, Bool pause = false, Bool resubmit = false)
- : m_sleepTime(time)
- , m_barrier(barrier)
- , m_count(count)
- , m_id(id)
- , m_pause(pause)
- , m_resubmit(resubmit)
- {
- }
- Error operator()(AsyncLoaderTaskContext& ctx)
- {
- if(m_count)
- {
- auto x = m_count->fetchAdd(1);
- if(m_id >= 0)
- {
- if(m_id != static_cast<I32>(x))
- {
- ANKI_LOGE("Wrong excecution order");
- return Error::FUNCTION_FAILED;
- }
- }
- }
- if(m_sleepTime != 0.0)
- {
- HighRezTimer::sleep(m_sleepTime);
- }
- if(m_barrier)
- {
- m_barrier->wait();
- }
- ctx.m_pause = m_pause;
- ctx.m_resubmitTask = m_resubmit;
- m_resubmit = false;
- return Error::NONE;
- }
- };
- class MemTask : public AsyncLoaderTask
- {
- public:
- HeapAllocator<U8> m_alloc;
- Barrier* m_barrier = nullptr;
- MemTask(HeapAllocator<U8> alloc, Barrier* barrier)
- : m_alloc(alloc)
- , m_barrier(barrier)
- {
- }
- Error operator()(AsyncLoaderTaskContext& ctx)
- {
- void* mem = m_alloc.allocate(10);
- if(!mem)
- return Error::FUNCTION_FAILED;
- HighRezTimer::sleep(0.1);
- m_alloc.deallocate(mem, 10);
- if(m_barrier)
- {
- m_barrier->wait();
- }
- return Error::NONE;
- }
- };
- ANKI_TEST(Resource, AsyncLoader)
- {
- HeapAllocator<U8> alloc(allocAligned, nullptr);
- // Simple create destroy
- {
- AsyncLoader a;
- a.init(alloc);
- }
- // Simple task that will finish
- {
- AsyncLoader a;
- a.init(alloc);
- Barrier barrier(2);
- a.submitNewTask<Task>(0.0f, &barrier, nullptr);
- barrier.wait();
- }
- // Many tasks that will finish
- {
- AsyncLoader a;
- a.init(alloc);
- Barrier barrier(2);
- Atomic<U32> counter = {0};
- const U COUNT = 100;
- for(U i = 0; i < COUNT; i++)
- {
- Barrier* pbarrier = nullptr;
- if(i == COUNT - 1)
- {
- pbarrier = &barrier;
- }
- a.submitNewTask<Task>(0.01f, pbarrier, &counter);
- }
- barrier.wait();
- ANKI_TEST_EXPECT_EQ(counter.load(), COUNT);
- }
- // Many tasks that will _not_ finish
- {
- AsyncLoader a;
- a.init(alloc);
- for(U i = 0; i < 100; i++)
- {
- a.submitNewTask<Task>(0.0f, nullptr, nullptr);
- }
- }
- // Tasks that allocate
- {
- AsyncLoader a;
- a.init(alloc);
- Barrier barrier(2);
- for(U i = 0; i < 10; i++)
- {
- Barrier* pbarrier = nullptr;
- if(i == 9)
- {
- pbarrier = &barrier;
- }
- a.submitNewTask<MemTask>(alloc, pbarrier);
- }
- barrier.wait();
- }
- // Tasks that allocate and never finished
- {
- AsyncLoader a;
- a.init(alloc);
- for(U i = 0; i < 10; i++)
- {
- a.submitNewTask<MemTask>(alloc, nullptr);
- }
- }
- // Pause/resume
- {
- AsyncLoader a;
- a.init(alloc);
- Atomic<U32> counter(0);
- Barrier barrier(2);
- // Check if the pause will sync
- a.submitNewTask<Task>(0.5f, nullptr, &counter, 0);
- HighRezTimer::sleep(0.25); // Wait for the thread to pick the task...
- a.pause(); /// ...and then sync
- ANKI_TEST_EXPECT_EQ(counter.load(), 1);
- // Test resume
- a.submitNewTask<Task>(0.1f, nullptr, &counter, 1);
- HighRezTimer::sleep(1.0);
- ANKI_TEST_EXPECT_EQ(counter.load(), 1);
- a.resume();
- // Sync
- a.submitNewTask<Task>(0.1f, &barrier, &counter, 2);
- barrier.wait();
- ANKI_TEST_EXPECT_EQ(counter.load(), 3);
- }
- // Pause/resume
- {
- AsyncLoader a;
- a.init(alloc);
- Atomic<U32> counter(0);
- Barrier barrier(2);
- // Check task resubmit
- a.submitNewTask<Task>(0.0f, &barrier, &counter, -1, false, true);
- barrier.wait();
- barrier.wait();
- ANKI_TEST_EXPECT_EQ(counter.load(), 2);
- // Check task pause
- a.submitNewTask<Task>(0.0f, nullptr, &counter, -1, true, false);
- a.submitNewTask<Task>(0.0f, nullptr, &counter, -1, false, false);
- HighRezTimer::sleep(1.0);
- ANKI_TEST_EXPECT_EQ(counter.load(), 3);
- a.resume();
- HighRezTimer::sleep(1.0);
- ANKI_TEST_EXPECT_EQ(counter.load(), 4);
- // Check both
- counter.setNonAtomically(0);
- a.submitNewTask<Task>(0.0f, nullptr, &counter, 0, false, false);
- a.submitNewTask<Task>(0.0f, nullptr, &counter, -1, true, true);
- a.submitNewTask<Task>(0.0f, nullptr, &counter, 2, false, false);
- HighRezTimer::sleep(1.0);
- ANKI_TEST_EXPECT_EQ(counter.load(), 2);
- a.resume();
- HighRezTimer::sleep(1.0);
- ANKI_TEST_EXPECT_EQ(counter.load(), 4);
- }
- // Fuzzy test
- {
- AsyncLoader a;
- a.init(alloc);
- Barrier barrier(2);
- Atomic<U32> counter = {0};
- for(U32 i = 0; i < 10; i++)
- {
- Barrier* pbarrier = nullptr;
- if(i == 9)
- {
- pbarrier = &barrier;
- }
- a.submitNewTask<Task>(getRandomRange(0.0f, 0.5f), pbarrier, &counter, i);
- }
- barrier.wait();
- ANKI_TEST_EXPECT_EQ(counter.load(), 10);
- }
- }
- } // end namespace anki
|