-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy patharray-sum.cpp
129 lines (99 loc) · 3.34 KB
/
array-sum.cpp
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
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
#include <pthread.h>
#include <cerrno>
#include <chrono>
#include <cstddef>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <iostream>
#include <span>
#include <vector>
#define ERROR(fmt, ...) \
do { \
fprintf(stderr, "%s:%d: " fmt "\n", __FILE__, __LINE__, \
##__VA_ARGS__); \
exit(1); \
} while (false)
unsigned long parse_number(char* str) {
char* end = nullptr;
unsigned long n = strtoul(str, &end, 10);
if (*end != '\0') {
ERROR("Invalid number");
}
return n;
}
struct ThreadArgs {
int* start;
int* end;
int result;
};
void* sum_thread(void* args_untyped) {
ThreadArgs* args = (ThreadArgs*)args_untyped;
int result = 0;
for (int* ptr = args->start; ptr != args->end; ++ptr) {
result += *ptr;
}
args->result = result;
return nullptr;
}
int sum_array_with_threads(std::span<int> array, size_t thread_count) {
std::vector<pthread_t> threads(thread_count);
std::vector<ThreadArgs> args(thread_count);
size_t butch_size = array.size() / thread_count;
for (size_t i = 0; i < thread_count; ++i) {
size_t end_idx =
(i < thread_count - 1) ? (i + 1) * butch_size : array.size();
args[i] = ThreadArgs{.start = array.data() + i * butch_size,
.end = array.data() + end_idx,
.result = 0};
pthread_create(&threads[i], nullptr, sum_thread, &args[i]);
}
int total = 0;
for (size_t i = 0; i < thread_count; ++i) {
pthread_join(threads[i], nullptr);
total += args[i].result;
}
return total;
}
int sum_array(std::span<int> array) {
int total = 0;
for (int num : array) {
total += num;
}
return total;
}
int main(int argc, char* argv[]) {
if (argc != 3) {
ERROR("Usage: %s ARRAY_SIZE THREAD_COUNT", argv[0]);
}
unsigned long array_size = parse_number(argv[1]);
unsigned long thread_count = parse_number(argv[2]);
std::vector<int> array(array_size);
{
auto t0 = std::chrono::high_resolution_clock::now();
for (int& num : array) {
// num = 7;
num = rand() % 16;
}
auto t1 = std::chrono::high_resolution_clock::now();
auto duration = std::chrono::duration<double>(t1 - t0);
std::cout << "Time spent generating numbers: " << duration << '\n';
}
{
auto t0 = std::chrono::high_resolution_clock::now();
int total = sum_array(array);
auto t1 = std::chrono::high_resolution_clock::now();
auto duration = std::chrono::duration<double>(t1 - t0);
std::cout << "Time spent without threads: " << duration << '\n';
std::cout << "Total: " << total << '\n';
}
{
auto t0 = std::chrono::high_resolution_clock::now();
int total = sum_array_with_threads(array, thread_count);
auto t1 = std::chrono::high_resolution_clock::now();
auto duration = std::chrono::duration<double>(t1 - t0);
std::cout << "Time spent with " << thread_count
<< " threads: " << duration << '\n';
std::cout << "Total: " << total << '\n';
}
}