On Windows, things work a bit differently. While UNIX models almost everything as “files” you interact with, Windows uses other abstractions. On Windows, you get a handle that represents some object you can interact with in specific ways depending on exactly what kind of handle you have.
We will use the same main function as before, but we need to link to different functions in the Windows API and make changes to our syscall function.
ch03/b-normal-syscall
#[link(name = “kernel32”)]
extern “system” {
fn GetStdHandle(nStdHandle: i32) -> i32;
fn WriteConsoleW(
hConsoleOutput: i32,
lpBuffer: *const u16,
numberOfCharsToWrite: u32,
lpNumberOfCharsWritten: *mut u32,
lpReserved: *const std::ffi::c_void,
) -> i32;
}
The first thing you notice is that we no longer link to the “C” library. Instead, we link to the kernel32 library. The next change is the use of the system calling convention. This calling convention is a bit peculiar. You see, Windows uses different calling conventions depending on whether you write for a 32-bit x86 Windows version or a 64-bit x86_64 Windows version. Newer Windows versions running on x86_64 use the “C” calling convention, so if you have a newer system you can try changing that out and see that it still works. “Specifying system” lets the compiler figure out the right one to use based on the system.
We link to two different syscalls in Windows:
- GetStdHandle: This retrieves a reference to a standard device like stdout
- WriteConsoleW: WriteConsole comes in two types. WriteConsoleW takes Unicode text and WriteConsoleA takes ANSI-encoded text. We’re using the one that takes Unicode text in our program.
Now, ANSI-encoded text works fine if you only write English text, but as soon as you write text in other languages, you might need to use special characters that are not possible to represent in ANSI but possible in Unicode. If you mix them up, your program will not work as you expect.
Next is our new syscall function:
ch03/b-normal-syscall
fn syscall(message: String) -> io::Result<()> {
let msg: Vec<u16> = message.encode_utf16().collect();
let msg_ptr = msg.as_ptr();
let len = msg.len() as u32;
let mut output: u32 = 0;
let handle = unsafe { GetStdHandle(-11) };
if handle == -1 {
return Err(io::Error::last_os_error())
}
let res = unsafe {
WriteConsoleW(
handle,
msg_ptr,
len,
&mut output,
std::ptr::null()
)};
if res == 0 {
return Err(io::Error::last_os_error());
}
Ok(())
}
The first thing we do is convert the text to utf-16-encoded text, which Windows uses. Fortunately, Rust has a built-in function to convert our utf-8-encoded text to utf-16 code points. encode_utf16 returns an iterator over u16 code points that we can collect to a Vec.
The next two lines should be familiar by now. We get the pointer to where the text is stored and the length of the text in bytes.
The next thing we do is call GetStdHandle and pass in the value –11. The values we need to pass in for the different standard devices are described together with the GetStdHandle documentation at https://learn.microsoft.com/en-us/windows/console/getstdhandle. This is convenient, as we don’t have to dig through C header files to find all the constant values we need.
The return code to expect is also documented thoroughly for all functions, so we handle potential errors here in the same way as we did for the Linux/macOS syscalls.
Finally, we have the call to the WriteConsoleW function. There is nothing too fancy about this, and you’ll notice similarities with the write syscall we used for Linux. One difference is that the output is not returned from the function but written to an address location we pass in in the form of a pointer to our output variable.
Note
Now that you’ve seen how we create cross-platform syscalls, you will probably also understand why we’re not including the code to make every example in this book cross-platform. It’s simply the case that the book would be extremely long if we did, and it’s not apparent that all that extra information will actually benefit our understanding of the key concepts.
Leave a Reply