278 lines
7.2 KiB
C++
278 lines
7.2 KiB
C++
/*
|
|
* 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/.
|
|
*
|
|
* Copyright 2015 Danny Robson <danny@nerdcruft.net>
|
|
*/
|
|
|
|
#include "registry.hpp"
|
|
|
|
#include "../cast.hpp"
|
|
#include "except.hpp"
|
|
#include <cruft/util/debug/panic.hpp>
|
|
|
|
#include <winerror.h>
|
|
|
|
#include <string>
|
|
#include <cstdint>
|
|
|
|
using cruft::win32::key;
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
typedef uint64_t QWORD;
|
|
|
|
//-----------------------------------------------------------------------------
|
|
template <typename T> constexpr DWORD type_to_id (void);
|
|
|
|
template <> constexpr DWORD type_to_id<DWORD> (void) { return REG_DWORD; }
|
|
template <> constexpr DWORD type_to_id<QWORD> (void) { return REG_QWORD; }
|
|
template <> constexpr DWORD type_to_id<void> (void) { return REG_NONE; }
|
|
template <> constexpr DWORD type_to_id<std::string> (void) { return REG_SZ; }
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
template <typename T> constexpr DWORD restrict_to_id (void);
|
|
|
|
template <> constexpr DWORD restrict_to_id<DWORD> (void) { return RRF_RT_DWORD; }
|
|
template <> constexpr DWORD restrict_to_id<QWORD> (void) { return RRF_RT_QWORD; }
|
|
template <> constexpr DWORD restrict_to_id<void> (void) { return RRF_RT_REG_NONE; }
|
|
template <> constexpr DWORD restrict_to_id<std::string> (void) { return RRF_RT_REG_SZ; }
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
key::key (key const &parent, char const *path, REGSAM rights)
|
|
{
|
|
auto err = RegOpenKeyEx (parent.m_handle, path, 0, rights, &m_handle);
|
|
win32::error::try_code (err);
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
key::key (HKEY root, const char *child, REGSAM rights)
|
|
{
|
|
auto err = RegOpenKeyEx (root, child, 0, rights, &m_handle);
|
|
win32::error::try_code (err);
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
key::~key ()
|
|
{
|
|
auto err = RegCloseKey (m_handle);
|
|
win32::error::try_code (err);
|
|
}
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
cruft::view<key::child_iterator>
|
|
key::subkeys (void)
|
|
{
|
|
DWORD size;
|
|
|
|
auto err = RegQueryInfoKey (
|
|
m_handle,
|
|
nullptr,
|
|
nullptr,
|
|
nullptr,
|
|
&size,
|
|
nullptr, // longest subkey size
|
|
nullptr, // longest class string
|
|
nullptr, // number of values for this key
|
|
nullptr, // longest value name
|
|
nullptr, // longest value data
|
|
nullptr, // security descriptor
|
|
nullptr // last write time
|
|
);
|
|
error::try_code (err);
|
|
|
|
return {
|
|
child_iterator (*this),
|
|
child_iterator (*this, size)
|
|
};
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
key::child_iterator::child_iterator (key const &_parent)
|
|
: child_iterator (_parent, 0)
|
|
{ ; }
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
key::child_iterator::child_iterator (key const &_parent, int _index)
|
|
: m_parent (_parent)
|
|
, m_index (_index)
|
|
{ ; }
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
key
|
|
key::child_iterator::operator* (void) const
|
|
{
|
|
DWORD name_size = strlen ("{00000000-0000-0000-0000-000000000000}");
|
|
std::string name (name_size, '\0');
|
|
|
|
while (1) {
|
|
auto const err = RegEnumKeyEx(
|
|
m_parent.m_handle,
|
|
m_index,
|
|
name.data(), &name_size,
|
|
nullptr,
|
|
nullptr, nullptr,
|
|
nullptr
|
|
);
|
|
|
|
switch (err) {
|
|
case ERROR_SUCCESS:
|
|
name.resize (name_size);
|
|
return key (m_parent, name.c_str ());
|
|
|
|
case ERROR_MORE_DATA:
|
|
name_size *= 2;
|
|
name.resize (name_size);
|
|
continue;
|
|
|
|
default:
|
|
error::throw_code (err);
|
|
}
|
|
}
|
|
|
|
unreachable ();
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
key::child_iterator&
|
|
key::child_iterator::operator++ ()
|
|
{
|
|
++m_index;
|
|
return *this;
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
bool
|
|
key::child_iterator::operator== (child_iterator const &rhs)
|
|
{
|
|
CHECK_EQ (&m_parent, &rhs.m_parent);
|
|
return m_index == rhs.m_index;
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
bool
|
|
key::child_iterator::operator!= (child_iterator const &rhs)
|
|
{
|
|
return !(*this == rhs);
|
|
}
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
std::string
|
|
key::name (void) const
|
|
{
|
|
// allocate an initial buffer large enough to contain a GUID given that
|
|
// strings like these are pretty common.
|
|
DWORD size = strlen ("{00000000-0000-0000-0000-000000000000}");
|
|
std::string value (size, '\0');
|
|
|
|
// keep attempting to read the name of the key. if it fails then we double
|
|
// the size of the string and try again.
|
|
while (1) {
|
|
auto res = RegQueryInfoKey (
|
|
m_handle,
|
|
value.data (), &size,
|
|
nullptr, nullptr, nullptr, nullptr, nullptr, nullptr, nullptr, nullptr, nullptr
|
|
);
|
|
|
|
switch (res) {
|
|
case ERROR_SUCCESS:
|
|
value.resize (size);
|
|
return value;
|
|
|
|
case ERROR_MORE_DATA:
|
|
size *= 2;
|
|
continue;
|
|
|
|
default:
|
|
win32::error::throw_code (res);
|
|
}
|
|
}
|
|
|
|
unreachable ();
|
|
}
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
template <typename T>
|
|
T
|
|
key::data (const char *name) const
|
|
{
|
|
T value;
|
|
DWORD type;
|
|
DWORD size = sizeof (value);
|
|
|
|
auto err = RegGetValue (m_handle, name, &value, restrict_to_id<T> (), &type, &value, &size);
|
|
win32::error::try_code (err);
|
|
|
|
return value;
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
template <>
|
|
std::string
|
|
key::data (char const *name) const
|
|
{
|
|
DWORD size = strlen ("{00000000-0000-0000-0000-000000000000}");
|
|
std::string value (size, '\0');
|
|
|
|
while (1) {
|
|
DWORD type;
|
|
auto const err = RegGetValue(
|
|
m_handle, nullptr, name,
|
|
restrict_to_id<std::string>(), &type,
|
|
value.data(), &size
|
|
);
|
|
|
|
switch (err) {
|
|
case ERROR_SUCCESS:
|
|
value.resize (size);
|
|
return value;
|
|
|
|
case ERROR_MORE_DATA:
|
|
size *= 2;
|
|
value.resize (size);
|
|
continue;
|
|
|
|
default:
|
|
error::throw_code (err);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
std::set<std::string>
|
|
key::values (void) const
|
|
{
|
|
std::set<std::string> all;
|
|
|
|
for (DWORD i = 0; ; ++i) {
|
|
std::string name (255, '\0');
|
|
DWORD size = cruft::cast::narrow<DWORD> (name.size ());
|
|
|
|
auto err = RegEnumValue (m_handle, i, &name[0], &size, nullptr, nullptr, nullptr, nullptr);
|
|
if (ERROR_NO_MORE_ITEMS == err)
|
|
return all;
|
|
if (ERROR_SUCCESS != err)
|
|
win32::error::throw_code (err);
|
|
|
|
CHECK_GT (size, 0u);
|
|
name.resize (size);
|
|
|
|
all.insert (std::move (name));
|
|
}
|
|
}
|