Qual é a maneira correta de alocar dados para passar para uma chamada FFI?

Depois dediscutindo / aprendendo sobre a maneira correta de chamar uma FFI da API do Windows da Rust, Brinquei um pouco mais e gostaria de verificar novamente meu entendimento.

Eu tenho uma API do Windows chamada duas vezes. Na primeira chamada, ele retorna o tamanho do buffer necessário para seu parâmetro de saída real. Em seguida, é chamado uma segunda vez com um buffer de tamanho suficiente. Atualmente, estou usando umVec como um tipo de dados para esse buffer (veja o exemplo abaixo).

O código funciona, mas estou me perguntando se essa é a maneira correta de fazer isso ou se seria melhor utilizar uma função comoalloc::heap::allocate reservar diretamente alguma memória e depois usartransmute para converter o resultado do FFI de volta. Mais uma vez, meu código funciona, mas estou tentando parecer um pouco nos bastidores.

extern crate advapi32;
extern crate winapi;
extern crate widestring;
use widestring::WideCString;
use std::io::Error as IOError;
use winapi::winnt;

fn main() {
    let mut lp_buffer: Vec<winnt::WCHAR> = Vec::new();
    let mut pcb_buffer: winapi::DWORD = 0;

    let rtrn_bool = unsafe {
        advapi32::GetUserNameW(lp_buffer.as_mut_ptr(),
                               &mut pcb_buffer )
    };

    if rtrn_bool == 0 {

        match IOError::last_os_error().raw_os_error() {
            Some(122) => {
                // Resizing the buffers sizes so that the data fits in after 2nd 
                lp_buffer.resize(pcb_buffer as usize, 0 as winnt::WCHAR);
            } // This error is to be expected
            Some(e) => panic!("Unknown OS error {}", e),
            None => panic!("That should not happen"),
        }
    }


    let rtrn_bool2 = unsafe {
        advapi32::GetUserNameW(lp_buffer.as_mut_ptr(), 
                               &mut pcb_buffer )
    };

    if rtrn_bool2 == 0 {
        match IOError::last_os_error().raw_os_error() {
            Some(e) => panic!("Unknown OS error {}", e),
            None => panic!("That should not happen"),
        }
    }

    let widestr: WideCString = unsafe { WideCString::from_ptr_str(lp_buffer.as_ptr()) };

    println!("The owner of the file is {:?}", widestr.to_string_lossy());
}

Dependências:

[dependencies]
advapi32-sys = "0.2"
winapi = "0.2"
widestring = "*"

questionAnswers(2)

yourAnswerToTheQuestion