libcruft-util/win32/registry.cpp

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 "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));
}
}