-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathstart_affinity.cpp
148 lines (112 loc) · 3.96 KB
/
start_affinity.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
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
/*
* StartAffinity 1.4.0 (c) 2020 Rudy Alex Kohn
*
* StartAffinity 1.3.1 (c) 2006 Adsci Engineering, LLC
*
* 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 2 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, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
#include <cstdio>
#include <string>
#include <string_view>
#include <numeric>
#include <windows.h>
#include <processthreadsapi.h>
#include <fmt/format.h>
#include <spdlog/spdlog.h>
#include <spdlog/sinks/rotating_file_sink.h>
#include "start_affinity.hpp"
#include "misc.hpp"
namespace
{
constexpr std::string_view processor_id_missing_msg = "Invalid processor id : ";
constexpr auto max_log_file_size = 1048576 * 5;
constexpr auto max_log_files = 3;
std::shared_ptr<spdlog::logger> logger =
spdlog::rotating_logger_mt("logger", "start_affinity_log.txt", max_log_file_size, max_log_files);
int tell_user_error()
{
const auto error = GetLastError();
if (!error)
return 0;
LPVOID msg_buf{};
FormatMessage(
FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS,
nullptr,
error,
MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
reinterpret_cast<LPTSTR>(&msg_buf),
0,
nullptr);
auto msg = std::string(static_cast<LPCTSTR>(msg_buf));
fmt::print(stderr, msg);
msg.erase(std::remove(msg.begin(), msg.end(), '\n'), msg.end());
logger->error(msg);
LocalFree(msg_buf);
return error;
}
} // namespace
int start_affinity::run(const ParserSettings *settings)
{
if (settings->verify_os && !misc::is_valid_os())
{
logger->error("WinNT/2000/XP or newer is required!");
return -1;
}
DWORD_PTR proc_mask{};
DWORD_PTR system_mask{};
GetProcessAffinityMask(GetCurrentProcess(), &proc_mask, &system_mask);
DWORD_PTR procs{};
for (const auto f : settings->processor_flags)
{
const auto b = 1 << f;
if (misc::is_set(system_mask, b))
procs = misc::set_bit(procs, b);
else
{
fmt::print("{} {}\n", processor_id_missing_msg, b);
logger->info("{} {}", processor_id_missing_msg, b);
}
}
// handle the security attributes
// lpSecurityDescriptor is set to null to force CreateProcess to use default descriptors
SECURITY_ATTRIBUTES sa_process{
.nLength = sizeof(SECURITY_ATTRIBUTES), .lpSecurityDescriptor = nullptr, .bInheritHandle = 0};
SECURITY_ATTRIBUTES sa_thread{
.nLength = sizeof(SECURITY_ATTRIBUTES), .lpSecurityDescriptor = nullptr, .bInheritHandle = 0};
STARTUPINFO startup_info{};
startup_info.cb = sizeof(STARTUPINFO);
GetStartupInfo(&startup_info);
PROCESS_INFORMATION process_info{};
auto *file_name = const_cast<char *>(settings->file_name.c_str());
if (!CreateProcess(nullptr, file_name, &sa_process, &sa_thread, 1, 0, nullptr, nullptr, &startup_info, &process_info))
return tell_user_error();
// set affinity mask
SetProcessAffinityMask(process_info.hProcess, procs);
// close handles
CloseHandle(process_info.hProcess);
CloseHandle(process_info.hThread);
const auto accumulator = [](const std::string &a, const int b) {
return a + ',' + std::to_string(b);
};
logger->info(
"Started {} with affinity : {}",
settings->file_name,
std::accumulate(
std::next(settings->processor_flags.begin(), 1), settings->processor_flags.end(),
std::to_string(settings->processor_flags.front()), accumulator));
return 0;
}