forked from Imagelibrary/binutils-gdb
This value will likely never change at runtime, so we might as well make it a template parameter. This has the "advantage" of being able to remove the unnecessary param from gdb::sequential_for_each. Change-Id: Ia172ab8e08964e30d4e3378a95ccfa782abce674 Approved-By: Tom Tromey <tom@tromey.com>
121 lines
3.2 KiB
C
121 lines
3.2 KiB
C
/* Self tests for parallel_for_each
|
|
|
|
Copyright (C) 2021-2025 Free Software Foundation, Inc.
|
|
|
|
This file is part of GDB.
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>. */
|
|
|
|
#include "gdbsupport/selftest.h"
|
|
#include "gdbsupport/parallel-for.h"
|
|
|
|
#if CXX_STD_THREAD
|
|
|
|
#include "gdbsupport/thread-pool.h"
|
|
|
|
namespace selftests {
|
|
namespace parallel_for {
|
|
|
|
struct save_restore_n_threads
|
|
{
|
|
save_restore_n_threads ()
|
|
: n_threads (gdb::thread_pool::g_thread_pool->thread_count ())
|
|
{
|
|
}
|
|
|
|
~save_restore_n_threads ()
|
|
{
|
|
gdb::thread_pool::g_thread_pool->set_thread_count (n_threads);
|
|
}
|
|
|
|
int n_threads;
|
|
};
|
|
|
|
using foreach_callback_t = gdb::function_view<void (int first, int last)>;
|
|
using do_foreach_t = gdb::function_view<void (int first, int last,
|
|
foreach_callback_t)>;
|
|
|
|
static void
|
|
test_one (int n_threads, do_foreach_t do_foreach)
|
|
{
|
|
save_restore_n_threads saver;
|
|
gdb::thread_pool::g_thread_pool->set_thread_count (n_threads);
|
|
|
|
{
|
|
constexpr int upper_bound = 1000;
|
|
std::atomic<int> counter (0);
|
|
do_foreach (0, upper_bound,
|
|
[&] (int start, int end) { counter += end - start; });
|
|
SELF_CHECK (counter == upper_bound);
|
|
}
|
|
|
|
{
|
|
std::atomic<int> counter (0);
|
|
do_foreach (0, 0, [&] (int start, int end) { counter += end - start; });
|
|
SELF_CHECK (counter == 0);
|
|
}
|
|
|
|
{
|
|
/* Check that if there are fewer tasks than threads, then we won't
|
|
end up with a null result. */
|
|
std::vector<std::unique_ptr<int>> intresults;
|
|
std::atomic<bool> any_empty_tasks (false);
|
|
|
|
do_foreach (0, 1,
|
|
[&] (int start, int end)
|
|
{
|
|
if (start == end)
|
|
any_empty_tasks = true;
|
|
|
|
return std::make_unique<int> (end - start);
|
|
});
|
|
|
|
SELF_CHECK (!any_empty_tasks);
|
|
SELF_CHECK (std::all_of (intresults.begin (), intresults.end (),
|
|
[] (const std::unique_ptr<int> &entry)
|
|
{ return entry != nullptr; }));
|
|
}
|
|
}
|
|
|
|
static void
|
|
test_parallel_for_each ()
|
|
{
|
|
const std::vector<do_foreach_t> for_each_functions
|
|
{
|
|
[] (int start, int end, foreach_callback_t callback)
|
|
{ gdb::parallel_for_each<1> (start, end, callback); },
|
|
[] (int start, int end, foreach_callback_t callback)
|
|
{ gdb::sequential_for_each (start, end, callback);}
|
|
};
|
|
|
|
for (int n_threads : { 0, 1, 3 })
|
|
for (const auto &for_each_function : for_each_functions)
|
|
test_one (n_threads, for_each_function);
|
|
}
|
|
|
|
} /* namespace parallel_for */
|
|
} /* namespace selftests */
|
|
|
|
#endif /* CXX_STD_THREAD */
|
|
|
|
void _initialize_parallel_for_selftests ();
|
|
void
|
|
_initialize_parallel_for_selftests ()
|
|
{
|
|
#ifdef CXX_STD_THREAD
|
|
selftests::register_test ("parallel_for",
|
|
selftests::parallel_for::test_parallel_for_each);
|
|
#endif /* CXX_STD_THREAD */
|
|
}
|