pcsx2/3rdparty/winwil/include/wil/registry_helpers.h

1951 lines
78 KiB
C
Raw Normal View History

//*********************************************************
//
// Copyright (c) Microsoft. All rights reserved.
// This code is licensed under the MIT License.
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF
// ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
// TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A
// PARTICULAR PURPOSE AND NONINFRINGEMENT.
//
//*********************************************************
//! @file
//! Helpers for iterating over keys and values in the registry.
#ifndef __WIL_REGISTRY_HELPERS_INCLUDED
#define __WIL_REGISTRY_HELPERS_INCLUDED
#if defined(_STRING_) || defined(_VECTOR_) || (defined(__cpp_lib_optional) && defined(_OPTIONAL_)) || defined(WIL_DOXYGEN)
#include <functional>
#include <iterator>
#endif
#include <stdint.h>
#include <Windows.h>
#include "resource.h"
#ifdef _KERNEL_MODE
#error This header is not supported in kernel-mode.
#endif
namespace wil
{
namespace reg
{
/**
* @brief Helper function to translate registry return values if the value was not found
* @param hr HRESULT to test from registry APIs
* @return boolean if the HRESULT indicates the registry value was not found
*/
constexpr bool is_registry_not_found(HRESULT hr) WI_NOEXCEPT
{
return (hr == __HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND)) || (hr == __HRESULT_FROM_WIN32(ERROR_PATH_NOT_FOUND));
}
/**
* @brief Helper function to translate registry return values if the buffer was too small
* @param hr HRESULT to test from registry APIs
* @return boolean if the HRESULT indicates the buffer was too small for the value being read
*/
constexpr bool is_registry_buffer_too_small(HRESULT hr) WI_NOEXCEPT
{
return hr == __HRESULT_FROM_WIN32(ERROR_MORE_DATA);
}
// Access rights for opening registry keys. See https://learn.microsoft.com/en-us/windows/win32/sysinfo/registry-key-security-and-access-rights.
enum class key_access
{
// Open key for reading.
read,
// Open key for reading and writing. Equivalent to KEY_ALL_ACCESS.
readwrite,
};
/// @cond
namespace reg_view_details
{
constexpr DWORD get_value_flags_from_value_type(DWORD type) WI_NOEXCEPT
{
switch (type)
{
case REG_DWORD:
return RRF_RT_REG_DWORD;
case REG_QWORD:
return RRF_RT_REG_QWORD;
case REG_SZ:
return RRF_RT_REG_SZ | RRF_RT_REG_EXPAND_SZ | RRF_NOEXPAND;
case REG_EXPAND_SZ:
return RRF_RT_REG_SZ | RRF_RT_REG_EXPAND_SZ;
case REG_MULTI_SZ:
return RRF_RT_REG_MULTI_SZ;
case REG_BINARY:
return RRF_RT_REG_BINARY;
// the caller can directly specify their own flags if they need to
default:
return type;
}
}
constexpr DWORD get_access_flags(key_access access) WI_NOEXCEPT
{
switch (access)
{
case key_access::read:
return KEY_READ;
case key_access::readwrite:
return KEY_ALL_ACCESS;
}
FAIL_FAST();
RESULT_NORETURN_RESULT(0);
}
/**
* @brief A utility function that walks a contiguous wchar_t container looking for strings within a multi-string
* @tparam InputIt An iterator type that reference a container that holds wchar_t characters to translate into individual
* strings
* @tparam Fn A callback function to be called each time a string is found - given the [begin, end] iterators referencing
* the found string
* @param first An iterator referencing to the beginning of the target container (like a std::begin iterator)
* @param last An iterator referencing one-past-the-end of the target container (like a std::end iterator)
* @param func A callback function to be called each time a string is found - given the [begin, end] iterators referencing
* the found string
*/
template <class InputIt, class Fn>
void walk_multistring(const InputIt& first, const InputIt& last, Fn func)
{
auto current = first;
const auto end_iterator = last;
const auto last_null = (end_iterator - 1);
while (current != end_iterator)
{
// hand rolling ::std::find(current, end_iterator, L'\0');
// as this may be called when <algorithm> isn't available
auto next = current;
while (next != end_iterator && *next != L'\0')
{
++next;
}
if (next != end_iterator)
{
// don't add an empty string for the final 2nd-null-terminator
if (next != last_null)
{
// call the function provided with the [begin, end] pair referencing a string found
func(current, next);
}
current = next + 1;
}
else
{
current = next;
}
}
}
#if defined(_VECTOR_) && defined(_STRING_) && defined(WIL_ENABLE_EXCEPTIONS)
/**
* @brief A translation function taking iterators referencing std::wstring objects and returns a corresponding
* std::vector<wchar_t> to be written to a MULTI_SZ registry value. The translation follows the rules for how
* MULTI_SZ registry values should be formatted, notably how null characters should be embedded within the returned
* vector
* @tparam InputIt An iterator type that references a container that holds std::wstring objects to translate into a
* wchar_t buffer
* @param first An iterator referencing to the beginning of the target container (like a std::begin iterator)
* @param last An iterator referencing one-past-the-end of the target container (like a std::end iterator)
* @return A std::vector<wchar_t> with the raw wchar_t buffer of bytes prepared to write to a MULTI_SZ registry value
*/
template <class InputIt>
::std::vector<wchar_t> get_multistring_from_wstrings(const InputIt& first, const InputIt& last)
{
::std::vector<wchar_t> multistring;
if (first == last)
{
multistring.push_back(L'\0');
multistring.push_back(L'\0');
return multistring;
}
for (const auto& wstr : ::wil::make_range(first, last))
{
multistring.insert(multistring.end(), ::std::begin(wstr), ::std::end(wstr));
multistring.push_back(L'\0');
}
// double-null-terminate the last string
multistring.push_back(L'\0');
return multistring;
}
/**
* @brief A translation function taking iterators referencing wchar_t characters and returns extracted individual
* std::wstring objects. The translation follows the rules for how MULTI_SZ registry value can be formatted,
* notably with embedded null characters. Note that this conversion avoids returning empty std::wstring objects
* even though the input may contain contiguous null wchar_t values
* @tparam InputIt An iterator type that reference a container that holds wchar_t characters to translate into individual
* strings
* @param first An iterator referencing to the beginning of the target container (like a std::begin iterator)
* @param last An iterator referencing one-past-the-end of the target container (like a std::end iterator)
* @return A std::vector<std::wstring> of the extracted strings from the input container of wchar_t characters
*/
template <class InputIt>
::std::vector<::std::wstring> get_wstring_vector_from_multistring(const InputIt& first, const InputIt& last)
{
if (last - first < 3)
{
// it doesn't have the required 2 terminating null characters - return an empty string
return ::std::vector<::std::wstring>(1);
}
::std::vector<::std::wstring> strings;
walk_multistring(first, last, [&](const InputIt& string_first, const InputIt& string_last) {
strings.emplace_back(string_first, string_last);
});
return strings;
}
#endif // #if defined(_VECTOR_) && defined(_STRING_) && defined(WIL_ENABLE_EXCEPTIONS)
#if defined(__WIL_OBJBASE_H_)
template <size_t C>
void get_multistring_bytearray_from_strings_nothrow(const PCWSTR data[C], ::wil::unique_cotaskmem_array_ptr<BYTE>& multistring) WI_NOEXCEPT
{
constexpr uint8_t nullTermination[2]{0x00, 0x00};
size_t total_array_length_bytes = 0;
for (size_t i = 0; i < C; ++i)
{
total_array_length_bytes += wcslen(data[i]) * sizeof(wchar_t);
total_array_length_bytes += sizeof(wchar_t); // plus one for the null-terminator
}
total_array_length_bytes += sizeof(wchar_t); // plus one for the ending double-null-terminator
*multistring.addressof() = static_cast<uint8_t*>(::CoTaskMemAlloc(total_array_length_bytes));
if (!multistring.get())
{
multistring.reset();
return;
}
*multistring.size_address() = total_array_length_bytes;
size_t array_offset = 0;
for (size_t i = 0; i < C; ++i)
{
const auto string_length_bytes = wcslen(data[i]) * sizeof(wchar_t);
memcpy(multistring.get() + array_offset, data[i], string_length_bytes);
array_offset += string_length_bytes;
static_assert(sizeof(nullTermination) == sizeof(wchar_t), "null terminator must be a wchar");
memcpy(multistring.get() + array_offset, nullTermination, sizeof(nullTermination));
array_offset += sizeof(nullTermination);
}
// double-null-terminate the last string
memcpy(multistring.get() + array_offset, nullTermination, sizeof(nullTermination));
}
/**
* @brief A translation function taking iterators referencing wchar_t characters and returns extracted individual
* wil::unique_cotaskmem_string objects. The translation follows the rules for how MULTI_SZ registry value can be
* formatted, notably with embedded null characters. Note that this conversion avoids returning empty
* wil::unique_cotaskmem_string objects even though the input may contain contiguous null wchar_t values
* @tparam InputIt An iterator type that reference a container that holds wchar_t characters to translate into individual
* strings
* @param first An iterator referencing to the beginning of the target container (like a std::begin iterator)
* @param last An iterator referencing one-past-the-end of the target container (like a std::end iterator)
* @param cotaskmem_array The [out] wil::unique_cotaskmem_array_ptr<wil::unique_cotaskmem_string> to contain the array of
* strings. A wil::unique_cotaskmem_array_ptr<wil::unique_cotaskmem_string> of the extracted strings from the
* input container of wchar_t characters. An empty wil::unique_cotaskmem_array_ptr should be translated as out-of
* memory as there should always be at least one wil::unique_cotaskmem_string
*/
template <class InputIt>
void get_cotaskmemstring_array_from_multistring_nothrow(
const InputIt& first, const InputIt& last, ::wil::unique_cotaskmem_array_ptr<::wil::unique_cotaskmem_string>& cotaskmem_array) WI_NOEXCEPT
{
if (last - first < 3)
{
// it doesn't have the required 2 terminating null characters - return an empty string
*cotaskmem_array.addressof() = static_cast<PWSTR*>(::CoTaskMemAlloc(sizeof(PWSTR) * 1));
if (cotaskmem_array)
{
auto new_string = ::wil::make_cotaskmem_string_nothrow(L"");
if (new_string)
{
*cotaskmem_array.size_address() = 1;
cotaskmem_array[0] = new_string.release();
}
else
{
// oom will return an empty array
cotaskmem_array.reset();
}
}
else
{
// oom will return an empty array
cotaskmem_array.reset();
}
return;
}
// we must first count the # of strings for the array
size_t arraySize = 0;
walk_multistring(first, last, [&](const InputIt&, const InputIt&) {
++arraySize;
});
// allocate the array size necessary to hold all the unique_cotaskmem_strings
*cotaskmem_array.addressof() = static_cast<PWSTR*>(::CoTaskMemAlloc(sizeof(PWSTR) * arraySize));
if (!cotaskmem_array)
{
// oom will return an empty array
cotaskmem_array.reset();
return;
}
*cotaskmem_array.size_address() = arraySize;
ZeroMemory(cotaskmem_array.data(), sizeof(PWSTR) * arraySize);
size_t arrayOffset = 0;
walk_multistring(first, last, [&](const InputIt& string_first, const InputIt& string_last) {
FAIL_FAST_IF(arrayOffset >= arraySize);
const auto stringSize = string_last - string_first;
auto new_string = ::wil::make_cotaskmem_string_nothrow(&(*string_first), stringSize);
if (!new_string)
{
// oom will return an empty array
cotaskmem_array.reset();
return;
}
cotaskmem_array[arrayOffset] = new_string.release();
++arrayOffset;
});
}
#endif // #if defined(__WIL_OBJBASE_H_)
namespace reg_value_type_info
{
// supports_prepare_buffer is used to determine if the input buffer to read a registry value should be prepared
// before the first call to the registry read API
template <typename T>
constexpr bool supports_prepare_buffer() WI_NOEXCEPT
{
return false;
}
template <typename T>
HRESULT prepare_buffer(T&) WI_NOEXCEPT
{
// no-op in the default case
return S_OK;
}
// supports_resize_buffer_bytes is used to determine if the input buffer to read a registry value can be resized
// for those cases if the error from the registry read API indicates it needs a larger buffer
template <typename T>
constexpr bool supports_resize_buffer_bytes() WI_NOEXCEPT
{
return false;
}
template <typename T>
constexpr HRESULT resize_buffer_bytes(T&, DWORD) WI_NOEXCEPT
{
return E_NOTIMPL;
}
// supports_trim_buffer is used to determine if the input buffer to read a registry value must be trimmed
// after the registry read API has successfully written into the supplied buffer
// note that currently only std::wstring requires this as it cannot have embedded nulls
template <typename T>
constexpr bool supports_trim_buffer() WI_NOEXCEPT
{
return false;
}
// if called for a type that does not support trimming, will return a zero-length
template <typename T>
constexpr size_t trim_buffer(T&) WI_NOEXCEPT
{
return 0;
}
constexpr void* get_buffer(const int32_t& value) WI_NOEXCEPT
{
return const_cast<int32_t*>(&value);
}
constexpr DWORD get_buffer_size_bytes(int32_t) WI_NOEXCEPT
{
return static_cast<DWORD>(sizeof(int32_t));
}
constexpr void* get_buffer(const uint32_t& value) WI_NOEXCEPT
{
return const_cast<uint32_t*>(&value);
}
constexpr DWORD get_buffer_size_bytes(uint32_t) WI_NOEXCEPT
{
return static_cast<DWORD>(sizeof(uint32_t));
}
constexpr void* get_buffer(const long& value) WI_NOEXCEPT
{
return const_cast<long*>(&value);
}
constexpr DWORD get_buffer_size_bytes(long) WI_NOEXCEPT
{
return static_cast<DWORD>(sizeof(long));
}
constexpr void* get_buffer(const unsigned long& value) WI_NOEXCEPT
{
return const_cast<unsigned long*>(&value);
}
constexpr DWORD get_buffer_size_bytes(unsigned long) WI_NOEXCEPT
{
return static_cast<DWORD>(sizeof(unsigned long));
}
constexpr void* get_buffer(const int64_t& value) WI_NOEXCEPT
{
return const_cast<int64_t*>(&value);
}
constexpr DWORD get_buffer_size_bytes(int64_t) WI_NOEXCEPT
{
return static_cast<DWORD>(sizeof(int64_t));
}
constexpr void* get_buffer(const uint64_t& value) WI_NOEXCEPT
{
return const_cast<uint64_t*>(&value);
}
constexpr DWORD get_buffer_size_bytes(uint64_t) WI_NOEXCEPT
{
return static_cast<DWORD>(sizeof(uint64_t));
}
constexpr void* get_buffer(PCWSTR value) WI_NOEXCEPT
{
return const_cast<wchar_t*>(value);
}
inline DWORD get_buffer_size_bytes(PCWSTR value) WI_NOEXCEPT
{
if (!value)
{
return 0;
}
// including the last null buffer space in the returned buffer-size-bytes
// as the registry API we call guarantees null termination
return static_cast<DWORD>((::wcslen(value) + 1) * sizeof(wchar_t));
}
#if defined(_VECTOR_) && defined(WIL_ENABLE_EXCEPTIONS)
inline void* get_buffer(const ::std::vector<uint8_t>& buffer) WI_NOEXCEPT
{
return const_cast<uint8_t*>(buffer.data());
}
inline DWORD get_buffer_size_bytes(const ::std::vector<uint8_t>& value) WI_NOEXCEPT
{
return static_cast<DWORD>(value.size());
}
template <>
constexpr bool supports_prepare_buffer<::std::vector<uint8_t>>() WI_NOEXCEPT
{
return true;
}
inline HRESULT prepare_buffer(::std::vector<uint8_t>& value) WI_NOEXCEPT
try
{
// resize the initial vector to at least 1 byte
// this is needed so we can detect when the registry value exists
// but the value has zero-bytes
if (value.empty())
{
value.resize(1);
}
// zero out the buffer if pre-allocated
for (auto& string_char : value)
{
string_char = 0x00;
}
return S_OK;
}
CATCH_RETURN();
template <>
constexpr bool supports_resize_buffer_bytes<::std::vector<uint8_t>>() WI_NOEXCEPT
{
return true;
}
inline HRESULT resize_buffer_bytes(::std::vector<uint8_t>& buffer, DWORD byteSize) WI_NOEXCEPT
try
{
buffer.resize(byteSize);
return S_OK;
}
CATCH_RETURN();
// std::vector<wchar_t> does not implement resize_buffer_bytes
// because these support functions are only needed for set_value
// from the return of get_multistring_from_wstrings
inline void* get_buffer(const ::std::vector<wchar_t>& value) WI_NOEXCEPT
{
return const_cast<wchar_t*>(value.data());
}
inline DWORD get_buffer_size_bytes(const ::std::vector<wchar_t>& value) WI_NOEXCEPT
{
return static_cast<DWORD>(value.size()) * sizeof(wchar_t);
}
template <>
constexpr bool supports_prepare_buffer<::std::vector<wchar_t>>() WI_NOEXCEPT
{
return true;
}
inline HRESULT prepare_buffer(::std::vector<wchar_t>& value) WI_NOEXCEPT
{
// zero out the buffer if pre-allocated
for (auto& string_char : value)
{
string_char = L'\0';
}
return S_OK;
}
#endif // #if defined(_VECTOR_) && defined(WIL_ENABLE_EXCEPTIONS)
#if defined(_STRING_) && defined(WIL_ENABLE_EXCEPTIONS)
inline void* get_buffer(const ::std::wstring& string) WI_NOEXCEPT
{
return const_cast<wchar_t*>(string.data());
}
inline DWORD get_buffer_size_bytes(const ::std::wstring& string) WI_NOEXCEPT
{
// including the last null buffer space in the returned buffer-size-bytes
// as the registry API we call guarantees null termination
return static_cast<DWORD>((string.size() + 1) * sizeof(wchar_t));
}
template <>
constexpr bool supports_prepare_buffer<::std::wstring>() WI_NOEXCEPT
{
return true;
}
inline HRESULT prepare_buffer(::std::wstring& string) WI_NOEXCEPT
{
// zero out the buffer if pre-allocated
for (auto& string_char : string)
{
string_char = L'\0';
}
return S_OK;
}
template <>
constexpr bool supports_resize_buffer_bytes<::std::wstring>() WI_NOEXCEPT
{
return true;
}
inline HRESULT resize_buffer_bytes(::std::wstring& string, DWORD byteSize) WI_NOEXCEPT
try
{
string.resize(byteSize / sizeof(wchar_t));
return S_OK;
}
CATCH_RETURN();
template <>
constexpr bool supports_trim_buffer<::std::wstring>() WI_NOEXCEPT
{
return true;
}
inline size_t trim_buffer(::std::wstring& buffer) WI_NOEXCEPT
{
// remove any embedded null characters
const auto offset = buffer.find_first_of(L'\0');
if (offset != ::std::wstring::npos)
{
buffer.resize(offset);
}
return buffer.size();
}
#endif // #if defined(_STRING_) && defined(WIL_ENABLE_EXCEPTIONS)
#if defined(__WIL_OLEAUTO_H_)
inline void* get_buffer(const BSTR& value) WI_NOEXCEPT
{
return value;
}
inline DWORD get_buffer_size_bytes(const BSTR& value) WI_NOEXCEPT
{
auto length = ::SysStringLen(value);
if (length > 0)
{
// SysStringLen does not count the null-terminator
// including the last null buffer space in the returned buffer-size-bytes
// as the registry API we call guarantees null termination
length += 1;
}
return length * sizeof(WCHAR);
}
template <>
constexpr bool supports_prepare_buffer<BSTR>() WI_NOEXCEPT
{
return true;
}
inline HRESULT prepare_buffer(const BSTR& value) WI_NOEXCEPT
{
if (value)
{
// zero out the buffer if pre-allocated
for (auto& string_char : ::wil::make_range(value, get_buffer_size_bytes(value) / sizeof(WCHAR)))
{
string_char = L'\0';
}
}
return S_OK;
}
template <>
constexpr bool supports_resize_buffer_bytes<BSTR>() WI_NOEXCEPT
{
return true;
}
// transferringOwnership is only set to false if this is a 'shallow' copy of the BSTR
// and the caller maintained ownership of the original BSTR.
inline HRESULT resize_buffer_bytes(BSTR& string, DWORD byteSize, bool transferringOwnership = true) WI_NOEXCEPT
{
// copy the original BSTR to copy from later if needed
const BSTR original_string = string;
const DWORD original_string_length = string == nullptr ? 0 : ::SysStringLen(string);
// SysStringLen doesn't count the null-terminator, but our buffer size does
const bool original_string_length_too_small = (original_string_length + 1) < byteSize / sizeof(WCHAR);
if (original_string_length_too_small)
{
// pass a null BSTR value because SysAllocStringLen will copy the contents of the original BSTR,
// but in this case it's not long enough to copy the new length to be allocated
string = nullptr;
}
// convert bytes to length (number of WCHAR's)
DWORD length_to_alloc = byteSize / sizeof(WCHAR);
// SysAllocStringLen adds a null, so subtract a wchar_t from the input length
length_to_alloc = length_to_alloc > 0 ? length_to_alloc - 1 : length_to_alloc;
const BSTR new_bstr{::SysAllocStringLen(string, length_to_alloc)};
RETURN_IF_NULL_ALLOC(new_bstr);
// copy back the original BSTR if it was too small for SysAllocStringLen
// also assuming that both lengths are greater than zero
const DWORD sourceLengthToCopy = original_string_length < length_to_alloc ? original_string_length : length_to_alloc;
if (sourceLengthToCopy > 0 && original_string_length_too_small)
{
::memcpy_s(new_bstr, length_to_alloc * sizeof(WCHAR), original_string, sourceLengthToCopy * sizeof(WCHAR));
}
// if not transferring ownership, the caller will still own the original BSTR
if (transferringOwnership)
{
::SysFreeString(string);
}
string = new_bstr;
return S_OK;
}
inline void* get_buffer(const ::wil::unique_bstr& value) WI_NOEXCEPT
{
return value.get();
}
inline DWORD get_buffer_size_bytes(const ::wil::unique_bstr& value) WI_NOEXCEPT
{
return get_buffer_size_bytes(value.get());
}
template <>
constexpr bool supports_prepare_buffer<::wil::unique_bstr>() WI_NOEXCEPT
{
return true;
}
inline HRESULT prepare_buffer(const ::wil::unique_bstr& value) WI_NOEXCEPT
{
if (value)
{
// zero out the buffer if pre-allocated
for (auto& string_char : ::wil::make_range(value.get(), get_buffer_size_bytes(value) / sizeof(WCHAR)))
{
string_char = L'\0';
}
}
return S_OK;
}
template <>
constexpr bool supports_resize_buffer_bytes<::wil::unique_bstr>() WI_NOEXCEPT
{
return true;
}
inline HRESULT resize_buffer_bytes(::wil::unique_bstr& string, DWORD byteSize) WI_NOEXCEPT
{
BSTR temp_bstr = string.get();
// not transferring ownership of the BSTR within 'string' to resize_buffer_bytes()
// resize_buffer_bytes() will overwrite temp_bstr with a newly-allocated BSTR
constexpr bool transferringOwnership = false;
RETURN_IF_FAILED(resize_buffer_bytes(temp_bstr, byteSize, transferringOwnership));
// if succeeded in creating a new BSTR, move ownership of the new BSTR into string
string.reset(temp_bstr);
return S_OK;
}
#endif // #if defined(__WIL_OLEAUTO_H_)
#if defined(__WIL_OLEAUTO_H_STL)
inline void* get_buffer(const ::wil::shared_bstr& value) WI_NOEXCEPT
{
return value.get();
}
inline DWORD get_buffer_size_bytes(const ::wil::shared_bstr& value) WI_NOEXCEPT
{
return get_buffer_size_bytes(value.get());
}
template <>
constexpr bool supports_prepare_buffer<::wil::shared_bstr>() WI_NOEXCEPT
{
return true;
}
inline HRESULT prepare_buffer(const ::wil::shared_bstr& value) WI_NOEXCEPT
{
if (value)
{
// zero out the buffer if pre-allocated
for (auto& string_char : ::wil::make_range(value.get(), get_buffer_size_bytes(value) / sizeof(WCHAR)))
{
string_char = L'\0';
}
}
return S_OK;
}
template <>
constexpr bool supports_resize_buffer_bytes<::wil::shared_bstr>() WI_NOEXCEPT
{
return true;
}
inline HRESULT resize_buffer_bytes(::wil::shared_bstr& string, DWORD byteSize) WI_NOEXCEPT
{
BSTR temp_bstr = string.get();
// not transferring ownership of the BSTR within 'string' to resize_buffer_bytes()
// resize_buffer_bytes() will overwrite temp_bstr with a newly-allocated BSTR
constexpr bool transferringOwnership = false;
RETURN_IF_FAILED(resize_buffer_bytes(temp_bstr, byteSize, transferringOwnership));
// if succeeded in creating a new BSTR, move ownership of the new BSTR into string
string.reset(temp_bstr);
return S_OK;
}
#endif // #if defined(__WIL_OLEAUTO_H_STL)
#if defined(__WIL_OBJBASE_H_)
inline void* get_buffer(const ::wil::unique_cotaskmem_string& value) WI_NOEXCEPT
{
return value.get();
}
constexpr DWORD get_buffer_size_bytes(const ::wil::unique_cotaskmem_string&) WI_NOEXCEPT
{
// wil::unique_cotaskmem_string does not intrinsically track its internal buffer size
// thus the caller must track the buffer size it requested to be allocated
return 0;
}
template <>
constexpr bool supports_resize_buffer_bytes<::wil::unique_cotaskmem_string>() WI_NOEXCEPT
{
return true;
}
inline HRESULT resize_buffer_bytes(::wil::unique_cotaskmem_string& string, DWORD byteSize) WI_NOEXCEPT
{
// convert bytes to length (number of WCHAR's)
size_t length = byteSize / sizeof(wchar_t);
// ::wil::make_unique_string_nothrow adds one to the length when it allocates, so subtracting 1 from the input length
length = length > 0 ? length - 1 : length;
auto new_string = ::wil::make_unique_string_nothrow<::wil::unique_cotaskmem_string>(string.get(), length);
RETURN_IF_NULL_ALLOC(new_string.get());
string = ::wistd::move(new_string);
return S_OK;
}
inline void* get_buffer(const ::wil::unique_cotaskmem_array_ptr<uint8_t>& value) WI_NOEXCEPT
{
return value.get();
}
inline DWORD get_buffer_size_bytes(const ::wil::unique_cotaskmem_array_ptr<uint8_t>& value) WI_NOEXCEPT
{
return static_cast<DWORD>(value.size());
}
template <>
constexpr bool supports_resize_buffer_bytes<::wil::unique_cotaskmem_array_ptr<uint8_t>>() WI_NOEXCEPT
{
return true;
}
inline HRESULT resize_buffer_bytes(::wil::unique_cotaskmem_array_ptr<uint8_t>& arrayValue, DWORD byteSize) WI_NOEXCEPT
{
::wil::unique_cotaskmem_array_ptr<uint8_t> tempValue;
*tempValue.addressof() = static_cast<uint8_t*>(::CoTaskMemAlloc(byteSize));
RETURN_IF_NULL_ALLOC(tempValue.get());
*tempValue.size_address() = byteSize;
const auto bytesToCopy = arrayValue.size() < byteSize ? arrayValue.size() : byteSize;
CopyMemory(tempValue.get(), arrayValue.get(), bytesToCopy);
arrayValue = ::wistd::move(tempValue);
return S_OK;
}
#endif // #if defined(__WIL_OBJBASE_H_)
#if defined(__WIL_OBJBASE_H_STL)
inline void* get_buffer(const ::wil::shared_cotaskmem_string& value) WI_NOEXCEPT
{
return value.get();
}
constexpr DWORD get_buffer_size_bytes(const ::wil::shared_cotaskmem_string&) WI_NOEXCEPT
{
// wil::shared_cotaskmem_string does not intrinsically track its internal buffer size
// thus the caller must track the buffer size it requested to be allocated
return 0;
}
template <>
constexpr bool supports_resize_buffer_bytes<::wil::shared_cotaskmem_string>() WI_NOEXCEPT
{
return true;
}
inline HRESULT resize_buffer_bytes(::wil::shared_cotaskmem_string& string, DWORD byteSize) WI_NOEXCEPT
{
// convert bytes to length (number of WCHAR's)
size_t length = byteSize / sizeof(WCHAR);
// ::wil::make_unique_string_nothrow adds one to the length when it allocates, so subtracting 1 from the input length
length = length > 0 ? length - 1 : length;
auto new_string = ::wil::make_unique_string_nothrow<::wil::unique_cotaskmem_string>(string.get(), length);
RETURN_IF_NULL_ALLOC(new_string.get());
string = ::wistd::move(new_string);
return S_OK;
}
#endif // #if defined(__WIL_OBJBASE_H_STL)
inline void* get_buffer(const ::wil::unique_process_heap_string& value) WI_NOEXCEPT
{
return value.get();
}
constexpr DWORD get_buffer_size_bytes(const ::wil::unique_process_heap_string&) WI_NOEXCEPT
{
// wil::unique_process_heap_string does not intrinsically track its internal buffer size
// thus the caller must track the buffer size it requested to be allocated
return 0;
}
template <>
constexpr bool supports_resize_buffer_bytes<::wil::unique_process_heap_string>() WI_NOEXCEPT
{
return true;
}
inline HRESULT resize_buffer_bytes(::wil::unique_process_heap_string& string, DWORD byteSize) WI_NOEXCEPT
{
// convert bytes to length (number of WCHAR's)
size_t length = byteSize / sizeof(wchar_t);
// ::wil::make_unique_string_nothrow adds one to the length when it allocates, so subtracting 1 from the input length
length = length > 0 ? length - 1 : length;
auto new_string = ::wil::make_unique_string_nothrow<::wil::unique_process_heap_string>(string.get(), length);
RETURN_IF_NULL_ALLOC(new_string.get());
string = ::wistd::move(new_string);
return S_OK;
}
// constexpr expressions to determining the get* and set* registry value types
// for all supported types T to read/write values
template <typename T>
DWORD get_value_type() WI_NOEXCEPT
{
static_assert(sizeof(T) != sizeof(T), "Unsupported type for get_value_type");
}
template <typename T>
DWORD set_value_type() WI_NOEXCEPT
{
static_assert(sizeof(T) != sizeof(T), "Unsupported type for set_value_type");
}
template <>
constexpr DWORD get_value_type<int32_t>() WI_NOEXCEPT
{
return get_value_flags_from_value_type(REG_DWORD);
}
template <>
constexpr DWORD set_value_type<int32_t>() WI_NOEXCEPT
{
return REG_DWORD;
}
template <>
constexpr DWORD get_value_type<uint32_t>() WI_NOEXCEPT
{
return get_value_flags_from_value_type(REG_DWORD);
}
template <>
constexpr DWORD set_value_type<uint32_t>() WI_NOEXCEPT
{
return REG_DWORD;
}
template <>
constexpr DWORD get_value_type<long>() WI_NOEXCEPT
{
return get_value_flags_from_value_type(REG_DWORD);
}
template <>
constexpr DWORD set_value_type<long>() WI_NOEXCEPT
{
return REG_DWORD;
}
template <>
constexpr DWORD get_value_type<unsigned long>() WI_NOEXCEPT
{
return get_value_flags_from_value_type(REG_DWORD);
}
template <>
constexpr DWORD set_value_type<unsigned long>() WI_NOEXCEPT
{
return REG_DWORD;
}
template <>
constexpr DWORD get_value_type<int64_t>() WI_NOEXCEPT
{
return get_value_flags_from_value_type(REG_QWORD);
}
template <>
constexpr DWORD set_value_type<int64_t>() WI_NOEXCEPT
{
return REG_QWORD;
}
template <>
constexpr DWORD get_value_type<uint64_t>() WI_NOEXCEPT
{
return get_value_flags_from_value_type(REG_QWORD);
}
template <>
constexpr DWORD set_value_type<uint64_t>() WI_NOEXCEPT
{
return REG_QWORD;
}
template <>
constexpr DWORD get_value_type<PCWSTR>() WI_NOEXCEPT
{
return get_value_flags_from_value_type(REG_SZ);
}
template <>
constexpr DWORD set_value_type<PCWSTR>() WI_NOEXCEPT
{
return REG_SZ;
}
#if defined(_STRING_) && defined(WIL_ENABLE_EXCEPTIONS)
template <>
constexpr DWORD get_value_type<::std::wstring>() WI_NOEXCEPT
{
return get_value_flags_from_value_type(REG_SZ);
}
template <>
constexpr DWORD set_value_type<const ::std::wstring>() WI_NOEXCEPT
{
return REG_SZ;
}
#endif // #if defined(_STRING_) && defined(WIL_ENABLE_EXCEPTIONS)
#if defined(__WIL_OLEAUTO_H_)
template <>
constexpr DWORD get_value_type<BSTR>() WI_NOEXCEPT
{
return get_value_flags_from_value_type(REG_SZ);
}
template <>
constexpr DWORD get_value_type<::wil::unique_bstr>() WI_NOEXCEPT
{
return get_value_flags_from_value_type(REG_SZ);
}
template <>
constexpr DWORD set_value_type<const BSTR>() WI_NOEXCEPT
{
return REG_SZ;
}
template <>
constexpr DWORD set_value_type<const ::wil::unique_bstr>() WI_NOEXCEPT
{
return REG_SZ;
}
#endif // #if defined(__WIL_OLEAUTO_H_)
#if defined(__WIL_OLEAUTO_H_STL)
template <>
constexpr DWORD get_value_type<::wil::shared_bstr>() WI_NOEXCEPT
{
return get_value_flags_from_value_type(REG_SZ);
}
template <>
constexpr DWORD set_value_type<const ::wil::shared_bstr>() WI_NOEXCEPT
{
return REG_SZ;
}
#endif // #if defined(__WIL_OLEAUTO_H_STL)
#if defined(__WIL_OBJBASE_H_)
template <>
constexpr DWORD get_value_type<::wil::unique_cotaskmem_string>() WI_NOEXCEPT
{
return get_value_flags_from_value_type(REG_SZ);
}
template <>
constexpr DWORD set_value_type<const ::wil::unique_cotaskmem_string>() WI_NOEXCEPT
{
return REG_SZ;
}
#endif // defined(__WIL_OBJBASE_H_)
#if defined(__WIL_OBJBASE_H_STL)
template <>
constexpr DWORD get_value_type<::wil::shared_cotaskmem_string>() WI_NOEXCEPT
{
return get_value_flags_from_value_type(REG_SZ);
}
template <>
constexpr DWORD set_value_type<const ::wil::shared_cotaskmem_string>() WI_NOEXCEPT
{
return REG_SZ;
}
#endif // #if defined(__WIL_OBJBASE_H_STL)
} // namespace reg_value_type_info
template <typename err_policy = ::wil::err_exception_policy>
class reg_view_t
{
public:
explicit reg_view_t(HKEY key) WI_NOEXCEPT : m_key(key)
{
}
~reg_view_t() WI_NOEXCEPT = default;
reg_view_t(const reg_view_t&) = delete;
reg_view_t& operator=(const reg_view_t&) = delete;
reg_view_t(reg_view_t&&) = delete;
reg_view_t& operator=(reg_view_t&&) = delete;
typename err_policy::result open_key(
_In_opt_ _In_opt_ PCWSTR subKey, _Out_ HKEY* hkey, ::wil::reg::key_access access = ::wil::reg::key_access::read) const
{
constexpr DWORD zero_options{0};
return err_policy::HResult(
HRESULT_FROM_WIN32(::RegOpenKeyExW(m_key, subKey, zero_options, get_access_flags(access), hkey)));
}
typename err_policy::result create_key(PCWSTR subKey, _Out_ HKEY* hkey, ::wil::reg::key_access access = ::wil::reg::key_access::read) const
{
*hkey = nullptr;
constexpr DWORD zero_reserved{0};
constexpr PWSTR null_class{nullptr};
constexpr DWORD zero_options{0};
constexpr SECURITY_ATTRIBUTES* null_security_attributes{nullptr};
DWORD disposition{0};
return err_policy::HResult(HRESULT_FROM_WIN32(::RegCreateKeyExW(
m_key, subKey, zero_reserved, null_class, zero_options, get_access_flags(access), null_security_attributes, hkey, &disposition)));
}
typename err_policy::result delete_tree(_In_opt_ PCWSTR sub_key) const
{
auto hr = HRESULT_FROM_WIN32(::RegDeleteTreeW(m_key, sub_key));
if (::wil::reg::is_registry_not_found(hr))
{
hr = S_OK;
}
return err_policy::HResult(hr);
}
typename err_policy::result delete_value(_In_opt_ PCWSTR value_name) const
{
return err_policy::HResult(HRESULT_FROM_WIN32(::RegDeleteValueW(m_key, value_name)));
}
template <typename R>
typename err_policy::result get_value(
_In_opt_ PCWSTR subkey, _In_opt_ PCWSTR value_name, R& return_value, DWORD type = reg_value_type_info::get_value_type<R>()) const
{
return get_value_with_type(subkey, value_name, return_value, type);
}
// typename D supports unsigned 32-bit values; i.e. allows the caller to pass a DWORD* as well as uint32_t*
template <size_t Length, typename DwordType, wistd::enable_if_t<wistd::is_same_v<DwordType, uint32_t> || wistd::is_same_v<DwordType, unsigned long>>* = nullptr>
typename err_policy::result get_value_char_array(
_In_opt_ PCWSTR subkey, _In_opt_ PCWSTR value_name, WCHAR (&return_value)[Length], DWORD type, _Out_opt_ DwordType* requiredBytes) const
{
constexpr DwordType zero_value{0ul};
::wil::assign_to_opt_param(requiredBytes, zero_value);
DWORD data_size_bytes{Length * sizeof(WCHAR)};
const auto hr = HRESULT_FROM_WIN32(::RegGetValueW(
m_key, subkey, value_name, ::wil::reg::reg_view_details::get_value_flags_from_value_type(type), nullptr, return_value, &data_size_bytes));
if (SUCCEEDED(hr) || ::wil::reg::is_registry_buffer_too_small(hr))
{
const DwordType updated_value{data_size_bytes};
::wil::assign_to_opt_param(requiredBytes, updated_value);
}
return err_policy::HResult(hr);
}
#if defined(_OPTIONAL_) && defined(__cpp_lib_optional)
// intended for err_exception_policy as err_returncode_policy will not get an error code
template <typename R>
::std::optional<R> try_get_value(
_In_opt_ PCWSTR subkey, _In_opt_ PCWSTR value_name, DWORD type = reg_value_type_info::get_value_type<R>()) const
{
R value{};
const auto hr = get_value_with_type<R, ::wil::err_returncode_policy>(subkey, value_name, value, type);
if (SUCCEEDED(hr))
{
return ::std::optional(::wistd::move(value));
}
if (::wil::reg::is_registry_not_found(hr))
{
return ::std::nullopt;
}
// throw if exception policy
err_policy::HResult(hr);
return ::std::nullopt;
}
#endif // #if defined (_OPTIONAL_) && defined(__cpp_lib_optional)
template <typename R>
typename err_policy::result set_value(
_In_opt_ PCWSTR subkey, _In_opt_ PCWSTR value_name, const R& value, DWORD type = reg_value_type_info::set_value_type<R>()) const
{
return set_value_with_type(subkey, value_name, value, type);
}
private:
const HKEY m_key{};
template <typename R>
typename err_policy::result set_value_with_type(_In_opt_ PCWSTR subkey, _In_opt_ PCWSTR value_name, const R& value, DWORD type) const
{
return err_policy::HResult(HRESULT_FROM_WIN32(::RegSetKeyValueW(
m_key,
subkey,
value_name,
type,
static_cast<uint8_t*>(reg_value_type_info::get_buffer(value)),
reg_value_type_info::get_buffer_size_bytes(value))));
}
template <typename R, typename get_value_with_type_policy = err_policy>
typename get_value_with_type_policy::result get_value_with_type(
_In_opt_ PCWSTR subkey, _In_opt_ PCWSTR value_name, R& return_value, DWORD type = reg_value_type_info::get_value_type<R>()) const
{
if
#if defined(__cpp_if_constexpr)
constexpr
#endif
(reg_value_type_info::supports_prepare_buffer<R>())
{
const auto prepare_buffer_hr = reg_value_type_info::prepare_buffer(return_value);
if (FAILED(prepare_buffer_hr))
{
return get_value_with_type_policy::HResult(prepare_buffer_hr);
}
}
// get_buffer_size_bytes should include the null terminator when used for strings.
DWORD bytes_allocated{reg_value_type_info::get_buffer_size_bytes(return_value)};
HRESULT get_value_hresult = S_OK;
for (;;)
{
constexpr DWORD* null_type{nullptr};
DWORD data_size_bytes{bytes_allocated};
get_value_hresult = HRESULT_FROM_WIN32(::RegGetValueW(
m_key,
subkey,
value_name,
get_value_flags_from_value_type(type),
null_type,
reg_value_type_info::get_buffer(return_value),
&data_size_bytes));
// some return types we can grow as needed - e.g. when writing to a std::wstring
// only compile and resize_buffer for those types that support dynamically growing the buffer
if
#if defined(__cpp_if_constexpr)
constexpr
#endif
(reg_value_type_info::supports_resize_buffer_bytes<R>())
{
// Attempt to grow the buffer with the data_size_bytes returned from GetRegValueW
// GetRegValueW will indicate the caller allocate the returned number of bytes in one of two cases:
// 1. returns ERROR_MORE_DATA
// 2. returns ERROR_SUCCESS when we gave it a nullptr for the out buffer
const bool shouldReallocate =
(::wil::reg::is_registry_buffer_too_small(get_value_hresult)) ||
(SUCCEEDED(get_value_hresult) && (reg_value_type_info::get_buffer(return_value) == nullptr) &&
(data_size_bytes > 0));
if (shouldReallocate)
{
// verify if resize_buffer succeeded allocation
const auto resize_buffer_hr = reg_value_type_info::resize_buffer_bytes(return_value, data_size_bytes);
if (FAILED(resize_buffer_hr))
{
// if resize fails, return this error back to the caller
return get_value_with_type_policy::HResult(resize_buffer_hr);
}
// if it resize succeeds, continue the for loop to try again
bytes_allocated = data_size_bytes;
continue;
}
// if the RegGetValueW call succeeded with a non-null [out] param,
// and the type supports resize_buffer_bytes
// and the bytes we allocated don't match data_size_bytes returned from RegGetValueW
// resize the buffer to match what RegGetValueW returned
if (SUCCEEDED(get_value_hresult))
{
const auto current_byte_size = reg_value_type_info::get_buffer_size_bytes(return_value);
if (current_byte_size != data_size_bytes)
{
// verify if resize_buffer_bytes succeeded allocation
const auto resize_buffer_hr = reg_value_type_info::resize_buffer_bytes(return_value, data_size_bytes);
if (FAILED(resize_buffer_hr))
{
// if resize fails, return this error back to the caller
return get_value_with_type_policy::HResult(resize_buffer_hr);
}
}
}
}
// we don't need to reallocate and retry the call to RegGetValueW so breaking out of the loop
break;
}
// some types (generally string types) require trimming its internal buffer after RegGetValueW successfully wrote into its buffer
if
#if defined(__cpp_if_constexpr)
constexpr
#endif
(reg_value_type_info::supports_trim_buffer<R>())
{
if (SUCCEEDED(get_value_hresult))
{
reg_value_type_info::trim_buffer(return_value);
}
}
return get_value_with_type_policy::HResult(get_value_hresult);
}
};
using reg_view_nothrow = ::wil::reg::reg_view_details::reg_view_t<::wil::err_returncode_policy>;
#if defined(WIL_ENABLE_EXCEPTIONS)
using reg_view = ::wil::reg::reg_view_details::reg_view_t<::wil::err_exception_policy>;
#endif // #if defined(WIL_ENABLE_EXCEPTIONS)
} // namespace reg_view_details
/// @endcond
/// @cond
namespace reg_iterator_details
{
constexpr uint32_t iterator_end_offset = 0xffffffff;
constexpr size_t iterator_default_buffer_length = 32;
constexpr size_t iterator_max_keyname_length = 255;
constexpr size_t iterator_max_valuename_length = 16383;
// function overloads to allow *_enumerator objects to be constructed from all 3 types of HKEY representatives
inline HKEY get_hkey(HKEY h) WI_NOEXCEPT
{
return h;
}
inline HKEY get_hkey(const ::wil::unique_hkey& h) WI_NOEXCEPT
{
return h.get();
}
#if defined(__WIL_WINREG_STL)
inline HKEY get_hkey(const ::wil::shared_hkey& h) WI_NOEXCEPT
{
return h.get();
}
#endif // #if defined(__WIL_WINREG_STL)
#if defined(WIL_ENABLE_EXCEPTIONS) && defined(_STRING_)
// overloads for some of the below string functions - specific for std::wstring
// these overloads must be declared before the template functions below, as some of those template functions
// reference these overload functions
inline void clear_name(::std::wstring& name, size_t) WI_NOEXCEPT
{
name.assign(name.size(), L'\0');
}
inline ::std::wstring copy_name(const ::std::wstring& str, size_t length) WI_NOEXCEPT
{
try
{
// guarantee that the copied string has the specified internal length
// i.e., the same length assumptions hold when the string is copied
::std::wstring tempString(length, L'0');
tempString.assign(str);
return tempString;
}
catch (...)
{
return {};
}
}
inline bool can_derive_length(const ::std::wstring& name) WI_NOEXCEPT
{
return !name.empty();
}
#endif // #if defined(WIL_ENABLE_EXCEPTIONS) && defined(_STRING_)
// string manipulation functions needed for iterator functions
template <typename T>
PWSTR address_of_name(const T& name) WI_NOEXCEPT
{
return static_cast<PWSTR>(::wil::reg::reg_view_details::reg_value_type_info::get_buffer(name));
}
template <typename T>
bool can_derive_length(const T& name) WI_NOEXCEPT
{
return static_cast<bool>(address_of_name(name));
}
template <typename T>
bool compare_name(const T& name, PCWSTR comparand) WI_NOEXCEPT
{
if (!can_derive_length(name) || !comparand)
{
return false;
}
return 0 == wcscmp(address_of_name(name), comparand);
}
template <typename T>
void clear_name(const T& name, size_t length) WI_NOEXCEPT
{
if (can_derive_length(name) && length > 0)
{
memset(address_of_name(name), 0, length * sizeof(wchar_t));
}
}
// failure returns zero
template <typename T>
size_t resize_name_cch(T& name, size_t current_length, size_t new_length) WI_NOEXCEPT
{
if (new_length > current_length)
{
if (FAILED(::wil::reg::reg_view_details::reg_value_type_info::resize_buffer_bytes(
name, static_cast<DWORD>(new_length * sizeof(wchar_t)))))
{
return 0;
}
current_length = new_length;
// fall through to clear the newly allocated buffer
// and return the new length
}
// continue to use the existing buffer since the requested length is less than or equals to the current length
clear_name(name, current_length);
return current_length;
}
template <typename T>
T copy_name(const T& name, size_t length) WI_NOEXCEPT
{
if (!can_derive_length(name))
{
return {};
}
return ::wil::make_unique_string_nothrow<T>(address_of_name(name), length);
}
#if defined(__WIL_OLEAUTO_H_)
inline ::wil::unique_bstr copy_name(const ::wil::unique_bstr& name, size_t length) WI_NOEXCEPT
{
if (!can_derive_length(name))
{
return {};
}
// SysAllocStringLen adds a null, so subtract a wchar_t from the input length
length = length > 0 ? length - 1 : length;
return ::wil::unique_bstr{::SysAllocStringLen(name.get(), static_cast<UINT>(length))};
}
#endif // #if defined(__WIL_OLEAUTO_H_)
} // namespace reg_iterator_details
/// @endcond
// forward declaration to allow friend-ing the template iterator class
#if defined(WIL_ENABLE_EXCEPTIONS)
template <typename T>
class iterator_t;
#endif
template <typename T>
class iterator_nothrow_t;
// all methods must be noexcept - to be usable with any iterator type (throwing or non-throwing)
template <typename T>
class key_iterator_data
{
public:
T name{};
key_iterator_data(HKEY key = nullptr) WI_NOEXCEPT : m_hkey{key}
{
}
~key_iterator_data() WI_NOEXCEPT = default;
key_iterator_data(const key_iterator_data& rhs) WI_NOEXCEPT
{
// might return null/empty string on failure
name = ::wil::reg::reg_iterator_details::copy_name(rhs.name, rhs.m_name_length);
m_hkey = rhs.m_hkey;
m_index = rhs.m_index;
m_name_length = ::wil::reg::reg_iterator_details::can_derive_length(name) ? rhs.m_name_length : 0;
}
key_iterator_data& operator=(const key_iterator_data& rhs) WI_NOEXCEPT
{
if (&rhs != this)
{
key_iterator_data temp(rhs);
*this = ::wistd::move(temp);
}
return *this;
}
key_iterator_data(key_iterator_data&& rhs) WI_NOEXCEPT : name{wistd::move(rhs.name)},
m_hkey{wistd::move(rhs.m_hkey)},
m_index{wistd::move(rhs.m_index)},
m_name_length{wistd::move(rhs.m_name_length)}
{
// once name is moved, we must reset m_name_length as name is now empty
rhs.m_name_length = 0;
}
key_iterator_data& operator=(key_iterator_data&& rhs) WI_NOEXCEPT
{
if (&rhs != this)
{
name = ::wistd::move(rhs.name);
m_hkey = ::wistd::move(rhs.m_hkey);
m_index = ::wistd::move(rhs.m_index);
m_name_length = ::wistd::move(rhs.m_name_length);
// once name is moved, we must reset m_name_length as name is now empty
rhs.m_name_length = 0;
}
return *this;
}
// Case-sensitive comparison
bool operator==(PCWSTR comparand) const WI_NOEXCEPT
{
return ::wil::reg::reg_iterator_details::compare_name(name, comparand);
}
private:
#if defined(WIL_ENABLE_EXCEPTIONS)
friend class ::wil::reg::iterator_t<key_iterator_data>;
#endif
friend class ::wil::reg::iterator_nothrow_t<key_iterator_data>;
bool at_end() const WI_NOEXCEPT
{
return m_index == ::wil::reg::reg_iterator_details::iterator_end_offset;
}
void make_end_iterator() WI_NOEXCEPT
{
::wil::reg::reg_iterator_details::clear_name(name, m_name_length);
m_index = ::wil::reg::reg_iterator_details::iterator_end_offset;
}
bool resize(size_t new_length) WI_NOEXCEPT
{
// if resize fails, will return 0
m_name_length = ::wil::reg::reg_iterator_details::resize_name_cch(name, m_name_length, new_length);
return m_name_length > 0;
}
HRESULT enumerate_current_index() WI_NOEXCEPT
{
FAIL_FAST_IF(at_end());
auto string_length = static_cast<DWORD>(m_name_length) > 0
? static_cast<DWORD>(m_name_length)
: static_cast<DWORD>(::wil::reg::reg_iterator_details::iterator_default_buffer_length);
for (;;)
{
if (!resize(string_length))
{
RETURN_HR(E_OUTOFMEMORY);
}
const auto error = ::RegEnumKeyExW(
m_hkey, // hKey
m_index, // dwIndex
string_length == 0 ? nullptr : ::wil::reg::reg_iterator_details::address_of_name(name), // lpName
&string_length, // lpcchName
nullptr, // lpReserved
nullptr, // lpClass
nullptr, // lpcchClass
nullptr); // lpftLastWriteTime
if (error == ERROR_SUCCESS)
{
// string_length returned from RegEnumKeyExW does not include the null terminator
++string_length;
// if the string type supports resize, we must resize it to the returned string size
// to ensure continuity of the string type with the actual string size
if (::wil::reg::reg_view_details::reg_value_type_info::supports_resize_buffer_bytes<T>())
{
RETURN_IF_FAILED(::wil::reg::reg_view_details::reg_value_type_info::resize_buffer_bytes(
name, string_length * sizeof(WCHAR)));
m_name_length = string_length;
}
if (::wil::reg::reg_view_details::reg_value_type_info::supports_trim_buffer<T>())
{
m_name_length = ::wil::reg::reg_view_details::reg_value_type_info::trim_buffer(name);
}
break;
}
if (error == ERROR_NO_MORE_ITEMS)
{
make_end_iterator();
break;
}
if (error == ERROR_MORE_DATA)
{
// if our default wasn't big enough, resize to either half max or max length and try again
if (string_length < reg_iterator_details::iterator_max_keyname_length / 2)
{
string_length = reg_iterator_details::iterator_max_keyname_length / 2;
}
else if (string_length < reg_iterator_details::iterator_max_keyname_length + 1)
{
string_length = reg_iterator_details::iterator_max_keyname_length + 1; // plus one for null
}
else
{
// if we're already at max length, we can't grow anymore, so we'll just return the error
break;
}
continue;
}
// any other error will fail
RETURN_WIN32(error);
}
return S_OK;
}
HKEY m_hkey{};
uint32_t m_index = ::wil::reg::reg_iterator_details::iterator_end_offset;
size_t m_name_length{};
};
// all methods must be noexcept - to be usable with any iterator type (throwing or non-throwing)
template <typename T>
class value_iterator_data
{
public:
T name{};
DWORD type = REG_NONE;
value_iterator_data(HKEY key = nullptr) WI_NOEXCEPT : m_hkey{key}
{
}
~value_iterator_data() WI_NOEXCEPT = default;
value_iterator_data(const value_iterator_data& rhs) WI_NOEXCEPT
{
// might return null/empty string on failure
name = ::wil::reg::reg_iterator_details::copy_name(rhs.name, rhs.m_name_length);
type = rhs.type;
m_hkey = rhs.m_hkey;
m_index = rhs.m_index;
m_name_length = ::wil::reg::reg_iterator_details::can_derive_length(name) ? rhs.m_name_length : 0;
}
value_iterator_data& operator=(const value_iterator_data& rhs) WI_NOEXCEPT
{
if (&rhs != this)
{
value_iterator_data temp(rhs);
*this = ::wistd::move(temp);
}
return *this;
}
value_iterator_data(value_iterator_data&&) WI_NOEXCEPT = default;
value_iterator_data& operator=(value_iterator_data&& rhs) WI_NOEXCEPT = default;
bool at_end() const WI_NOEXCEPT
{
return m_index == ::wil::reg::reg_iterator_details::iterator_end_offset;
}
private:
#if defined(WIL_ENABLE_EXCEPTIONS)
friend class ::wil::reg::iterator_t<value_iterator_data>;
#endif
friend class ::wil::reg::iterator_nothrow_t<value_iterator_data>;
void make_end_iterator() WI_NOEXCEPT
{
::wil::reg::reg_iterator_details::clear_name(name, m_name_length);
m_index = ::wil::reg::reg_iterator_details::iterator_end_offset;
}
bool resize(size_t new_length)
{
// if resize fails, will return 0
m_name_length = ::wil::reg::reg_iterator_details::resize_name_cch(name, m_name_length, new_length);
return m_name_length > 0;
}
HRESULT enumerate_current_index() WI_NOEXCEPT
{
FAIL_FAST_IF(at_end());
auto string_length = static_cast<DWORD>(m_name_length) > 0
? static_cast<DWORD>(m_name_length)
: static_cast<DWORD>(::wil::reg::reg_iterator_details::iterator_default_buffer_length);
for (;;)
{
if (!resize(string_length))
{
RETURN_HR(E_OUTOFMEMORY);
}
const auto error = ::RegEnumValueW(
m_hkey, // hKey
m_index, // dwIndex
string_length == 0 ? nullptr : ::wil::reg::reg_iterator_details::address_of_name(name), // lpValueName
&string_length, // lpcchValueName
nullptr, // lpReserved
&type, // lpType
nullptr, // lpData
nullptr); // lpcbData
if (error == ERROR_SUCCESS)
{
// string_length returned from RegEnumValueW does not include the null terminator
++string_length;
// if the string type supports resize, we must resize it to the returned string size
// to ensure continuity of the string type with the actual string size
if (::wil::reg::reg_view_details::reg_value_type_info::supports_resize_buffer_bytes<T>())
{
RETURN_IF_FAILED(::wil::reg::reg_view_details::reg_value_type_info::resize_buffer_bytes(
name, string_length * sizeof(WCHAR)));
m_name_length = string_length;
}
if (::wil::reg::reg_view_details::reg_value_type_info::supports_trim_buffer<T>())
{
m_name_length = ::wil::reg::reg_view_details::reg_value_type_info::trim_buffer(name);
}
break;
}
if (error == ERROR_NO_MORE_ITEMS)
{
make_end_iterator();
break;
}
if (error == ERROR_MORE_DATA)
{
if (string_length == ::wil::reg::reg_iterator_details::iterator_max_valuename_length + 1)
{
// this is the max size, so we can't grow anymore, so we'll just return the error
break;
}
// resize and try again - growing exponentially up to the max
string_length *= 2;
if (string_length > ::wil::reg::reg_iterator_details::iterator_max_valuename_length + 1)
{
string_length = ::wil::reg::reg_iterator_details::iterator_max_valuename_length + 1;
}
continue;
}
// any other error will fail
RETURN_WIN32(error);
}
return S_OK;
}
HKEY m_hkey{};
uint32_t m_index = ::wil::reg::reg_iterator_details::iterator_end_offset;
size_t m_name_length{};
};
#if defined(WIL_ENABLE_EXCEPTIONS)
template <typename T>
class iterator_t
{
public:
// defining iterator_traits allows STL <algorithm> functions to be used with this iterator class.
// Notice this is a forward_iterator
// - does not support random-access (e.g. vector::iterator)
// - does not support bidirectional access (e.g. list::iterator)
#if defined(_ITERATOR_) || defined(WIL_DOXYGEN)
using iterator_category = ::std::forward_iterator_tag;
#endif
using value_type = T;
using difference_type = size_t;
using distance_type = size_t;
using pointer = T*;
using reference = T&;
iterator_t() WI_NOEXCEPT = default;
~iterator_t() WI_NOEXCEPT = default;
iterator_t(HKEY hkey) : m_data(hkey)
{
if (hkey != nullptr)
{
m_data.m_index = 0;
THROW_IF_FAILED(m_data.enumerate_current_index());
}
}
iterator_t(const iterator_t&) = default;
iterator_t& operator=(const iterator_t&) = default;
iterator_t(iterator_t&&) WI_NOEXCEPT = default;
iterator_t& operator=(iterator_t&&) WI_NOEXCEPT = default;
// operator support
const T& operator*() const
{
FAIL_FAST_IF(m_data.at_end());
return m_data;
}
const T& operator*()
{
FAIL_FAST_IF(m_data.at_end());
return m_data;
}
const T* operator->() const
{
FAIL_FAST_IF(m_data.at_end());
return &m_data;
}
const T* operator->()
{
FAIL_FAST_IF(m_data.at_end());
return &m_data;
}
bool operator==(const iterator_t& rhs) const WI_NOEXCEPT
{
if (m_data.at_end() || rhs.m_data.at_end())
{
// if either is not initialized (or end), both must not be initialized (or end) to be equal
return m_data.m_index == rhs.m_data.m_index;
}
return m_data.m_hkey == rhs.m_data.m_hkey && m_data.m_index == rhs.m_data.m_index;
}
bool operator!=(const iterator_t& rhs) const WI_NOEXCEPT
{
return !(*this == rhs);
}
// pre-increment
iterator_t& operator++()
{
this->operator+=(1);
return *this;
}
const iterator_t& operator++() const
{
this->operator+=(1);
return *this;
}
// increment by integer
iterator_t& operator+=(size_t offset)
{
uint32_t newIndex = m_data.m_index + static_cast<uint32_t>(offset);
if (newIndex < m_data.m_index)
{
// fail on integer overflow
THROW_HR(E_INVALIDARG);
}
if (newIndex == ::wil::reg::reg_iterator_details::iterator_end_offset)
{
// fail if this creates an end iterator
THROW_HR(E_INVALIDARG);
}
// iterate by the integer offset
for (size_t count = 0; count < offset; ++count)
{
++m_data.m_index;
THROW_IF_FAILED(m_data.enumerate_current_index());
}
return *this;
}
// not supporting post-increment - which would require copy-construction
iterator_t operator++(int) = delete;
private:
// container based on the class template type
T m_data{};
};
#endif
template <typename T>
class iterator_nothrow_t
{
public:
iterator_nothrow_t() WI_NOEXCEPT = default;
~iterator_nothrow_t() WI_NOEXCEPT = default;
iterator_nothrow_t(HKEY hkey) WI_NOEXCEPT : m_data(hkey)
{
if (hkey != nullptr)
{
m_data.m_index = 0;
m_last_error = m_data.enumerate_current_index();
}
}
iterator_nothrow_t(const iterator_nothrow_t&) WI_NOEXCEPT = default;
iterator_nothrow_t& operator=(const iterator_nothrow_t&) WI_NOEXCEPT = default;
iterator_nothrow_t(iterator_nothrow_t&&) WI_NOEXCEPT = default;
iterator_nothrow_t& operator=(iterator_nothrow_t&&) WI_NOEXCEPT = default;
bool at_end() const WI_NOEXCEPT
{
return m_data.at_end();
}
HRESULT last_error() const WI_NOEXCEPT
{
return m_last_error;
}
HRESULT move_next() WI_NOEXCEPT
{
const auto newIndex = m_data.m_index + 1;
if (newIndex < m_data.m_index)
{
// fail on integer overflow
m_last_error = E_INVALIDARG;
}
else if (newIndex == ::wil::reg::reg_iterator_details::iterator_end_offset)
{
// fail if this creates an end iterator
m_last_error = E_INVALIDARG;
}
else
{
m_data.m_index = newIndex;
m_last_error = m_data.enumerate_current_index();
}
if (FAILED(m_last_error))
{
// on failure, set the iterator to an end iterator
m_data.make_end_iterator();
}
return m_last_error;
}
// operator support
const T& operator*() const WI_NOEXCEPT
{
return m_data;
}
const T& operator*() WI_NOEXCEPT
{
return m_data;
}
const T* operator->() const WI_NOEXCEPT
{
return &m_data;
}
const T* operator->() WI_NOEXCEPT
{
return &m_data;
}
bool operator==(const iterator_nothrow_t& rhs) const WI_NOEXCEPT
{
if (m_data.at_end() || rhs.m_data.at_end())
{
// if either is not initialized (or end), both must not be initialized (or end) to be equal
return m_data.m_index == rhs.m_data.m_index;
}
return m_data.m_hkey == rhs.m_data.m_hkey && m_data.m_index == rhs.m_data.m_index;
}
bool operator!=(const iterator_nothrow_t& rhs) const WI_NOEXCEPT
{
return !(*this == rhs);
}
iterator_nothrow_t& operator++() WI_NOEXCEPT
{
move_next();
return *this;
}
const iterator_nothrow_t& operator++() const WI_NOEXCEPT
{
move_next();
return *this;
}
private:
// container based on the class template type
T m_data{};
HRESULT m_last_error{};
};
} // namespace reg
} // namespace wil
#endif // __WIL_REGISTRY_HELPERS_INCLUDED