File: thread_pool.hpp

package info (click to toggle)
vart 2.5-5
  • links: PTS, VCS
  • area: main
  • in suites: sid, trixie
  • size: 4,404 kB
  • sloc: cpp: 30,188; python: 7,493; sh: 969; makefile: 37; ansic: 36
file content (76 lines) | stat: -rw-r--r-- 2,327 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

/*
 * Copyright 2019 Xilinx Inc.
 *
 * 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.
 */

#include <functional>
#include <future>
#include <memory>
#include <type_traits>
#include <vector>

#include "./erl_msg_box.hpp"
namespace vitis {
namespace ai {
class ThreadPool {
 public:
  static std::unique_ptr<ThreadPool> create(size_t num_of_threads);
#if __cplusplus > 201700
  template <class Function, class... Args>
  using result_t =
      std::invoke_result_t<std::decay_t<Function>, std::decay_t<Args>...>;
#elif __cplusplus > 201400
  template <class Function, class... Args>
  using result_t =
      std::result_of_t<std::decay_t<Function>(std::decay_t<Args>...)>;
#else
#error "not supported, c++14 or c++17 is needed."
#endif

  template <class Function, class... Args>
  std::future<result_t<Function, Args...>> async(Function&& f, Args&&... args) {
    std::packaged_task<result_t<Function, Args...>()> task(
        std::bind(std::forward<Function>(f), std::forward<Args>(args)...));
    std::future<result_t<Function, Args...>> ret = task.get_future();
    queue_.emplace_send(std::move(task));
    return ret;
  }
  ~ThreadPool();

 private:
  explicit ThreadPool(size_t num_of_thread);

 private:
  static void thread_main(ThreadPool* self);

 private:
  std::vector<std::thread> pool_;
  vitis::ai::ErlMsgBox<std::packaged_task<void()>> queue_;
  int running_;

 private:
  /// thanks akk
  /// https://stackoverflow.com/questions/20843271/passing-a-non-copyable-closure-object-to-stdfunction-parameter
  template <class F>
  auto make_copyable_function(F&& f) {
    using dF = std::decay_t<F>;
    auto spf = std::make_shared<dF>(std::forward<F>(f));
    return [spf](auto&&... args) -> decltype(auto) {
      return (*spf)(decltype(args)(args)...);
    };
  }
};
}  // namespace ai
}  // namespace vitis