File: test_utils.h

package info (click to toggle)
stdgpu 1.3.0%2Bgit20220507.32e0517-6
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid, trixie
  • size: 2,528 kB
  • sloc: cpp: 7,818; pascal: 1,893; xml: 214; sh: 181; makefile: 50
file content (102 lines) | stat: -rw-r--r-- 2,894 bytes parent folder | download | duplicates (2)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
/*
 *  Copyright 2019 Patrick Stotko
 *  Licensed under the Apache License, Version 2.0 (the "License");
 *  you may not use this file except in compliance with the License.
 *  You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 *  Unless required by applicable law or agreed to in writing, software
 *  distributed under the License is distributed on an "AS IS" BASIS,
 *  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 *  See the License for the specific language governing permissions and
 *  limitations under the License.
 */

#ifndef TEST_UTILS_H
#define TEST_UTILS_H

#include <chrono>
#include <cmath>
#include <cstddef>
#include <limits>
#include <random>
#include <stdexcept>
#include <string>
#include <thread>
#include <utility>
#include <vector>

#include <stdgpu/cstddef.h>

namespace test_utils
{
/**
 * \brief Returns a seed chosen "truly" at random
 * \return The randomly chosen seed
 */
inline std::size_t
random_seed()
{
    try
    {
        std::random_device rd("/dev/urandom");

        // rd.entropy() != 0.0
        if (std::abs(rd.entropy()) >= std::numeric_limits<double>::min())
        {
            return rd();
        }

        throw std::runtime_error("Entropy is 0.0");
    }
    // For some reason, the following code fails to compile with NVCC+MSVC using the CUDA backend:
    // [[maybe_unused]] const std::exception& e
    // Thus, use the version below to fix unused parameter warnings
    catch (const std::exception&)
    {
    }

    return static_cast<std::size_t>(std::chrono::system_clock::now().time_since_epoch().count());
}

/**
 * \brief Returns a seed chosen "truly" at random for the this local thread
 * \return The randomly chosen thread-local seed
 */
inline std::size_t
random_thread_seed()
{
    return random_seed() + std::hash<std::thread::id>()(std::this_thread::get_id());
}

/**
 * \brief Runs the given function in N threads where N is the number of available concurrent threads the CPU can handle
 * \param[in] f A function
 * \param[in] args The arguments passed to f
 */
template <typename F, typename... Args>
inline void
for_each_concurrent_thread(F&& f, Args&&... args)
{
    const stdgpu::index_t concurrent_threads =
            std::max<stdgpu::index_t>(1, static_cast<stdgpu::index_t>(std::thread::hardware_concurrency()));
    std::vector<std::thread> threads;
    threads.reserve(static_cast<std::size_t>(concurrent_threads));

    for (stdgpu::index_t i = 0; i < concurrent_threads; ++i)
    {
        threads.emplace_back(std::forward<F>(f), std::forward<Args>(args)...);
    }

    for (stdgpu::index_t i = 0; i < concurrent_threads; ++i)
    {
        if (threads[static_cast<std::size_t>(i)].joinable())
        {
            threads[static_cast<std::size_t>(i)].join();
        }
    }
}
} // namespace test_utils

#endif // TEST_UTILS_H