mirror of https://github.com/qTox/qTox.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
135 lines
4.5 KiB
135 lines
4.5 KiB
#include "settings.h" |
|
#include <QFile> |
|
#include <QSettings> |
|
#include <QDir> |
|
#include <QDebug> |
|
#include <QStandardPaths> |
|
|
|
#ifdef Q_OS_WIN |
|
#ifdef _WIN32_WINNT |
|
#undef _WIN32_WINNT |
|
#endif |
|
#define _WIN32_WINNT 0x0600 // Vista for SHGetKnownFolderPath |
|
#include <windows.h> |
|
#include <shldisp.h> |
|
#include <shlobj.h> |
|
#include <exdisp.h> |
|
#endif |
|
|
|
Settings::Settings() |
|
{ |
|
portable = false; |
|
QFile portableSettings(SETTINGS_FILE); |
|
if (portableSettings.exists()) |
|
{ |
|
QSettings ps(SETTINGS_FILE, QSettings::IniFormat); |
|
ps.beginGroup("General"); |
|
portable = ps.value("makeToxPortable", false).toBool(); |
|
} |
|
qDebug() << "Portable: "<<portable; |
|
|
|
#ifdef Q_OS_WIN |
|
// Get a primary unelevated token of the actual user |
|
hPrimaryToken = nullptr; |
|
HANDLE hShellProcess = nullptr, hShellProcessToken = nullptr; |
|
const DWORD dwTokenRights = TOKEN_QUERY | TOKEN_IMPERSONATE | TOKEN_ASSIGN_PRIMARY |
|
| TOKEN_DUPLICATE | TOKEN_ADJUST_DEFAULT | TOKEN_ADJUST_SESSIONID; |
|
DWORD dwPID = 0; |
|
HWND hwnd = nullptr; |
|
DWORD dwLastErr = 0; |
|
|
|
// Enable SeIncreaseQuotaPrivilege |
|
HANDLE hProcessToken = NULL; |
|
if (!OpenProcessToken(GetCurrentProcess(), TOKEN_ADJUST_PRIVILEGES, &hProcessToken)) |
|
goto unelevateFail; |
|
TOKEN_PRIVILEGES tkp; |
|
tkp.PrivilegeCount = 1; |
|
LookupPrivilegeValueW(NULL, SE_INCREASE_QUOTA_NAME, &tkp.Privileges[0].Luid); |
|
tkp.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED; |
|
AdjustTokenPrivileges(hProcessToken, FALSE, &tkp, 0, NULL, NULL); |
|
dwLastErr = GetLastError(); |
|
CloseHandle(hProcessToken); |
|
if (ERROR_SUCCESS != dwLastErr) |
|
goto unelevateFail; |
|
|
|
// Get a primary copy of the desktop shell's token, |
|
// we're assuming the shell is running as the actual user |
|
hwnd = GetShellWindow(); |
|
if (!hwnd) |
|
goto unelevateFail; |
|
GetWindowThreadProcessId(hwnd, &dwPID); |
|
if (!dwPID) |
|
goto unelevateFail; |
|
hShellProcess = OpenProcess(PROCESS_QUERY_INFORMATION, FALSE, dwPID); |
|
if (!hShellProcess) |
|
goto unelevateFail; |
|
if (!OpenProcessToken(hShellProcess, TOKEN_DUPLICATE, &hShellProcessToken)) |
|
goto unelevateFail; |
|
|
|
// Duplicate the shell's process token to get a primary token. |
|
// Based on experimentation, this is the minimal set of rights required for CreateProcessWithTokenW (contrary to current documentation). |
|
if (!DuplicateTokenEx(hShellProcessToken, dwTokenRights, NULL, SecurityImpersonation, TokenPrimary, &hPrimaryToken)) |
|
goto unelevateFail; |
|
|
|
qDebug() << "Unelevated primary access token acquired"; |
|
goto unelevateCleanup; |
|
unelevateFail: |
|
qWarning() << "Unelevate failed, couldn't get access token"; |
|
unelevateCleanup: |
|
CloseHandle(hShellProcessToken); |
|
CloseHandle(hShellProcess); |
|
#endif |
|
} |
|
|
|
Settings::~Settings() |
|
{ |
|
#ifdef Q_OS_WIN |
|
CloseHandle(hPrimaryToken); |
|
#endif |
|
} |
|
|
|
QString Settings::getSettingsDirPath() const |
|
{ |
|
if (portable) |
|
return QString(".")+QDir::separator(); |
|
|
|
// workaround for https://bugreports.qt-project.org/browse/QTBUG-38845 |
|
#ifdef Q_OS_WIN |
|
wchar_t* path; |
|
bool isOld = false; // If true, we can't unelevate and just return the path for our current home |
|
|
|
auto shell32H = LoadLibrary(TEXT("shell32.dll")); |
|
if (!(isOld = (shell32H == nullptr))) |
|
{ |
|
auto SHGetKnownFolderPathH = (decltype(&SHGetKnownFolderPath)) |
|
GetProcAddress(shell32H, "SHGetKnownFolderPath"); |
|
if (!(isOld = (SHGetKnownFolderPathH == nullptr))) |
|
SHGetKnownFolderPathH(FOLDERID_RoamingAppData, 0, hPrimaryToken, &path); |
|
} |
|
|
|
if (isOld) |
|
{ |
|
return QDir::cleanPath(QStandardPaths::writableLocation(QStandardPaths::HomeLocation) + QDir::separator() |
|
+ "AppData" + QDir::separator() + "Roaming" + QDir::separator() + "tox" + QDir::separator()); |
|
} |
|
else |
|
{ |
|
QString pathStr = QString::fromStdWString(path); |
|
pathStr.replace("\\", "/"); |
|
return pathStr + "/tox"; |
|
} |
|
#elif defined(Q_OS_OSX) |
|
return QDir::cleanPath(QStandardPaths::writableLocation(QStandardPaths::HomeLocation) + QDir::separator() |
|
+ "Library" + QDir::separator() + "Application Support" + QDir::separator() + "Tox")+QDir::separator(); |
|
#else |
|
return QDir::cleanPath(QStandardPaths::writableLocation(QStandardPaths::ConfigLocation) |
|
+ QDir::separator() + "tox")+QDir::separator(); |
|
#endif |
|
} |
|
|
|
#ifdef Q_OS_WIN |
|
HANDLE Settings::getPrimaryToken() const |
|
{ |
|
return hPrimaryToken; |
|
} |
|
#endif
|
|
|