WireGuard rework for MacOS and Windows (#314)
WireGuard rework for MacOS and Windows
This commit is contained in:
parent
421a27ceae
commit
07c38e9b6c
60 changed files with 4779 additions and 434 deletions
547
client/platforms/windows/daemon/windowssplittunnel.cpp
Normal file
547
client/platforms/windows/daemon/windowssplittunnel.cpp
Normal file
|
|
@ -0,0 +1,547 @@
|
|||
/* This Source Code Form is subject to the terms of the Mozilla Public
|
||||
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
||||
|
||||
#include "windowssplittunnel.h"
|
||||
|
||||
#include "../windowscommons.h"
|
||||
#include "../windowsservicemanager.h"
|
||||
#include "logger.h"
|
||||
#include "platforms/windows/windowsutils.h"
|
||||
#include "windowsfirewall.h"
|
||||
|
||||
#define PSAPI_VERSION 2
|
||||
#include <Windows.h>
|
||||
#include <psapi.h>
|
||||
|
||||
#include <QCoreApplication>
|
||||
#include <QFileInfo>
|
||||
#include <QNetworkInterface>
|
||||
#include <QScopeGuard>
|
||||
|
||||
namespace {
|
||||
Logger logger("WindowsSplitTunnel");
|
||||
}
|
||||
|
||||
WindowsSplitTunnel::WindowsSplitTunnel(QObject* parent) : QObject(parent) {
|
||||
if (detectConflict()) {
|
||||
logger.error() << "Conflict detected, abort Split-Tunnel init.";
|
||||
uninstallDriver();
|
||||
return;
|
||||
}
|
||||
if (!isInstalled()) {
|
||||
logger.debug() << "Driver is not Installed, doing so";
|
||||
auto handle = installDriver();
|
||||
if (handle == INVALID_HANDLE_VALUE) {
|
||||
WindowsUtils::windowsLog("Failed to install Driver");
|
||||
return;
|
||||
}
|
||||
logger.debug() << "Driver installed";
|
||||
CloseServiceHandle(handle);
|
||||
} else {
|
||||
logger.debug() << "Driver is installed";
|
||||
}
|
||||
initDriver();
|
||||
}
|
||||
|
||||
WindowsSplitTunnel::~WindowsSplitTunnel() {
|
||||
CloseHandle(m_driver);
|
||||
uninstallDriver();
|
||||
}
|
||||
|
||||
void WindowsSplitTunnel::initDriver() {
|
||||
if (detectConflict()) {
|
||||
logger.error() << "Conflict detected, abort Split-Tunnel init.";
|
||||
return;
|
||||
}
|
||||
logger.debug() << "Try to open Split Tunnel Driver";
|
||||
// Open the Driver Symlink
|
||||
m_driver = CreateFileW(DRIVER_SYMLINK, GENERIC_READ | GENERIC_WRITE, 0,
|
||||
nullptr, OPEN_EXISTING, 0, nullptr);
|
||||
;
|
||||
|
||||
if (m_driver == INVALID_HANDLE_VALUE) {
|
||||
WindowsUtils::windowsLog("Failed to open Driver: ");
|
||||
|
||||
// If the handle is not present, try again after the serivce has started;
|
||||
auto driver_manager = WindowsServiceManager(DRIVER_SERVICE_NAME);
|
||||
QObject::connect(&driver_manager, &WindowsServiceManager::serviceStarted,
|
||||
this, &WindowsSplitTunnel::initDriver);
|
||||
driver_manager.startService();
|
||||
return;
|
||||
}
|
||||
|
||||
logger.debug() << "Connected to the Driver";
|
||||
// Reset Driver as it has wfp handles probably >:(
|
||||
|
||||
if (!WindowsFirewall::instance()->init()) {
|
||||
logger.error() << "Init WFP-Sublayer failed, driver won't be functional";
|
||||
return;
|
||||
}
|
||||
|
||||
// We need to now check the state and init it, if required
|
||||
|
||||
auto state = getState();
|
||||
if (state == STATE_UNKNOWN) {
|
||||
logger.debug() << "Cannot check if driver is initialized";
|
||||
}
|
||||
if (state >= STATE_INITIALIZED) {
|
||||
logger.debug() << "Driver already initialized: " << state;
|
||||
reset();
|
||||
|
||||
auto newState = getState();
|
||||
logger.debug() << "New state after reset:" << newState;
|
||||
if (newState >= STATE_INITIALIZED) {
|
||||
logger.debug() << "Reset unsuccesfull";
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
DWORD bytesReturned;
|
||||
auto ok = DeviceIoControl(m_driver, IOCTL_INITIALIZE, nullptr, 0, nullptr, 0,
|
||||
&bytesReturned, nullptr);
|
||||
if (!ok) {
|
||||
auto err = GetLastError();
|
||||
logger.error() << "Driver init failed err -" << err;
|
||||
logger.error() << "State:" << getState();
|
||||
|
||||
return;
|
||||
}
|
||||
logger.debug() << "Driver initialized" << getState();
|
||||
}
|
||||
|
||||
void WindowsSplitTunnel::setRules(const QStringList& appPaths) {
|
||||
auto state = getState();
|
||||
if (state != STATE_READY && state != STATE_RUNNING) {
|
||||
logger.warning() << "Driver is not in the right State to set Rules"
|
||||
<< state;
|
||||
return;
|
||||
}
|
||||
|
||||
logger.debug() << "Pushing new Ruleset for Split-Tunnel " << state;
|
||||
auto config = generateAppConfiguration(appPaths);
|
||||
|
||||
DWORD bytesReturned;
|
||||
auto ok = DeviceIoControl(m_driver, IOCTL_SET_CONFIGURATION, &config[0],
|
||||
(DWORD)config.size(), nullptr, 0, &bytesReturned,
|
||||
nullptr);
|
||||
if (!ok) {
|
||||
auto err = GetLastError();
|
||||
WindowsUtils::windowsLog("Set Config Failed:");
|
||||
logger.error() << "Failed to set Config err code " << err;
|
||||
return;
|
||||
}
|
||||
logger.debug() << "New Configuration applied: " << getState();
|
||||
}
|
||||
|
||||
void WindowsSplitTunnel::start(int inetAdapterIndex) {
|
||||
// To Start we need to send 2 things:
|
||||
// Network info (what is vpn what is network)
|
||||
logger.debug() << "Starting SplitTunnel";
|
||||
DWORD bytesReturned;
|
||||
|
||||
if (getState() == STATE_STARTED) {
|
||||
logger.debug() << "Driver needs Init Call";
|
||||
DWORD bytesReturned;
|
||||
auto ok = DeviceIoControl(m_driver, IOCTL_INITIALIZE, nullptr, 0, nullptr,
|
||||
0, &bytesReturned, nullptr);
|
||||
if (!ok) {
|
||||
logger.error() << "Driver init failed";
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
// Process Info (what is running already)
|
||||
if (getState() == STATE_INITIALIZED) {
|
||||
logger.debug() << "State is Init, requires process config";
|
||||
auto config = generateProcessBlob();
|
||||
auto ok = DeviceIoControl(m_driver, IOCTL_REGISTER_PROCESSES, &config[0],
|
||||
(DWORD)config.size(), nullptr, 0, &bytesReturned,
|
||||
nullptr);
|
||||
if (!ok) {
|
||||
logger.error() << "Failed to set Process Config";
|
||||
return;
|
||||
}
|
||||
logger.debug() << "Set Process Config ok || new State:" << getState();
|
||||
}
|
||||
|
||||
if (getState() == STATE_INITIALIZED) {
|
||||
logger.warning() << "Driver is still not ready after process list send";
|
||||
return;
|
||||
}
|
||||
logger.debug() << "Driver is ready || new State:" << getState();
|
||||
|
||||
auto config = generateIPConfiguration(inetAdapterIndex);
|
||||
auto ok = DeviceIoControl(m_driver, IOCTL_REGISTER_IP_ADDRESSES, &config[0],
|
||||
(DWORD)config.size(), nullptr, 0, &bytesReturned,
|
||||
nullptr);
|
||||
if (!ok) {
|
||||
logger.error() << "Failed to set Network Config";
|
||||
return;
|
||||
}
|
||||
logger.debug() << "New Network Config Applied || new State:" << getState();
|
||||
}
|
||||
|
||||
void WindowsSplitTunnel::stop() {
|
||||
DWORD bytesReturned;
|
||||
auto ok = DeviceIoControl(m_driver, IOCTL_CLEAR_CONFIGURATION, nullptr, 0,
|
||||
nullptr, 0, &bytesReturned, nullptr);
|
||||
if (!ok) {
|
||||
logger.error() << "Stopping Split tunnel not successfull";
|
||||
return;
|
||||
}
|
||||
logger.debug() << "Stopping Split tunnel successfull";
|
||||
}
|
||||
|
||||
void WindowsSplitTunnel::reset() {
|
||||
DWORD bytesReturned;
|
||||
auto ok = DeviceIoControl(m_driver, IOCTL_ST_RESET, nullptr, 0, nullptr, 0,
|
||||
&bytesReturned, nullptr);
|
||||
if (!ok) {
|
||||
logger.error() << "Reset Split tunnel not successfull";
|
||||
return;
|
||||
}
|
||||
logger.debug() << "Reset Split tunnel successfull";
|
||||
}
|
||||
|
||||
DRIVER_STATE WindowsSplitTunnel::getState() {
|
||||
if (m_driver == INVALID_HANDLE_VALUE) {
|
||||
logger.debug() << "Can't query State from non Opened Driver";
|
||||
return STATE_UNKNOWN;
|
||||
}
|
||||
DWORD bytesReturned;
|
||||
SIZE_T outBuffer;
|
||||
bool ok = DeviceIoControl(m_driver, IOCTL_GET_STATE, nullptr, 0, &outBuffer,
|
||||
sizeof(outBuffer), &bytesReturned, nullptr);
|
||||
if (!ok) {
|
||||
WindowsUtils::windowsLog("getState response failure");
|
||||
return STATE_UNKNOWN;
|
||||
}
|
||||
if (bytesReturned == 0) {
|
||||
WindowsUtils::windowsLog("getState response is empty");
|
||||
return STATE_UNKNOWN;
|
||||
}
|
||||
return static_cast<DRIVER_STATE>(outBuffer);
|
||||
}
|
||||
|
||||
std::vector<uint8_t> WindowsSplitTunnel::generateAppConfiguration(
|
||||
const QStringList& appPaths) {
|
||||
// Step 1: Calculate how much size the buffer will need
|
||||
size_t cummulated_string_size = 0;
|
||||
QStringList dosPaths;
|
||||
for (auto const& path : appPaths) {
|
||||
auto dosPath = convertPath(path);
|
||||
dosPaths.append(dosPath);
|
||||
cummulated_string_size += dosPath.toStdWString().size() * sizeof(wchar_t);
|
||||
logger.debug() << dosPath;
|
||||
}
|
||||
size_t bufferSize = sizeof(CONFIGURATION_HEADER) +
|
||||
(sizeof(CONFIGURATION_ENTRY) * appPaths.size()) +
|
||||
cummulated_string_size;
|
||||
std::vector<uint8_t> outBuffer(bufferSize);
|
||||
|
||||
auto header = (CONFIGURATION_HEADER*)&outBuffer[0];
|
||||
auto entry = (CONFIGURATION_ENTRY*)(header + 1);
|
||||
|
||||
auto stringDest = &outBuffer[0] + sizeof(CONFIGURATION_HEADER) +
|
||||
(sizeof(CONFIGURATION_ENTRY) * appPaths.size());
|
||||
|
||||
SIZE_T stringOffset = 0;
|
||||
|
||||
for (const QString& path : dosPaths) {
|
||||
auto wstr = path.toStdWString();
|
||||
auto cstr = wstr.c_str();
|
||||
auto stringLength = wstr.size() * sizeof(wchar_t);
|
||||
|
||||
entry->ImageNameLength = (USHORT)stringLength;
|
||||
entry->ImageNameOffset = stringOffset;
|
||||
|
||||
memcpy(stringDest, cstr, stringLength);
|
||||
|
||||
++entry;
|
||||
stringDest += stringLength;
|
||||
stringOffset += stringLength;
|
||||
}
|
||||
|
||||
header->NumEntries = appPaths.length();
|
||||
header->TotalLength = bufferSize;
|
||||
|
||||
return outBuffer;
|
||||
}
|
||||
|
||||
std::vector<uint8_t> WindowsSplitTunnel::generateIPConfiguration(
|
||||
int inetAdapterIndex) {
|
||||
std::vector<uint8_t> out(sizeof(IP_ADDRESSES_CONFIG));
|
||||
|
||||
auto config = reinterpret_cast<IP_ADDRESSES_CONFIG*>(&out[0]);
|
||||
|
||||
auto ifaces = QNetworkInterface::allInterfaces();
|
||||
// Always the VPN
|
||||
getAddress(WindowsCommons::VPNAdapterIndex(), &config->TunnelIpv4,
|
||||
&config->TunnelIpv6);
|
||||
// 2nd best route
|
||||
getAddress(inetAdapterIndex, &config->InternetIpv4, &config->InternetIpv6);
|
||||
return out;
|
||||
}
|
||||
void WindowsSplitTunnel::getAddress(int adapterIndex, IN_ADDR* out_ipv4,
|
||||
IN6_ADDR* out_ipv6) {
|
||||
QNetworkInterface target =
|
||||
QNetworkInterface::interfaceFromIndex(adapterIndex);
|
||||
logger.debug() << "Getting adapter info for:" << target.humanReadableName();
|
||||
|
||||
// take the first v4/v6 Adress and convert to in_addr
|
||||
for (auto address : target.addressEntries()) {
|
||||
if (address.ip().protocol() == QAbstractSocket::IPv4Protocol) {
|
||||
auto adrr = address.ip().toString();
|
||||
std::wstring wstr = adrr.toStdWString();
|
||||
logger.debug() << "IpV4" << logger.sensitive(adrr);
|
||||
PCWSTR w_str_ip = wstr.c_str();
|
||||
auto ok = InetPtonW(AF_INET, w_str_ip, out_ipv4);
|
||||
if (ok != 1) {
|
||||
logger.debug() << "Ipv4 Conversation error" << WSAGetLastError();
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
for (auto address : target.addressEntries()) {
|
||||
if (address.ip().protocol() == QAbstractSocket::IPv6Protocol) {
|
||||
auto adrr = address.ip().toString();
|
||||
std::wstring wstr = adrr.toStdWString();
|
||||
logger.debug() << "IpV6" << logger.sensitive(adrr);
|
||||
PCWSTR w_str_ip = wstr.c_str();
|
||||
auto ok = InetPtonW(AF_INET6, w_str_ip, out_ipv6);
|
||||
if (ok != 1) {
|
||||
logger.error() << "Ipv6 Conversation error" << WSAGetLastError();
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
std::vector<uint8_t> WindowsSplitTunnel::generateProcessBlob() {
|
||||
// Get a Snapshot of all processes that are running:
|
||||
HANDLE snapshot_handle = CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS, 0);
|
||||
if (snapshot_handle == INVALID_HANDLE_VALUE) {
|
||||
WindowsUtils::windowsLog("Creating Process snapshot failed");
|
||||
return std::vector<uint8_t>(0);
|
||||
}
|
||||
auto cleanup = qScopeGuard([&] { CloseHandle(snapshot_handle); });
|
||||
// Load the First Entry, later iterate over all
|
||||
PROCESSENTRY32W currentProcess;
|
||||
currentProcess.dwSize = sizeof(PROCESSENTRY32W);
|
||||
|
||||
if (FALSE == (Process32First(snapshot_handle, ¤tProcess))) {
|
||||
WindowsUtils::windowsLog("Cant read first entry");
|
||||
}
|
||||
|
||||
QMap<DWORD, ProcessInfo> processes;
|
||||
|
||||
do {
|
||||
auto process_handle = OpenProcess(PROCESS_QUERY_LIMITED_INFORMATION, FALSE,
|
||||
currentProcess.th32ProcessID);
|
||||
|
||||
if (process_handle == INVALID_HANDLE_VALUE) {
|
||||
continue;
|
||||
}
|
||||
ProcessInfo info = getProcessInfo(process_handle, currentProcess);
|
||||
processes.insert(info.ProcessId, info);
|
||||
CloseHandle(process_handle);
|
||||
|
||||
} while (FALSE != (Process32NextW(snapshot_handle, ¤tProcess)));
|
||||
|
||||
auto process_list = processes.values();
|
||||
if (process_list.isEmpty()) {
|
||||
logger.debug() << "Process Snapshot list was empty";
|
||||
return std::vector<uint8_t>(0);
|
||||
}
|
||||
|
||||
logger.debug() << "Reading Processes NUM: " << process_list.size();
|
||||
// Determine the Size of the outBuffer:
|
||||
size_t totalStringSize = 0;
|
||||
|
||||
for (const auto& process : process_list) {
|
||||
totalStringSize += (process.DevicePath.size() * sizeof(wchar_t));
|
||||
}
|
||||
auto bufferSize = sizeof(PROCESS_DISCOVERY_HEADER) +
|
||||
(sizeof(PROCESS_DISCOVERY_ENTRY) * processes.size()) +
|
||||
totalStringSize;
|
||||
|
||||
std::vector<uint8_t> out(bufferSize);
|
||||
|
||||
auto header = reinterpret_cast<PROCESS_DISCOVERY_HEADER*>(&out[0]);
|
||||
auto entry = reinterpret_cast<PROCESS_DISCOVERY_ENTRY*>(header + 1);
|
||||
auto stringBuffer = reinterpret_cast<uint8_t*>(entry + processes.size());
|
||||
|
||||
SIZE_T currentStringOffset = 0;
|
||||
|
||||
for (const auto& process : process_list) {
|
||||
// Wierd DWORD -> Handle Pointer magic.
|
||||
entry->ProcessId = (HANDLE)((size_t)process.ProcessId);
|
||||
entry->ParentProcessId = (HANDLE)((size_t)process.ParentProcessId);
|
||||
|
||||
if (process.DevicePath.empty()) {
|
||||
entry->ImageNameOffset = 0;
|
||||
entry->ImageNameLength = 0;
|
||||
} else {
|
||||
const auto imageNameLength = process.DevicePath.size() * sizeof(wchar_t);
|
||||
|
||||
entry->ImageNameOffset = currentStringOffset;
|
||||
entry->ImageNameLength = static_cast<USHORT>(imageNameLength);
|
||||
|
||||
RtlCopyMemory(stringBuffer + currentStringOffset, &process.DevicePath[0],
|
||||
imageNameLength);
|
||||
|
||||
currentStringOffset += imageNameLength;
|
||||
}
|
||||
++entry;
|
||||
}
|
||||
|
||||
header->NumEntries = processes.size();
|
||||
header->TotalLength = bufferSize;
|
||||
|
||||
return out;
|
||||
}
|
||||
|
||||
void WindowsSplitTunnel::close() {
|
||||
CloseHandle(m_driver);
|
||||
m_driver = INVALID_HANDLE_VALUE;
|
||||
}
|
||||
|
||||
ProcessInfo WindowsSplitTunnel::getProcessInfo(
|
||||
HANDLE process, const PROCESSENTRY32W& processMeta) {
|
||||
ProcessInfo pi;
|
||||
pi.ParentProcessId = processMeta.th32ParentProcessID;
|
||||
pi.ProcessId = processMeta.th32ProcessID;
|
||||
pi.CreationTime = {0, 0};
|
||||
pi.DevicePath = L"";
|
||||
|
||||
FILETIME creationTime, null_time;
|
||||
auto ok = GetProcessTimes(process, &creationTime, &null_time, &null_time,
|
||||
&null_time);
|
||||
if (ok) {
|
||||
pi.CreationTime = creationTime;
|
||||
}
|
||||
wchar_t imagepath[MAX_PATH + 1];
|
||||
if (K32GetProcessImageFileNameW(
|
||||
process, imagepath, sizeof(imagepath) / sizeof(*imagepath)) != 0) {
|
||||
pi.DevicePath = imagepath;
|
||||
}
|
||||
return pi;
|
||||
}
|
||||
|
||||
// static
|
||||
SC_HANDLE WindowsSplitTunnel::installDriver() {
|
||||
LPCWSTR displayName = L"Amnezia Split Tunnel Service";
|
||||
QFileInfo driver(qApp->applicationDirPath() + "/" + DRIVER_FILENAME);
|
||||
if (!driver.exists()) {
|
||||
logger.error() << "Split Tunnel Driver File not found "
|
||||
<< driver.absoluteFilePath();
|
||||
return (SC_HANDLE)INVALID_HANDLE_VALUE;
|
||||
}
|
||||
auto path = driver.absolutePath() + "/" + DRIVER_FILENAME;
|
||||
LPCWSTR binPath = (const wchar_t*)path.utf16();
|
||||
auto scm_rights = SC_MANAGER_ALL_ACCESS;
|
||||
auto serviceManager = OpenSCManager(NULL, // local computer
|
||||
NULL, // servicesActive database
|
||||
scm_rights);
|
||||
auto service = CreateService(serviceManager, DRIVER_SERVICE_NAME, displayName,
|
||||
SERVICE_ALL_ACCESS, SERVICE_KERNEL_DRIVER,
|
||||
SERVICE_DEMAND_START, SERVICE_ERROR_NORMAL,
|
||||
binPath, nullptr, 0, nullptr, nullptr, nullptr);
|
||||
CloseServiceHandle(serviceManager);
|
||||
return service;
|
||||
}
|
||||
// static
|
||||
bool WindowsSplitTunnel::uninstallDriver() {
|
||||
auto scm_rights = SC_MANAGER_ALL_ACCESS;
|
||||
auto serviceManager = OpenSCManager(NULL, // local computer
|
||||
NULL, // servicesActive database
|
||||
scm_rights);
|
||||
|
||||
auto servicehandle =
|
||||
OpenService(serviceManager, DRIVER_SERVICE_NAME, GENERIC_READ);
|
||||
auto result = DeleteService(servicehandle);
|
||||
if (result) {
|
||||
logger.debug() << "Split Tunnel Driver Removed";
|
||||
}
|
||||
return result;
|
||||
}
|
||||
// static
|
||||
bool WindowsSplitTunnel::isInstalled() {
|
||||
// Check if the Drivers I/O File is present
|
||||
auto symlink = QFileInfo(QString::fromWCharArray(DRIVER_SYMLINK));
|
||||
if (symlink.exists()) {
|
||||
return true;
|
||||
}
|
||||
// If not check with SCM, if the kernel service exists
|
||||
auto scm_rights = SC_MANAGER_ALL_ACCESS;
|
||||
auto serviceManager = OpenSCManager(NULL, // local computer
|
||||
NULL, // servicesActive database
|
||||
scm_rights);
|
||||
auto servicehandle =
|
||||
OpenService(serviceManager, DRIVER_SERVICE_NAME, GENERIC_READ);
|
||||
auto err = GetLastError();
|
||||
CloseServiceHandle(serviceManager);
|
||||
CloseServiceHandle(servicehandle);
|
||||
return err != ERROR_SERVICE_DOES_NOT_EXIST;
|
||||
}
|
||||
|
||||
QString WindowsSplitTunnel::convertPath(const QString& path) {
|
||||
auto parts = path.split("/");
|
||||
QString driveLetter = parts.takeFirst();
|
||||
if (!driveLetter.contains(":") || parts.size() == 0) {
|
||||
// device should contain : for e.g C:
|
||||
return "";
|
||||
}
|
||||
QByteArray buffer(2048, 0xFF);
|
||||
auto ok = QueryDosDeviceW(qUtf16Printable(driveLetter),
|
||||
(wchar_t*)buffer.data(), buffer.size() / 2);
|
||||
|
||||
if (ok == ERROR_INSUFFICIENT_BUFFER) {
|
||||
buffer.resize(buffer.size() * 2);
|
||||
ok = QueryDosDeviceW(qUtf16Printable(driveLetter), (wchar_t*)buffer.data(),
|
||||
buffer.size() / 2);
|
||||
}
|
||||
if (ok == 0) {
|
||||
WindowsUtils::windowsLog("Err fetching dos path");
|
||||
return "";
|
||||
}
|
||||
QString deviceName;
|
||||
deviceName = QString::fromWCharArray((wchar_t*)buffer.data());
|
||||
parts.prepend(deviceName);
|
||||
|
||||
return parts.join("\\");
|
||||
}
|
||||
|
||||
// static
|
||||
bool WindowsSplitTunnel::detectConflict() {
|
||||
auto scm_rights = SC_MANAGER_ENUMERATE_SERVICE;
|
||||
auto serviceManager = OpenSCManager(NULL, // local computer
|
||||
NULL, // servicesActive database
|
||||
scm_rights);
|
||||
auto cleanup = qScopeGuard([&] { CloseServiceHandle(serviceManager); });
|
||||
// Query for Mullvad Service.
|
||||
auto servicehandle =
|
||||
OpenService(serviceManager, MV_SERVICE_NAME, GENERIC_READ);
|
||||
auto err = GetLastError();
|
||||
CloseServiceHandle(servicehandle);
|
||||
if (err != ERROR_SERVICE_DOES_NOT_EXIST) {
|
||||
WindowsUtils::windowsLog("Mullvad Detected - Disabling SplitTunnel: ");
|
||||
// Mullvad is installed, so we would certainly break things.
|
||||
return true;
|
||||
}
|
||||
auto symlink = QFileInfo(QString::fromWCharArray(DRIVER_SYMLINK));
|
||||
if (!symlink.exists()) {
|
||||
// The driver is not loaded / installed.. MV is not installed, all good!
|
||||
logger.info() << "No Split-Tunnel Conflict detected, continue.";
|
||||
return false;
|
||||
}
|
||||
// The driver exists, so let's check if it has been created by us.
|
||||
// If our service is not present, it's has been created by
|
||||
// someone else so we should not use that :)
|
||||
servicehandle =
|
||||
OpenService(serviceManager, DRIVER_SERVICE_NAME, GENERIC_READ);
|
||||
err = GetLastError();
|
||||
CloseServiceHandle(servicehandle);
|
||||
return err == ERROR_SERVICE_DOES_NOT_EXIST;
|
||||
}
|
||||
Loading…
Add table
Add a link
Reference in a new issue