Skip to content

Commit

Permalink
Add flexible registry type and byte query support (#3120)
Browse files Browse the repository at this point in the history
  • Loading branch information
kennykerr authored Jun 23, 2024
1 parent 2ec4e06 commit 863a7cf
Show file tree
Hide file tree
Showing 4 changed files with 117 additions and 5 deletions.
62 changes: 57 additions & 5 deletions crates/libs/registry/src/key.rs
Original file line number Diff line number Diff line change
Expand Up @@ -115,6 +115,34 @@ impl Key {
unsafe { self.set_value(name, REG_BINARY, value.as_ptr() as _, value.len()) }
}

/// Gets the type for the name in the registry key.
pub fn get_type<T: AsRef<str>>(&self, name: T) -> Result<Type> {
let name = pcwstr(name);
let mut ty = 0;

let result = unsafe {
RegQueryValueExW(
self.0,
name.as_ptr(),
null(),
&mut ty,
null_mut(),
null_mut(),
)
};

win32_error(result)?;

Ok(match ty {
REG_DWORD => Type::U32,
REG_QWORD => Type::U64,
REG_BINARY => Type::Bytes,
REG_SZ | REG_EXPAND_SZ => Type::String,
REG_MULTI_SZ => Type::MultiString,
rest => Type::Unknown(rest),
})
}

/// Gets the value for the name in the registry key.
pub fn get_value<T: AsRef<str>>(&self, name: T) -> Result<Value> {
let name = pcwstr(name);
Expand Down Expand Up @@ -252,11 +280,35 @@ impl Key {

/// Gets the value for the name in the registry key.
pub fn get_bytes<T: AsRef<str>>(&self, name: T) -> Result<Vec<u8>> {
if let Value::Bytes(value) = self.get_value(name)? {
Ok(value)
} else {
Err(invalid_data())
}
let name = pcwstr(name);
let mut len = 0;

let result = unsafe {
RegQueryValueExW(
self.0,
name.as_ptr(),
null(),
null_mut(),
null_mut(),
&mut len,
)
};

win32_error(result)?;
let mut value = vec![0u8; len as usize];

let result = unsafe {
RegQueryValueExW(
self.0,
name.as_ptr(),
null(),
null_mut(),
value.as_mut_ptr() as _,
&mut len,
)
};

win32_error(result).map(|_| value)
}

/// Gets the value for the name in the registry key.
Expand Down
3 changes: 3 additions & 0 deletions crates/libs/registry/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,9 @@ pub use key_iterator::KeyIterator;
mod value_iterator;
pub use value_iterator::ValueIterator;

mod r#type;
pub use r#type::Type;

pub use windows_result::Result;
use windows_result::*;

Expand Down
21 changes: 21 additions & 0 deletions crates/libs/registry/src/type.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
/// The possible types that a registry value could have.
#[derive(Clone, PartialEq, Eq, Debug)]
pub enum Type {
/// A 32-bit unsigned integer value.
U32,

/// A 64-bit unsigned integer value.
U64,

/// A string value.
String,

/// An array u8 bytes.
Bytes,

/// An array of string values.
MultiString,

/// An unknown or unsupported type.
Unknown(u32),
}
36 changes: 36 additions & 0 deletions crates/tests/registry/tests/bad_string.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
use windows::{core::w, Win32::System::Registry::*};
use windows_registry::*;

#[test]
fn bad_string() -> Result<()> {
let bad_string_bytes = vec![
0x00, 0xD8, // leading surrogate
0x01, 0x01, // bogus trailing surrogate
0x00, 0x00, // null
];

let test_key = "software\\windows-rs\\tests\\bad_string";
_ = CURRENT_USER.remove_tree(test_key);
let key = CURRENT_USER.create(test_key)?;

unsafe {
RegSetValueExW(
HKEY(key.as_raw()),
w!("name"),
0,
REG_SZ,
Some(&bad_string_bytes),
)
.ok()?
};

let ty = key.get_type("name")?;
assert_eq!(ty, Type::String);

let value_as_string = key.get_string("name")?;
assert_eq!(value_as_string, "�ā");

let value_as_bytes = key.get_bytes("name")?;
assert_eq!(value_as_bytes, bad_string_bytes);
Ok(())
}

0 comments on commit 863a7cf

Please sign in to comment.