-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathstream.hpp
113 lines (94 loc) · 3.33 KB
/
stream.hpp
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
103
104
105
106
107
108
109
110
111
112
113
#pragma once
#ifndef __HPC_STREAM_HPP__
#define __HPC_STREAM_HPP__
#include <iostream>
#include <vector>
#include "context.hpp"
#include "affinity.hpp"
#include "cpu.hpp"
namespace hpc {
template <typename T>
class stream {
public:
stream(std::shared_ptr<thread_context<T>> context, int id);
void create_threads();
void clean_threads();
void reset_threads();
private:
static void worker(stream *, int);
std::vector<std::thread> work_threads;
std::shared_ptr<thread_context<T>> context;
bool terminate;
int thread_num;
int id;
};
template <typename T>
stream<T>::stream(std::shared_ptr<thread_context<T>> context, int id) {
this->context = context;
this->id = id;
this->create_threads();
};
template <typename T>
void stream<T>::create_threads() {
this->terminate = false;
this->thread_num = this->context->affinity_infos[this->id].size();
for (int i = 0; i < thread_num; i++) {
this->work_threads.emplace_back(this->worker, this, i);
if (this->context->verbose) {
std::cout << "stream:" << this->id << ", created thread:" << i << std::endl;
}
}
}
template <typename T>
void stream<T>::clean_threads() {
this->terminate = true;
this->context->condition.notify_all();
for (int i = 0; i < this->work_threads.size(); i++) {
this->work_threads[i].join();
if (this->context->verbose) {
std::cout << "stream:" << this->id << ", joined thread:" << i << std::endl;
}
}
std::vector<std::thread> ().swap(this->work_threads);
}
template <typename T>
void stream<T>::reset_threads() {
this->clean_threads();
this->create_threads();
}
template <typename T>
void stream<T>::worker(stream *ptr, int i) {
set_thread_affinity(ptr->context->affinity_infos[ptr->id][i]);
while (!ptr->terminate) {
std::unique_lock<std::mutex> lock(ptr->context->mt);
ptr->context->condition.wait(lock, [ptr] {
return (ptr->terminate || !ptr->context->queue.empty());
});
if (ptr->terminate) {
if (ptr->context->verbose) {
std::cout << "stream:" << ptr->id << ", terminated worker:" << i << std::endl;
}
break;
} else {
auto task = ptr->context->queue.front();
ptr->context->queue.pop();
lock.unlock();
task->run();
ptr->context->condition.notify_one();
}
}
}
template <typename T>
std::vector<std::shared_ptr<stream<T>>> create_streams(
std::shared_ptr<thread_context<T>> context,
int stream_num, int thread_num, bool affinity) {
context->affinity_infos = cal_streams_affinity(
stream_num, thread_num, affinity, context->verbose);
std::vector<std::shared_ptr<stream<T>>> streams;
for (auto i = 0; i < context->affinity_infos.size(); i++) {
streams.emplace_back(std::make_shared<stream<T>>(context, i));
}
return streams;
}
}
#endif // __HPC_STREAM_HPP__