mirror of
https://sourceware.org/git/binutils-gdb.git
synced 2025-02-11 13:02:10 +08:00
0b72cde372
While GDB is still C++11, lets add a gdb::make_unique template function that can be used to create std::unique_ptr objects, just like the C++14 std::make_unique. If GDB is being compiled with a C++14 compiler then the new gdb::make_unique function will delegate to the std::make_unique. I checked with gcc, and at -O1 and above gdb::make_unique will be optimised away completely in this case. If C++14 (or later) becomes our minimum, then it will be easy enough to go through the code and replace gdb::make_unique with std::make_unique later on. I've make use of this function in all the places I think this can easily be used, though I'm sure I've probably missed some. Should be no user visible changes after this commit. Approved-By: Tom Tromey <tom@tromey.com>
194 lines
4.4 KiB
C
194 lines
4.4 KiB
C
/* Self tests for parallel_for_each
|
|
|
|
Copyright (C) 2021-2023 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/>. */
|
|
|
|
/* This file is divided in two parts:
|
|
- FOR_EACH-undefined, and
|
|
- FOR_EACH-defined.
|
|
The former includes the latter, more than once, with different values for
|
|
FOR_EACH. The FOR_EACH-defined part reads like a regular function. */
|
|
#ifndef FOR_EACH
|
|
|
|
#include "defs.h"
|
|
#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;
|
|
};
|
|
|
|
/* Define test_par using TEST in the FOR_EACH-defined part. */
|
|
#define TEST test_par
|
|
#define FOR_EACH gdb::parallel_for_each
|
|
#include "parallel-for-selftests.c"
|
|
#undef FOR_EACH
|
|
#undef TEST
|
|
|
|
/* Define test_seq using TEST in the FOR_EACH-defined part. */
|
|
#define TEST test_seq
|
|
#define FOR_EACH gdb::sequential_for_each
|
|
#include "parallel-for-selftests.c"
|
|
#undef FOR_EACH
|
|
#undef TEST
|
|
|
|
static void
|
|
test (int n_threads)
|
|
{
|
|
test_par (n_threads);
|
|
test_seq (n_threads);
|
|
}
|
|
|
|
static void
|
|
test_n_threads ()
|
|
{
|
|
test (0);
|
|
test (1);
|
|
test (3);
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
#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_n_threads);
|
|
#endif /* CXX_STD_THREAD */
|
|
}
|
|
|
|
#else /* FOR_EACH */
|
|
|
|
static void
|
|
TEST (int n_threads)
|
|
{
|
|
save_restore_n_threads saver;
|
|
gdb::thread_pool::g_thread_pool->set_thread_count (n_threads);
|
|
|
|
#define NUMBER 10000
|
|
|
|
std::atomic<int> counter (0);
|
|
FOR_EACH (1, 0, NUMBER,
|
|
[&] (int start, int end)
|
|
{
|
|
counter += end - start;
|
|
});
|
|
SELF_CHECK (counter == NUMBER);
|
|
|
|
counter = 0;
|
|
FOR_EACH (1, 0, 0,
|
|
[&] (int start, int end)
|
|
{
|
|
counter += end - start;
|
|
});
|
|
SELF_CHECK (counter == 0);
|
|
|
|
auto task_size_max_ = [] (int iter)
|
|
{
|
|
return (size_t)SIZE_MAX;
|
|
};
|
|
auto task_size_max = gdb::make_function_view (task_size_max_);
|
|
|
|
counter = 0;
|
|
FOR_EACH (1, 0, NUMBER,
|
|
[&] (int start, int end)
|
|
{
|
|
counter += end - start;
|
|
}, task_size_max);
|
|
SELF_CHECK (counter == NUMBER);
|
|
|
|
auto task_size_one_ = [] (int iter)
|
|
{
|
|
return (size_t)1;
|
|
};
|
|
auto task_size_one = gdb::make_function_view (task_size_one_);
|
|
|
|
counter = 0;
|
|
FOR_EACH (1, 0, NUMBER,
|
|
[&] (int start, int end)
|
|
{
|
|
counter += end - start;
|
|
}, task_size_one);
|
|
SELF_CHECK (counter == NUMBER);
|
|
|
|
#undef NUMBER
|
|
|
|
/* 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);
|
|
|
|
FOR_EACH (1, 0, 1,
|
|
[&] (int start, int end)
|
|
{
|
|
if (start == end)
|
|
any_empty_tasks = true;
|
|
return gdb::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;
|
|
}));
|
|
|
|
/* The same but using the task size parameter. */
|
|
intresults.clear ();
|
|
any_empty_tasks = false;
|
|
FOR_EACH (1, 0, 1,
|
|
[&] (int start, int end)
|
|
{
|
|
if (start == end)
|
|
any_empty_tasks = true;
|
|
return gdb::make_unique<int> (end - start);
|
|
},
|
|
task_size_one);
|
|
SELF_CHECK (!any_empty_tasks);
|
|
SELF_CHECK (std::all_of (intresults.begin (),
|
|
intresults.end (),
|
|
[] (const std::unique_ptr<int> &entry)
|
|
{
|
|
return entry != nullptr;
|
|
}));
|
|
}
|
|
|
|
#endif /* FOR_EACH */
|