| /**************************************************************************** |
| ** |
| ** Copyright (C) 2016 The Qt Company Ltd. |
| ** Contact: https://www.qt.io/licensing/ |
| ** |
| ** This file is part of the QtCore module of the Qt Toolkit. |
| ** |
| ** $QT_BEGIN_LICENSE:LGPL$ |
| ** Commercial License Usage |
| ** Licensees holding valid commercial Qt licenses may use this file in |
| ** accordance with the commercial license agreement provided with the |
| ** Software or, alternatively, in accordance with the terms contained in |
| ** a written agreement between you and The Qt Company. For licensing terms |
| ** and conditions see https://www.qt.io/terms-conditions. For further |
| ** information use the contact form at https://www.qt.io/contact-us. |
| ** |
| ** GNU Lesser General Public License Usage |
| ** Alternatively, this file may be used under the terms of the GNU Lesser |
| ** General Public License version 3 as published by the Free Software |
| ** Foundation and appearing in the file LICENSE.LGPL3 included in the |
| ** packaging of this file. Please review the following information to |
| ** ensure the GNU Lesser General Public License version 3 requirements |
| ** will be met: https://www.gnu.org/licenses/lgpl-3.0.html. |
| ** |
| ** GNU General Public License Usage |
| ** Alternatively, this file may be used under the terms of the GNU |
| ** General Public License version 2.0 or (at your option) the GNU General |
| ** Public license version 3 or any later version approved by the KDE Free |
| ** Qt Foundation. The licenses are as published by the Free Software |
| ** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3 |
| ** included in the packaging of this file. Please review the following |
| ** information to ensure the GNU General Public License requirements will |
| ** be met: https://www.gnu.org/licenses/gpl-2.0.html and |
| ** https://www.gnu.org/licenses/gpl-3.0.html. |
| ** |
| ** $QT_END_LICENSE$ |
| ** |
| ****************************************************************************/ |
| |
| #include "qsettings.h" |
| |
| #include "qsettings_p.h" |
| #include "qvector.h" |
| #include "qmap.h" |
| #include "qdebug.h" |
| #include <qt_windows.h> |
| |
| // See "Accessing an Alternate Registry View" at: |
| // http://msdn.microsoft.com/en-us/library/aa384129%28VS.85%29.aspx |
| #ifndef KEY_WOW64_64KEY |
| // Access a 32-bit key from either a 32-bit or 64-bit application. |
| # define KEY_WOW64_64KEY 0x0100 |
| #endif |
| |
| #ifndef KEY_WOW64_32KEY |
| // Access a 64-bit key from either a 32-bit or 64-bit application. |
| # define KEY_WOW64_32KEY 0x0200 |
| #endif |
| |
| QT_BEGIN_NAMESPACE |
| |
| /* Keys are stored in QStrings. If the variable name starts with 'u', this is a "user" |
| key, ie. "foo/bar/alpha/beta". If the variable name starts with 'r', this is a "registry" |
| key, ie. "\foo\bar\alpha\beta". */ |
| |
| /******************************************************************************* |
| ** Some convenience functions |
| */ |
| |
| /* |
| We don't use KEY_ALL_ACCESS because it gives more rights than what we |
| need. See task 199061. |
| */ |
| static const REGSAM registryPermissions = KEY_READ | KEY_WRITE; |
| |
| static QString keyPath(const QString &rKey) |
| { |
| int idx = rKey.lastIndexOf(QLatin1Char('\\')); |
| if (idx == -1) |
| return QString(); |
| return rKey.left(idx + 1); |
| } |
| |
| static QString keyName(const QString &rKey) |
| { |
| int idx = rKey.lastIndexOf(QLatin1Char('\\')); |
| |
| QString res; |
| if (idx == -1) |
| res = rKey; |
| else |
| res = rKey.mid(idx + 1); |
| |
| if (res == QLatin1String("Default") || res == QLatin1String(".")) |
| res = QLatin1String(""); |
| |
| return res; |
| } |
| |
| static QString escapedKey(QString uKey) |
| { |
| QChar *data = uKey.data(); |
| int l = uKey.length(); |
| for (int i = 0; i < l; ++i) { |
| ushort &ucs = data[i].unicode(); |
| if (ucs == '\\') |
| ucs = '/'; |
| else if (ucs == '/') |
| ucs = '\\'; |
| } |
| return uKey; |
| } |
| |
| static QString unescapedKey(QString rKey) |
| { |
| return escapedKey(rKey); |
| } |
| |
| typedef QMap<QString, QString> NameSet; |
| |
| static void mergeKeySets(NameSet *dest, const NameSet &src) |
| { |
| NameSet::const_iterator it = src.constBegin(); |
| for (; it != src.constEnd(); ++it) |
| dest->insert(unescapedKey(it.key()), QString()); |
| } |
| |
| static void mergeKeySets(NameSet *dest, const QStringList &src) |
| { |
| QStringList::const_iterator it = src.constBegin(); |
| for (; it != src.constEnd(); ++it) |
| dest->insert(unescapedKey(*it), QString()); |
| } |
| |
| /******************************************************************************* |
| ** Wrappers for the insane windows registry API |
| */ |
| |
| // ### Qt 6: Use new helpers from qwinregistry.cpp (once bootstrap builds are obsolete) |
| |
| // Open a key with the specified "perms". |
| // "access" is to explicitly use the 32- or 64-bit branch. |
| static HKEY openKey(HKEY parentHandle, REGSAM perms, const QString &rSubKey, REGSAM access = 0) |
| { |
| HKEY resultHandle = 0; |
| LONG res = RegOpenKeyEx(parentHandle, reinterpret_cast<const wchar_t *>(rSubKey.utf16()), |
| 0, perms | access, &resultHandle); |
| |
| if (res == ERROR_SUCCESS) |
| return resultHandle; |
| |
| return 0; |
| } |
| |
| // Open a key with the specified "perms", create it if it does not exist. |
| // "access" is to explicitly use the 32- or 64-bit branch. |
| static HKEY createOrOpenKey(HKEY parentHandle, REGSAM perms, const QString &rSubKey, REGSAM access = 0) |
| { |
| // try to open it |
| HKEY resultHandle = openKey(parentHandle, perms, rSubKey, access); |
| if (resultHandle != 0) |
| return resultHandle; |
| |
| // try to create it |
| LONG res = RegCreateKeyEx(parentHandle, reinterpret_cast<const wchar_t *>(rSubKey.utf16()), 0, 0, |
| REG_OPTION_NON_VOLATILE, perms | access, 0, &resultHandle, 0); |
| |
| if (res == ERROR_SUCCESS) |
| return resultHandle; |
| |
| //qErrnoWarning(int(res), "QSettings: Failed to create subkey \"%ls\"", |
| // qUtf16Printable(rSubKey)); |
| |
| return 0; |
| } |
| |
| // Open or create a key in read-write mode if possible, otherwise read-only. |
| // "access" is to explicitly use the 32- or 64-bit branch. |
| static HKEY createOrOpenKey(HKEY parentHandle, const QString &rSubKey, bool *readOnly, REGSAM access = 0) |
| { |
| // try to open or create it read/write |
| HKEY resultHandle = createOrOpenKey(parentHandle, registryPermissions, rSubKey, access); |
| if (resultHandle != 0) { |
| if (readOnly != 0) |
| *readOnly = false; |
| return resultHandle; |
| } |
| |
| // try to open or create it read/only |
| resultHandle = createOrOpenKey(parentHandle, KEY_READ, rSubKey, access); |
| if (resultHandle != 0) { |
| if (readOnly != 0) |
| *readOnly = true; |
| return resultHandle; |
| } |
| return 0; |
| } |
| |
| static QStringList childKeysOrGroups(HKEY parentHandle, QSettingsPrivate::ChildSpec spec) |
| { |
| QStringList result; |
| DWORD numKeys; |
| DWORD maxKeySize; |
| DWORD numSubgroups; |
| DWORD maxSubgroupSize; |
| |
| // Find the number of keys and subgroups, as well as the max of their lengths. |
| LONG res = RegQueryInfoKey(parentHandle, 0, 0, 0, &numSubgroups, &maxSubgroupSize, 0, |
| &numKeys, &maxKeySize, 0, 0, 0); |
| |
| if (res != ERROR_SUCCESS) { |
| qErrnoWarning(int(res), "QSettings: RegQueryInfoKey() failed"); |
| return result; |
| } |
| |
| ++maxSubgroupSize; |
| ++maxKeySize; |
| |
| int n; |
| int m; |
| if (spec == QSettingsPrivate::ChildKeys) { |
| n = numKeys; |
| m = maxKeySize; |
| } else { |
| n = numSubgroups; |
| m = maxSubgroupSize; |
| } |
| |
| /* The size does not include the terminating null character. */ |
| ++m; |
| |
| // Get the list |
| QByteArray buff(m * sizeof(wchar_t), 0); |
| for (int i = 0; i < n; ++i) { |
| QString item; |
| DWORD l = buff.size() / sizeof(wchar_t); |
| if (spec == QSettingsPrivate::ChildKeys) { |
| res = RegEnumValue(parentHandle, i, reinterpret_cast<wchar_t *>(buff.data()), &l, 0, 0, 0, 0); |
| } else { |
| res = RegEnumKeyEx(parentHandle, i, reinterpret_cast<wchar_t *>(buff.data()), &l, 0, 0, 0, 0); |
| } |
| if (res == ERROR_SUCCESS) |
| item = QString::fromWCharArray((const wchar_t *)buff.constData(), l); |
| |
| if (res != ERROR_SUCCESS) { |
| qErrnoWarning(int(res), "QSettings: RegEnumValue failed"); |
| continue; |
| } |
| if (item.isEmpty()) |
| item = QLatin1String("."); |
| result.append(item); |
| } |
| return result; |
| } |
| |
| static void allKeys(HKEY parentHandle, const QString &rSubKey, NameSet *result, REGSAM access = 0) |
| { |
| HKEY handle = openKey(parentHandle, KEY_READ, rSubKey, access); |
| if (handle == 0) |
| return; |
| |
| QStringList childKeys = childKeysOrGroups(handle, QSettingsPrivate::ChildKeys); |
| QStringList childGroups = childKeysOrGroups(handle, QSettingsPrivate::ChildGroups); |
| RegCloseKey(handle); |
| |
| for (int i = 0; i < childKeys.size(); ++i) { |
| QString s = rSubKey; |
| if (!s.isEmpty()) |
| s += QLatin1Char('\\'); |
| s += childKeys.at(i); |
| result->insert(s, QString()); |
| } |
| |
| for (int i = 0; i < childGroups.size(); ++i) { |
| QString s = rSubKey; |
| if (!s.isEmpty()) |
| s += QLatin1Char('\\'); |
| s += childGroups.at(i); |
| allKeys(parentHandle, s, result, access); |
| } |
| } |
| |
| static void deleteChildGroups(HKEY parentHandle, REGSAM access = 0) |
| { |
| QStringList childGroups = childKeysOrGroups(parentHandle, QSettingsPrivate::ChildGroups); |
| |
| for (int i = 0; i < childGroups.size(); ++i) { |
| QString group = childGroups.at(i); |
| |
| // delete subgroups in group |
| HKEY childGroupHandle = openKey(parentHandle, registryPermissions, group, access); |
| if (childGroupHandle == 0) |
| continue; |
| deleteChildGroups(childGroupHandle, access); |
| RegCloseKey(childGroupHandle); |
| |
| // delete group itself |
| LONG res = RegDeleteKey(parentHandle, reinterpret_cast<const wchar_t *>(group.utf16())); |
| if (res != ERROR_SUCCESS) { |
| qErrnoWarning(int(res), "QSettings: RegDeleteKey failed on subkey \"%ls\"", |
| qUtf16Printable(group)); |
| return; |
| } |
| } |
| } |
| |
| /******************************************************************************* |
| ** class RegistryKey |
| */ |
| |
| class RegistryKey |
| { |
| public: |
| RegistryKey(HKEY parent_handle = 0, const QString &key = QString(), bool read_only = true, REGSAM access = 0); |
| QString key() const; |
| HKEY handle() const; |
| HKEY parentHandle() const; |
| bool readOnly() const; |
| void close(); |
| private: |
| HKEY m_parent_handle; |
| mutable HKEY m_handle; |
| QString m_key; |
| mutable bool m_read_only; |
| REGSAM m_access; |
| }; |
| |
| RegistryKey::RegistryKey(HKEY parent_handle, const QString &key, bool read_only, REGSAM access) |
| : m_parent_handle(parent_handle), |
| m_handle(0), |
| m_key(key), |
| m_read_only(read_only), |
| m_access(access) |
| { |
| } |
| |
| QString RegistryKey::key() const |
| { |
| return m_key; |
| } |
| |
| HKEY RegistryKey::handle() const |
| { |
| if (m_handle != 0) |
| return m_handle; |
| |
| if (m_read_only) |
| m_handle = openKey(m_parent_handle, KEY_READ, m_key, m_access); |
| else |
| m_handle = createOrOpenKey(m_parent_handle, m_key, &m_read_only, m_access); |
| |
| return m_handle; |
| } |
| |
| HKEY RegistryKey::parentHandle() const |
| { |
| return m_parent_handle; |
| } |
| |
| bool RegistryKey::readOnly() const |
| { |
| return m_read_only; |
| } |
| |
| void RegistryKey::close() |
| { |
| if (m_handle != 0) |
| RegCloseKey(m_handle); |
| m_handle = 0; |
| } |
| |
| typedef QVector<RegistryKey> RegistryKeyList; |
| |
| /******************************************************************************* |
| ** class QWinSettingsPrivate |
| */ |
| |
| class QWinSettingsPrivate : public QSettingsPrivate |
| { |
| Q_DISABLE_COPY(QWinSettingsPrivate) |
| public: |
| QWinSettingsPrivate(QSettings::Scope scope, const QString &organization, |
| const QString &application, REGSAM access = 0); |
| QWinSettingsPrivate(QString rKey, REGSAM access = 0); |
| ~QWinSettingsPrivate() override; |
| |
| void remove(const QString &uKey) override; |
| void set(const QString &uKey, const QVariant &value) override; |
| bool get(const QString &uKey, QVariant *value) const override; |
| QStringList children(const QString &uKey, ChildSpec spec) const override; |
| void clear() override; |
| void sync() override; |
| void flush() override; |
| bool isWritable() const override; |
| HKEY writeHandle() const; |
| bool readKey(HKEY parentHandle, const QString &rSubKey, QVariant *value) const; |
| QString fileName() const override; |
| |
| private: |
| RegistryKeyList regList; // list of registry locations to search for keys |
| bool deleteWriteHandleOnExit; |
| REGSAM access; |
| }; |
| |
| QWinSettingsPrivate::QWinSettingsPrivate(QSettings::Scope scope, const QString &organization, |
| const QString &application, REGSAM access) |
| : QSettingsPrivate(QSettings::NativeFormat, scope, organization, application), |
| access(access) |
| { |
| deleteWriteHandleOnExit = false; |
| |
| if (!organization.isEmpty()) { |
| QString prefix = QLatin1String("Software\\") + organization; |
| QString orgPrefix = prefix + QLatin1String("\\OrganizationDefaults"); |
| QString appPrefix = prefix + QLatin1Char('\\') + application; |
| |
| if (scope == QSettings::UserScope) { |
| if (!application.isEmpty()) |
| regList.append(RegistryKey(HKEY_CURRENT_USER, appPrefix, !regList.isEmpty(), access)); |
| |
| regList.append(RegistryKey(HKEY_CURRENT_USER, orgPrefix, !regList.isEmpty(), access)); |
| } |
| |
| if (!application.isEmpty()) |
| regList.append(RegistryKey(HKEY_LOCAL_MACHINE, appPrefix, !regList.isEmpty(), access)); |
| |
| regList.append(RegistryKey(HKEY_LOCAL_MACHINE, orgPrefix, !regList.isEmpty(), access)); |
| } |
| |
| if (regList.isEmpty()) |
| setStatus(QSettings::AccessError); |
| } |
| |
| QWinSettingsPrivate::QWinSettingsPrivate(QString rPath, REGSAM access) |
| : QSettingsPrivate(QSettings::NativeFormat), |
| access(access) |
| { |
| deleteWriteHandleOnExit = false; |
| |
| if (rPath.startsWith(QLatin1Char('\\'))) |
| rPath.remove(0, 1); |
| |
| int keyLength; |
| HKEY keyName; |
| |
| if (rPath.startsWith(QLatin1String("HKEY_CURRENT_USER"))) { |
| keyLength = 17; |
| keyName = HKEY_CURRENT_USER; |
| } else if (rPath.startsWith(QLatin1String("HKCU"))) { |
| keyLength = 4; |
| keyName = HKEY_CURRENT_USER; |
| } else if (rPath.startsWith(QLatin1String("HKEY_LOCAL_MACHINE"))) { |
| keyLength = 18; |
| keyName = HKEY_LOCAL_MACHINE; |
| } else if (rPath.startsWith(QLatin1String("HKLM"))) { |
| keyLength = 4; |
| keyName = HKEY_LOCAL_MACHINE; |
| } else if (rPath.startsWith(QLatin1String("HKEY_CLASSES_ROOT"))) { |
| keyLength = 17; |
| keyName = HKEY_CLASSES_ROOT; |
| } else if (rPath.startsWith(QLatin1String("HKCR"))) { |
| keyLength = 4; |
| keyName = HKEY_CLASSES_ROOT; |
| } else if (rPath.startsWith(QLatin1String("HKEY_USERS"))) { |
| keyLength = 10; |
| keyName = HKEY_USERS; |
| } else if (rPath.startsWith(QLatin1String("HKU"))) { |
| keyLength = 3; |
| keyName = HKEY_USERS; |
| } else { |
| return; |
| } |
| |
| if (rPath.length() == keyLength) |
| regList.append(RegistryKey(keyName, QString(), false, access)); |
| else if (rPath[keyLength] == QLatin1Char('\\')) |
| regList.append(RegistryKey(keyName, rPath.mid(keyLength+1), false, access)); |
| } |
| |
| bool QWinSettingsPrivate::readKey(HKEY parentHandle, const QString &rSubKey, QVariant *value) const |
| { |
| QString rSubkeyName = keyName(rSubKey); |
| QString rSubkeyPath = keyPath(rSubKey); |
| |
| // open a handle on the subkey |
| HKEY handle = openKey(parentHandle, KEY_READ, rSubkeyPath, access); |
| if (handle == 0) |
| return false; |
| |
| // get the size and type of the value |
| DWORD dataType; |
| DWORD dataSize; |
| LONG res = RegQueryValueEx(handle, reinterpret_cast<const wchar_t *>(rSubkeyName.utf16()), 0, &dataType, 0, &dataSize); |
| if (res != ERROR_SUCCESS) { |
| RegCloseKey(handle); |
| return false; |
| } |
| |
| // workaround for rare cases where trailing '\0' are missing in registry |
| if (dataType == REG_SZ || dataType == REG_EXPAND_SZ) |
| dataSize += 2; |
| else if (dataType == REG_MULTI_SZ) |
| dataSize += 4; |
| |
| // get the value |
| QByteArray data(dataSize, 0); |
| res = RegQueryValueEx(handle, reinterpret_cast<const wchar_t *>(rSubkeyName.utf16()), 0, 0, |
| reinterpret_cast<unsigned char*>(data.data()), &dataSize); |
| if (res != ERROR_SUCCESS) { |
| RegCloseKey(handle); |
| return false; |
| } |
| |
| switch (dataType) { |
| case REG_EXPAND_SZ: |
| case REG_SZ: { |
| QString s; |
| if (dataSize) { |
| s = QString::fromWCharArray(reinterpret_cast<const wchar_t *>(data.constData())); |
| } |
| if (value != 0) |
| *value = stringToVariant(s); |
| break; |
| } |
| |
| case REG_MULTI_SZ: { |
| QStringList l; |
| if (dataSize) { |
| int i = 0; |
| for (;;) { |
| QString s = QString::fromWCharArray(reinterpret_cast<const wchar_t *>(data.constData()) + i); |
| i += s.length() + 1; |
| |
| if (s.isEmpty()) |
| break; |
| l.append(s); |
| } |
| } |
| if (value != 0) |
| *value = stringListToVariantList(l); |
| break; |
| } |
| |
| case REG_NONE: |
| case REG_BINARY: { |
| QString s; |
| if (dataSize) { |
| s = QString::fromWCharArray(reinterpret_cast<const wchar_t *>(data.constData()), data.size() / 2); |
| } |
| if (value != 0) |
| *value = stringToVariant(s); |
| break; |
| } |
| |
| case REG_DWORD_BIG_ENDIAN: |
| case REG_DWORD: { |
| Q_ASSERT(data.size() == sizeof(int)); |
| int i; |
| memcpy(reinterpret_cast<char*>(&i), data.constData(), sizeof(int)); |
| if (value != 0) |
| *value = i; |
| break; |
| } |
| |
| case REG_QWORD: { |
| Q_ASSERT(data.size() == sizeof(qint64)); |
| qint64 i; |
| memcpy(reinterpret_cast<char*>(&i), data.constData(), sizeof(qint64)); |
| if (value != 0) |
| *value = i; |
| break; |
| } |
| |
| default: |
| qWarning("QSettings: Unknown data %d type in Windows registry", static_cast<int>(dataType)); |
| if (value != 0) |
| *value = QVariant(); |
| break; |
| } |
| |
| RegCloseKey(handle); |
| return true; |
| } |
| |
| HKEY QWinSettingsPrivate::writeHandle() const |
| { |
| if (regList.isEmpty()) |
| return 0; |
| const RegistryKey &key = regList.at(0); |
| if (key.handle() == 0 || key.readOnly()) |
| return 0; |
| return key.handle(); |
| } |
| |
| QWinSettingsPrivate::~QWinSettingsPrivate() |
| { |
| if (deleteWriteHandleOnExit && writeHandle() != 0) { |
| QString emptyKey; |
| DWORD res = RegDeleteKey(writeHandle(), reinterpret_cast<const wchar_t *>(emptyKey.utf16())); |
| if (res != ERROR_SUCCESS) { |
| qErrnoWarning(int(res), "QSettings: Failed to delete key \"%ls\"", |
| qUtf16Printable(regList.constFirst().key())); |
| } |
| } |
| |
| for (int i = 0; i < regList.size(); ++i) |
| regList[i].close(); |
| } |
| |
| void QWinSettingsPrivate::remove(const QString &uKey) |
| { |
| if (writeHandle() == 0) { |
| setStatus(QSettings::AccessError); |
| return; |
| } |
| |
| QString rKey = escapedKey(uKey); |
| |
| // try to delete value bar in key foo |
| LONG res; |
| HKEY handle = openKey(writeHandle(), registryPermissions, keyPath(rKey), access); |
| if (handle != 0) { |
| res = RegDeleteValue(handle, reinterpret_cast<const wchar_t *>(keyName(rKey).utf16())); |
| RegCloseKey(handle); |
| } |
| |
| // try to delete key foo/bar and all subkeys |
| handle = openKey(writeHandle(), registryPermissions, rKey, access); |
| if (handle != 0) { |
| deleteChildGroups(handle, access); |
| |
| if (rKey.isEmpty()) { |
| const QStringList childKeys = childKeysOrGroups(handle, QSettingsPrivate::ChildKeys); |
| |
| for (const QString &group : childKeys) { |
| LONG res = RegDeleteValue(handle, reinterpret_cast<const wchar_t *>(group.utf16())); |
| if (res != ERROR_SUCCESS) { |
| qErrnoWarning(int(res), "QSettings: RegDeleteValue failed on subkey \"%ls\"", |
| qUtf16Printable(group)); |
| } |
| } |
| } else { |
| res = RegDeleteKey(writeHandle(), reinterpret_cast<const wchar_t *>(rKey.utf16())); |
| |
| if (res != ERROR_SUCCESS) { |
| qErrnoWarning(int(res), "QSettings: RegDeleteKey failed on key \"%ls\"", |
| qUtf16Printable(rKey)); |
| } |
| } |
| RegCloseKey(handle); |
| } |
| } |
| |
| void QWinSettingsPrivate::set(const QString &uKey, const QVariant &value) |
| { |
| if (writeHandle() == 0) { |
| setStatus(QSettings::AccessError); |
| return; |
| } |
| |
| QString rKey = escapedKey(uKey); |
| |
| HKEY handle = createOrOpenKey(writeHandle(), registryPermissions, keyPath(rKey), access); |
| if (handle == 0) { |
| setStatus(QSettings::AccessError); |
| return; |
| } |
| |
| DWORD type; |
| QByteArray regValueBuff; |
| |
| // Determine the type |
| switch (value.type()) { |
| case QVariant::List: |
| case QVariant::StringList: { |
| // If none of the elements contains '\0', we can use REG_MULTI_SZ, the |
| // native registry string list type. Otherwise we use REG_BINARY. |
| type = REG_MULTI_SZ; |
| QStringList l = variantListToStringList(value.toList()); |
| QStringList::const_iterator it = l.constBegin(); |
| for (; it != l.constEnd(); ++it) { |
| if ((*it).length() == 0 || it->contains(QChar::Null)) { |
| type = REG_BINARY; |
| break; |
| } |
| } |
| |
| if (type == REG_BINARY) { |
| QString s = variantToString(value); |
| regValueBuff = QByteArray(reinterpret_cast<const char*>(s.utf16()), s.length() * 2); |
| } else { |
| QStringList::const_iterator it = l.constBegin(); |
| for (; it != l.constEnd(); ++it) { |
| const QString &s = *it; |
| regValueBuff += QByteArray(reinterpret_cast<const char*>(s.utf16()), (s.length() + 1) * 2); |
| } |
| regValueBuff.append((char)0); |
| regValueBuff.append((char)0); |
| } |
| break; |
| } |
| |
| case QVariant::Int: |
| case QVariant::UInt: { |
| type = REG_DWORD; |
| qint32 i = value.toInt(); |
| regValueBuff = QByteArray(reinterpret_cast<const char*>(&i), sizeof(qint32)); |
| break; |
| } |
| |
| case QVariant::LongLong: |
| case QVariant::ULongLong: { |
| type = REG_QWORD; |
| qint64 i = value.toLongLong(); |
| regValueBuff = QByteArray(reinterpret_cast<const char*>(&i), sizeof(qint64)); |
| break; |
| } |
| |
| case QVariant::ByteArray: |
| Q_FALLTHROUGH(); |
| |
| default: { |
| // If the string does not contain '\0', we can use REG_SZ, the native registry |
| // string type. Otherwise we use REG_BINARY. |
| QString s = variantToString(value); |
| type = s.contains(QChar::Null) ? REG_BINARY : REG_SZ; |
| int length = s.length(); |
| if (type == REG_SZ) |
| ++length; |
| regValueBuff = QByteArray(reinterpret_cast<const char *>(s.utf16()), |
| int(sizeof(wchar_t)) * length); |
| break; |
| } |
| } |
| |
| // set the value |
| LONG res = RegSetValueEx(handle, reinterpret_cast<const wchar_t *>(keyName(rKey).utf16()), 0, type, |
| reinterpret_cast<const unsigned char*>(regValueBuff.constData()), |
| regValueBuff.size()); |
| |
| if (res == ERROR_SUCCESS) { |
| deleteWriteHandleOnExit = false; |
| } else { |
| qErrnoWarning(int(res), "QSettings: failed to set subkey \"%ls\"", |
| qUtf16Printable(rKey)); |
| setStatus(QSettings::AccessError); |
| } |
| |
| RegCloseKey(handle); |
| } |
| |
| bool QWinSettingsPrivate::get(const QString &uKey, QVariant *value) const |
| { |
| QString rKey = escapedKey(uKey); |
| |
| for (const RegistryKey &r : regList) { |
| HKEY handle = r.handle(); |
| if (handle != 0 && readKey(handle, rKey, value)) |
| return true; |
| |
| if (!fallbacks) |
| return false; |
| } |
| |
| return false; |
| } |
| |
| QStringList QWinSettingsPrivate::children(const QString &uKey, ChildSpec spec) const |
| { |
| NameSet result; |
| QString rKey = escapedKey(uKey); |
| |
| for (const RegistryKey &r : regList) { |
| HKEY parent_handle = r.handle(); |
| if (parent_handle == 0) |
| continue; |
| HKEY handle = openKey(parent_handle, KEY_READ, rKey, access); |
| if (handle == 0) |
| continue; |
| |
| if (spec == AllKeys) { |
| NameSet keys; |
| allKeys(handle, QLatin1String(""), &keys, access); |
| mergeKeySets(&result, keys); |
| } else { // ChildGroups or ChildKeys |
| QStringList names = childKeysOrGroups(handle, spec); |
| mergeKeySets(&result, names); |
| } |
| |
| RegCloseKey(handle); |
| |
| if (!fallbacks) |
| return result.keys(); |
| } |
| |
| return result.keys(); |
| } |
| |
| void QWinSettingsPrivate::clear() |
| { |
| remove(QString()); |
| deleteWriteHandleOnExit = true; |
| } |
| |
| void QWinSettingsPrivate::sync() |
| { |
| RegFlushKey(writeHandle()); |
| } |
| |
| void QWinSettingsPrivate::flush() |
| { |
| // Windows does this for us. |
| } |
| |
| QString QWinSettingsPrivate::fileName() const |
| { |
| if (regList.isEmpty()) |
| return QString(); |
| |
| const RegistryKey &key = regList.at(0); |
| QString result; |
| if (key.parentHandle() == HKEY_CURRENT_USER) |
| result = QLatin1String("\\HKEY_CURRENT_USER\\"); |
| else |
| result = QLatin1String("\\HKEY_LOCAL_MACHINE\\"); |
| |
| return result + regList.at(0).key(); |
| } |
| |
| bool QWinSettingsPrivate::isWritable() const |
| { |
| return writeHandle() != 0; |
| } |
| |
| QSettingsPrivate *QSettingsPrivate::create(QSettings::Format format, QSettings::Scope scope, |
| const QString &organization, const QString &application) |
| { |
| switch (format) { |
| case QSettings::NativeFormat: |
| return new QWinSettingsPrivate(scope, organization, application); |
| case QSettings::Registry32Format: |
| return new QWinSettingsPrivate(scope, organization, application, KEY_WOW64_32KEY); |
| case QSettings::Registry64Format: |
| return new QWinSettingsPrivate(scope, organization, application, KEY_WOW64_64KEY); |
| default: |
| break; |
| } |
| return new QConfFileSettingsPrivate(format, scope, organization, application); |
| } |
| |
| QSettingsPrivate *QSettingsPrivate::create(const QString &fileName, QSettings::Format format) |
| { |
| switch (format) { |
| case QSettings::NativeFormat: |
| return new QWinSettingsPrivate(fileName); |
| case QSettings::Registry32Format: |
| return new QWinSettingsPrivate(fileName, KEY_WOW64_32KEY); |
| case QSettings::Registry64Format: |
| return new QWinSettingsPrivate(fileName, KEY_WOW64_64KEY); |
| default: |
| break; |
| } |
| return new QConfFileSettingsPrivate(fileName, format); |
| } |
| |
| QT_END_NAMESPACE |