std: Remove the as_utf16_p functions

These functions are all much better expressed via RAII using the to_utf16()
method on strings. This refactoring also takes this opportunity to properly
handle when filenames aren't valid unicode when passed through to the windows
I/O layer by properly returning I/O errors.

All previous users of the `as_utf16_p` or `as_utf16_mut_p` functions will need
to convert their code to using `foo.to_utf16().append_one(0)` to get a
null-terminated utf16 string.

[breaking-change]
This commit is contained in:
Alex Crichton 2014-06-02 14:51:58 -07:00
parent 0ea7aa30cc
commit d42cc130f9
7 changed files with 178 additions and 189 deletions

View File

@ -69,7 +69,6 @@ extern "system" {
pub mod compat { pub mod compat {
use std::intrinsics::{atomic_store_relaxed, transmute}; use std::intrinsics::{atomic_store_relaxed, transmute};
use libc::types::os::arch::extra::{LPCWSTR, HMODULE, LPCSTR, LPVOID}; use libc::types::os::arch::extra::{LPCWSTR, HMODULE, LPCSTR, LPVOID};
use std::os::win32::as_utf16_p;
extern "system" { extern "system" {
fn GetModuleHandleW(lpModuleName: LPCWSTR) -> HMODULE; fn GetModuleHandleW(lpModuleName: LPCWSTR) -> HMODULE;
@ -80,12 +79,11 @@ pub mod compat {
// This way, calling a function in this compatibility layer (after it's loaded) shouldn't // This way, calling a function in this compatibility layer (after it's loaded) shouldn't
// be any slower than a regular DLL call. // be any slower than a regular DLL call.
unsafe fn store_func<T: Copy>(ptr: *mut T, module: &str, symbol: &str, fallback: T) { unsafe fn store_func<T: Copy>(ptr: *mut T, module: &str, symbol: &str, fallback: T) {
as_utf16_p(module, |module| { let module = module.to_utf16().append_one(0);
symbol.with_c_str(|symbol| { symbol.with_c_str(|symbol| {
let handle = GetModuleHandleW(module); let handle = GetModuleHandleW(module.as_ptr());
let func: Option<T> = transmute(GetProcAddress(handle, symbol)); let func: Option<T> = transmute(GetProcAddress(handle, symbol));
atomic_store_relaxed(ptr, func.unwrap_or(fallback)) atomic_store_relaxed(ptr, func.unwrap_or(fallback))
})
}) })
} }

View File

@ -15,7 +15,7 @@ use libc::{c_int, c_void};
use libc; use libc;
use std::c_str::CString; use std::c_str::CString;
use std::mem; use std::mem;
use std::os::win32::{as_utf16_p, fill_utf16_buf_and_decode}; use std::os::win32::fill_utf16_buf_and_decode;
use std::ptr; use std::ptr;
use std::rt::rtio; use std::rt::rtio;
use std::rt::rtio::IoResult; use std::rt::rtio::IoResult;
@ -253,6 +253,17 @@ impl Drop for Inner {
} }
} }
pub fn to_utf16(s: &CString) -> IoResult<Vec<u16>> {
match s.as_str() {
Some(s) => Ok(s.to_utf16().append_one(0)),
None => Err(IoError {
code: libc::ERROR_INVALID_NAME as uint,
extra: 0,
detail: Some("valid unicode input required".to_str()),
})
}
}
pub fn open(path: &CString, fm: rtio::FileMode, fa: rtio::FileAccess) pub fn open(path: &CString, fm: rtio::FileMode, fa: rtio::FileAccess)
-> IoResult<FileDesc> { -> IoResult<FileDesc> {
// Flags passed to open_osfhandle // Flags passed to open_osfhandle
@ -299,15 +310,16 @@ pub fn open(path: &CString, fm: rtio::FileMode, fa: rtio::FileAccess)
// Compat with unix, this allows opening directories (see libuv) // Compat with unix, this allows opening directories (see libuv)
dwFlagsAndAttributes |= libc::FILE_FLAG_BACKUP_SEMANTICS; dwFlagsAndAttributes |= libc::FILE_FLAG_BACKUP_SEMANTICS;
let handle = as_utf16_p(path.as_str().unwrap(), |buf| unsafe { let path = try!(to_utf16(path));
libc::CreateFileW(buf, let handle = unsafe {
libc::CreateFileW(path.as_ptr(),
dwDesiredAccess, dwDesiredAccess,
dwShareMode, dwShareMode,
ptr::mut_null(), ptr::mut_null(),
dwCreationDisposition, dwCreationDisposition,
dwFlagsAndAttributes, dwFlagsAndAttributes,
ptr::mut_null()) ptr::mut_null())
}); };
if handle == libc::INVALID_HANDLE_VALUE as libc::HANDLE { if handle == libc::INVALID_HANDLE_VALUE as libc::HANDLE {
Err(super::last_error()) Err(super::last_error())
} else { } else {
@ -324,11 +336,10 @@ pub fn open(path: &CString, fm: rtio::FileMode, fa: rtio::FileAccess)
} }
pub fn mkdir(p: &CString, _mode: uint) -> IoResult<()> { pub fn mkdir(p: &CString, _mode: uint) -> IoResult<()> {
let p = try!(to_utf16(p));
super::mkerr_winbool(unsafe { super::mkerr_winbool(unsafe {
// FIXME: turn mode into something useful? #2623 // FIXME: turn mode into something useful? #2623
as_utf16_p(p.as_str().unwrap(), |buf| { libc::CreateDirectoryW(p.as_ptr(), ptr::mut_null())
libc::CreateDirectoryW(buf, ptr::mut_null())
})
}) })
} }
@ -351,9 +362,11 @@ pub fn readdir(p: &CString) -> IoResult<Vec<CString>> {
let star = Path::new(unsafe { let star = Path::new(unsafe {
CString::new(p.with_ref(|p| p), false) CString::new(p.with_ref(|p| p), false)
}).join("*"); }).join("*");
as_utf16_p(star.as_str().unwrap(), |path_ptr| unsafe { let path = try!(to_utf16(&star.to_c_str()));
unsafe {
let wfd_ptr = malloc_raw(rust_list_dir_wfd_size() as uint); let wfd_ptr = malloc_raw(rust_list_dir_wfd_size() as uint);
let find_handle = libc::FindFirstFileW(path_ptr, wfd_ptr as libc::HANDLE); let find_handle = libc::FindFirstFileW(path.as_ptr(), wfd_ptr as libc::HANDLE);
if find_handle as libc::c_int != libc::INVALID_HANDLE_VALUE { if find_handle as libc::c_int != libc::INVALID_HANDLE_VALUE {
let mut paths = vec!(); let mut paths = vec!();
let mut more_files = 1 as libc::c_int; let mut more_files = 1 as libc::c_int;
@ -377,37 +390,35 @@ pub fn readdir(p: &CString) -> IoResult<Vec<CString>> {
} else { } else {
Err(super::last_error()) Err(super::last_error())
} }
}) }
} }
pub fn unlink(p: &CString) -> IoResult<()> { pub fn unlink(p: &CString) -> IoResult<()> {
let p = try!(to_utf16(p));
super::mkerr_winbool(unsafe { super::mkerr_winbool(unsafe {
as_utf16_p(p.as_str().unwrap(), |buf| { libc::DeleteFileW(p.as_ptr())
libc::DeleteFileW(buf)
})
}) })
} }
pub fn rename(old: &CString, new: &CString) -> IoResult<()> { pub fn rename(old: &CString, new: &CString) -> IoResult<()> {
let old = try!(to_utf16(old));
let new = try!(to_utf16(new));
super::mkerr_winbool(unsafe { super::mkerr_winbool(unsafe {
as_utf16_p(old.as_str().unwrap(), |old| { libc::MoveFileExW(old.as_ptr(), new.as_ptr(),
as_utf16_p(new.as_str().unwrap(), |new| { libc::MOVEFILE_REPLACE_EXISTING)
libc::MoveFileExW(old, new, libc::MOVEFILE_REPLACE_EXISTING)
})
})
}) })
} }
pub fn chmod(p: &CString, mode: uint) -> IoResult<()> { pub fn chmod(p: &CString, mode: uint) -> IoResult<()> {
super::mkerr_libc(as_utf16_p(p.as_str().unwrap(), |p| unsafe { let p = try!(to_utf16(p));
libc::wchmod(p, mode as libc::c_int) super::mkerr_libc(unsafe {
})) libc::wchmod(p.as_ptr(), mode.bits() as libc::c_int)
})
} }
pub fn rmdir(p: &CString) -> IoResult<()> { pub fn rmdir(p: &CString) -> IoResult<()> {
super::mkerr_libc(as_utf16_p(p.as_str().unwrap(), |p| unsafe { let p = try!(to_utf16(p));
libc::wrmdir(p) super::mkerr_libc(unsafe { libc::wrmdir(p.as_ptr()) })
}))
} }
pub fn chown(_p: &CString, _uid: int, _gid: int) -> IoResult<()> { pub fn chown(_p: &CString, _uid: int, _gid: int) -> IoResult<()> {
@ -418,16 +429,15 @@ pub fn chown(_p: &CString, _uid: int, _gid: int) -> IoResult<()> {
pub fn readlink(p: &CString) -> IoResult<CString> { pub fn readlink(p: &CString) -> IoResult<CString> {
// FIXME: I have a feeling that this reads intermediate symlinks as well. // FIXME: I have a feeling that this reads intermediate symlinks as well.
use io::c::compat::kernel32::GetFinalPathNameByHandleW; use io::c::compat::kernel32::GetFinalPathNameByHandleW;
let p = try!(to_utf16(p));
let handle = unsafe { let handle = unsafe {
as_utf16_p(p.as_str().unwrap(), |p| { libc::CreateFileW(p.as_ptr(),
libc::CreateFileW(p, libc::GENERIC_READ,
libc::GENERIC_READ, libc::FILE_SHARE_READ,
libc::FILE_SHARE_READ, ptr::mut_null(),
ptr::mut_null(), libc::OPEN_EXISTING,
libc::OPEN_EXISTING, libc::FILE_ATTRIBUTE_NORMAL,
libc::FILE_ATTRIBUTE_NORMAL, ptr::mut_null())
ptr::mut_null())
})
}; };
if handle as int == libc::INVALID_HANDLE_VALUE as int { if handle as int == libc::INVALID_HANDLE_VALUE as int {
return Err(super::last_error()) return Err(super::last_error())
@ -453,19 +463,19 @@ pub fn readlink(p: &CString) -> IoResult<CString> {
pub fn symlink(src: &CString, dst: &CString) -> IoResult<()> { pub fn symlink(src: &CString, dst: &CString) -> IoResult<()> {
use io::c::compat::kernel32::CreateSymbolicLinkW; use io::c::compat::kernel32::CreateSymbolicLinkW;
super::mkerr_winbool(as_utf16_p(src.as_str().unwrap(), |src| { let src = try!(to_utf16(src));
as_utf16_p(dst.as_str().unwrap(), |dst| { let dst = try!(to_utf16(dst));
unsafe { CreateSymbolicLinkW(dst, src, 0) } super::mkerr_winbool(unsafe {
}) as libc::BOOL CreateSymbolicLinkW(dst.as_ptr(), src.as_ptr(), 0) as libc::BOOL
})) })
} }
pub fn link(src: &CString, dst: &CString) -> IoResult<()> { pub fn link(src: &CString, dst: &CString) -> IoResult<()> {
super::mkerr_winbool(as_utf16_p(src.as_str().unwrap(), |src| { let src = try!(to_utf16(src));
as_utf16_p(dst.as_str().unwrap(), |dst| { let dst = try!(to_utf16(dst));
unsafe { libc::CreateHardLinkW(dst, src, ptr::mut_null()) } super::mkerr_winbool(unsafe {
}) libc::CreateHardLinkW(dst.as_ptr(), src.as_ptr(), ptr::mut_null())
})) })
} }
fn mkstat(stat: &libc::stat) -> rtio::FileStat { fn mkstat(stat: &libc::stat) -> rtio::FileStat {
@ -491,12 +501,11 @@ fn mkstat(stat: &libc::stat) -> rtio::FileStat {
pub fn stat(p: &CString) -> IoResult<rtio::FileStat> { pub fn stat(p: &CString) -> IoResult<rtio::FileStat> {
let mut stat: libc::stat = unsafe { mem::zeroed() }; let mut stat: libc::stat = unsafe { mem::zeroed() };
as_utf16_p(p.as_str().unwrap(), |up| { let p = try!(to_utf16(p));
match unsafe { libc::wstat(up, &mut stat) } { match unsafe { libc::wstat(p.as_ptr(), &mut stat) } {
0 => Ok(mkstat(&stat)), 0 => Ok(mkstat(&stat)),
_ => Err(super::last_error()), _ => Err(super::last_error()),
} }
})
} }
pub fn lstat(_p: &CString) -> IoResult<rtio::FileStat> { pub fn lstat(_p: &CString) -> IoResult<rtio::FileStat> {
@ -509,7 +518,8 @@ pub fn utime(p: &CString, atime: u64, mtime: u64) -> IoResult<()> {
actime: (atime / 1000) as libc::time64_t, actime: (atime / 1000) as libc::time64_t,
modtime: (mtime / 1000) as libc::time64_t, modtime: (mtime / 1000) as libc::time64_t,
}; };
super::mkerr_libc(as_utf16_p(p.as_str().unwrap(), |p| unsafe { let p = try!(to_utf16(p));
libc::wutime(p, &buf) super::mkerr_libc(unsafe {
})) libc::wutime(p.as_ptr(), &buf)
})
} }

View File

@ -88,7 +88,6 @@ use alloc::arc::Arc;
use libc; use libc;
use std::c_str::CString; use std::c_str::CString;
use std::mem; use std::mem;
use std::os::win32::as_utf16_p;
use std::os; use std::os;
use std::ptr; use std::ptr;
use std::rt::rtio; use std::rt::rtio;
@ -98,6 +97,7 @@ use std::rt::mutex;
use super::c; use super::c;
use super::util; use super::util;
use super::file::to_utf16;
struct Event(libc::HANDLE); struct Event(libc::HANDLE);
@ -261,67 +261,66 @@ impl UnixStream {
} }
pub fn connect(addr: &CString, timeout: Option<u64>) -> IoResult<UnixStream> { pub fn connect(addr: &CString, timeout: Option<u64>) -> IoResult<UnixStream> {
as_utf16_p(addr.as_str().unwrap(), |p| { let addr = try!(to_utf16(addr));
let start = ::io::timer::now(); let start = ::io::timer::now();
loop { loop {
match UnixStream::try_connect(p) { match UnixStream::try_connect(addr.as_ptr()) {
Some(handle) => { Some(handle) => {
let inner = Inner::new(handle); let inner = Inner::new(handle);
let mut mode = libc::PIPE_TYPE_BYTE | let mut mode = libc::PIPE_TYPE_BYTE |
libc::PIPE_READMODE_BYTE | libc::PIPE_READMODE_BYTE |
libc::PIPE_WAIT; libc::PIPE_WAIT;
let ret = unsafe { let ret = unsafe {
libc::SetNamedPipeHandleState(inner.handle, libc::SetNamedPipeHandleState(inner.handle,
&mut mode, &mut mode,
ptr::mut_null(), ptr::mut_null(),
ptr::mut_null()) ptr::mut_null())
}; };
return if ret == 0 { return if ret == 0 {
Err(super::last_error()) Err(super::last_error())
} else { } else {
Ok(UnixStream { Ok(UnixStream {
inner: Arc::new(inner), inner: Arc::new(inner),
read: None, read: None,
write: None, write: None,
read_deadline: 0, read_deadline: 0,
write_deadline: 0, write_deadline: 0,
}) })
} }
}
None => {}
}
// On windows, if you fail to connect, you may need to call the
// `WaitNamedPipe` function, and this is indicated with an error
// code of ERROR_PIPE_BUSY.
let code = unsafe { libc::GetLastError() };
if code as int != libc::ERROR_PIPE_BUSY as int {
return Err(super::last_error())
}
match timeout {
Some(timeout) => {
let now = ::io::timer::now();
let timed_out = (now - start) >= timeout || unsafe {
let ms = (timeout - (now - start)) as libc::DWORD;
libc::WaitNamedPipeW(addr.as_ptr(), ms) == 0
};
if timed_out {
return Err(util::timeout("connect timed out"))
} }
None => {}
} }
// On windows, if you fail to connect, you may need to call the // An example I found on microsoft's website used 20
// `WaitNamedPipe` function, and this is indicated with an error // seconds, libuv uses 30 seconds, hence we make the
// code of ERROR_PIPE_BUSY. // obvious choice of waiting for 25 seconds.
let code = unsafe { libc::GetLastError() }; None => {
if code as int != libc::ERROR_PIPE_BUSY as int { if unsafe { libc::WaitNamedPipeW(addr.as_ptr(), 25000) } == 0 {
return Err(super::last_error()) return Err(super::last_error())
}
match timeout {
Some(timeout) => {
let now = ::io::timer::now();
let timed_out = (now - start) >= timeout || unsafe {
let ms = (timeout - (now - start)) as libc::DWORD;
libc::WaitNamedPipeW(p, ms) == 0
};
if timed_out {
return Err(util::timeout("connect timed out"))
}
}
// An example I found on microsoft's website used 20
// seconds, libuv uses 30 seconds, hence we make the
// obvious choice of waiting for 25 seconds.
None => {
if unsafe { libc::WaitNamedPipeW(p, 25000) } == 0 {
return Err(super::last_error())
}
} }
} }
} }
}) }
} }
fn handle(&self) -> libc::HANDLE { self.inner.handle } fn handle(&self) -> libc::HANDLE { self.inner.handle }
@ -564,14 +563,13 @@ impl UnixListener {
// Although we technically don't need the pipe until much later, we // Although we technically don't need the pipe until much later, we
// create the initial handle up front to test the validity of the name // create the initial handle up front to test the validity of the name
// and such. // and such.
as_utf16_p(addr.as_str().unwrap(), |p| { let addr_v = try!(to_utf16(addr));
let ret = unsafe { pipe(p, true) }; let ret = unsafe { pipe(addr_v.as_ptr(), true) };
if ret == libc::INVALID_HANDLE_VALUE as libc::HANDLE { if ret == libc::INVALID_HANDLE_VALUE as libc::HANDLE {
Err(super::last_error()) Err(super::last_error())
} else { } else {
Ok(UnixListener { handle: ret, name: addr.clone() }) Ok(UnixListener { handle: ret, name: addr.clone() })
} }
})
} }
pub fn native_listen(self) -> IoResult<UnixAcceptor> { pub fn native_listen(self) -> IoResult<UnixAcceptor> {
@ -639,6 +637,8 @@ impl UnixAcceptor {
// using the original server pipe. // using the original server pipe.
let handle = self.listener.handle; let handle = self.listener.handle;
let name = try!(to_utf16(&self.listener.name));
// Once we've got a "server handle", we need to wait for a client to // Once we've got a "server handle", we need to wait for a client to
// connect. The ConnectNamedPipe function will block this thread until // connect. The ConnectNamedPipe function will block this thread until
// someone on the other end connects. This function can "fail" if a // someone on the other end connects. This function can "fail" if a
@ -678,9 +678,7 @@ impl UnixAcceptor {
// Now that we've got a connected client to our handle, we need to // Now that we've got a connected client to our handle, we need to
// create a second server pipe. If this fails, we disconnect the // create a second server pipe. If this fails, we disconnect the
// connected client and return an error (see comments above). // connected client and return an error (see comments above).
let new_handle = as_utf16_p(self.listener.name.as_str().unwrap(), |p| { let new_handle = unsafe { pipe(name.as_ptr(), false) };
unsafe { pipe(p, false) }
});
if new_handle == libc::INVALID_HANDLE_VALUE as libc::HANDLE { if new_handle == libc::INVALID_HANDLE_VALUE as libc::HANDLE {
let ret = Err(super::last_error()); let ret = Err(super::last_error());
// If our disconnection fails, then there's not really a whole lot // If our disconnection fails, then there's not really a whole lot

View File

@ -296,16 +296,15 @@ fn spawn_process_os(cfg: ProcessConfig,
lpSecurityDescriptor: ptr::mut_null(), lpSecurityDescriptor: ptr::mut_null(),
bInheritHandle: 1, bInheritHandle: 1,
}; };
*slot = os::win32::as_utf16_p("NUL", |filename| { let filename = "NUL".to_utf16().append_one(0);
libc::CreateFileW(filename, *slot = libc::CreateFileW(filename.as_ptr(),
access, access,
libc::FILE_SHARE_READ | libc::FILE_SHARE_READ |
libc::FILE_SHARE_WRITE, libc::FILE_SHARE_WRITE,
&mut sa, &mut sa,
libc::OPEN_EXISTING, libc::OPEN_EXISTING,
0, 0,
ptr::mut_null()) ptr::mut_null());
});
if *slot == INVALID_HANDLE_VALUE as libc::HANDLE { if *slot == INVALID_HANDLE_VALUE as libc::HANDLE {
return Err(super::last_error()) return Err(super::last_error())
} }
@ -338,18 +337,17 @@ fn spawn_process_os(cfg: ProcessConfig,
with_envp(cfg.env, |envp| { with_envp(cfg.env, |envp| {
with_dirp(cfg.cwd, |dirp| { with_dirp(cfg.cwd, |dirp| {
os::win32::as_mut_utf16_p(cmd_str.as_slice(), |cmdp| { let mut cmd_str = cmd_str.to_utf16().append_one(0);
let created = CreateProcessW(ptr::null(), let created = CreateProcessW(ptr::null(),
cmdp, cmd_str.as_mut_ptr(),
ptr::mut_null(), ptr::mut_null(),
ptr::mut_null(), ptr::mut_null(),
TRUE, TRUE,
flags, envp, dirp, flags, envp, dirp,
&mut si, &mut pi); &mut si, &mut pi);
if created == FALSE { if created == FALSE {
create_err = Some(super::last_error()); create_err = Some(super::last_error());
} }
})
}) })
}); });
@ -740,7 +738,8 @@ fn with_dirp<T>(d: Option<&CString>, cb: |*u16| -> T) -> T {
Some(dir) => { Some(dir) => {
let dir_str = dir.as_str() let dir_str = dir.as_str()
.expect("expected workingdirectory to be utf-8 encoded"); .expect("expected workingdirectory to be utf-8 encoded");
os::win32::as_utf16_p(dir_str, cb) let dir_str = dir_str.to_utf16().append_one(0);
cb(dir_str.as_ptr())
}, },
None => cb(ptr::null()) None => cb(ptr::null())
} }

View File

@ -135,7 +135,6 @@ mod imp {
mod imp { mod imp {
use libc; use libc;
use std::mem; use std::mem;
use std::os::win32::as_utf16_p;
use std::os; use std::os;
use std::ptr; use std::ptr;
@ -162,8 +161,9 @@ mod imp {
impl Lock { impl Lock {
pub fn new(p: &Path) -> Lock { pub fn new(p: &Path) -> Lock {
let handle = as_utf16_p(p.as_str().unwrap(), |p| unsafe { let p_16 = p.as_str().unwrap().to_utf16().append_one(0);
libc::CreateFileW(p, let handle = unsafe {
libc::CreateFileW(p_16.as_ptr(),
libc::FILE_GENERIC_READ | libc::FILE_GENERIC_READ |
libc::FILE_GENERIC_WRITE, libc::FILE_GENERIC_WRITE,
libc::FILE_SHARE_READ | libc::FILE_SHARE_READ |
@ -173,7 +173,7 @@ mod imp {
libc::CREATE_ALWAYS, libc::CREATE_ALWAYS,
libc::FILE_ATTRIBUTE_NORMAL, libc::FILE_ATTRIBUTE_NORMAL,
ptr::mut_null()) ptr::mut_null())
}); };
if handle as uint == libc::INVALID_HANDLE_VALUE as uint { if handle as uint == libc::INVALID_HANDLE_VALUE as uint {
fail!("create file error: {}", os::last_os_error()); fail!("create file error: {}", os::last_os_error());
} }

View File

@ -133,7 +133,7 @@ pub mod win32 {
use os::TMPBUF_SZ; use os::TMPBUF_SZ;
use slice::{MutableVector, ImmutableVector}; use slice::{MutableVector, ImmutableVector};
use string::String; use string::String;
use str::{StrSlice, StrAllocating}; use str::StrSlice;
use str; use str;
use vec::Vec; use vec::Vec;
@ -171,17 +171,6 @@ pub mod win32 {
return res; return res;
} }
} }
pub fn as_utf16_p<T>(s: &str, f: |*u16| -> T) -> T {
as_mut_utf16_p(s, |t| { f(t as *u16) })
}
pub fn as_mut_utf16_p<T>(s: &str, f: |*mut u16| -> T) -> T {
let mut t = s.to_utf16();
// Null terminate before passing on.
t.push(0u16);
f(t.as_mut_ptr())
}
} }
/* /*
@ -356,11 +345,10 @@ pub fn getenv_as_bytes(n: &str) -> Option<Vec<u8>> {
pub fn getenv(n: &str) -> Option<String> { pub fn getenv(n: &str) -> Option<String> {
unsafe { unsafe {
with_env_lock(|| { with_env_lock(|| {
use os::win32::{as_utf16_p, fill_utf16_buf_and_decode}; use os::win32::{fill_utf16_buf_and_decode};
as_utf16_p(n, |u| { let n = n.to_utf16().append_one(0);
fill_utf16_buf_and_decode(|buf, sz| { fill_utf16_buf_and_decode(|buf, sz| {
libc::GetEnvironmentVariableW(u, buf, sz) libc::GetEnvironmentVariableW(n.as_ptr(), buf, sz)
})
}) })
}) })
} }
@ -398,14 +386,11 @@ pub fn setenv(n: &str, v: &str) {
/// Sets the environment variable `n` to the value `v` for the currently running /// Sets the environment variable `n` to the value `v` for the currently running
/// process /// process
pub fn setenv(n: &str, v: &str) { pub fn setenv(n: &str, v: &str) {
let n = n.to_utf16().append_one(0);
let v = v.to_utf16().append_one(0);
unsafe { unsafe {
with_env_lock(|| { with_env_lock(|| {
use os::win32::as_utf16_p; libc::SetEnvironmentVariableW(n.as_ptr(), v.as_ptr());
as_utf16_p(n, |nbuf| {
as_utf16_p(v, |vbuf| {
libc::SetEnvironmentVariableW(nbuf, vbuf);
})
})
}) })
} }
} }
@ -428,12 +413,10 @@ pub fn unsetenv(n: &str) {
} }
#[cfg(windows)] #[cfg(windows)]
fn _unsetenv(n: &str) { fn _unsetenv(n: &str) {
let n = n.to_utf16().append_one(0);
unsafe { unsafe {
with_env_lock(|| { with_env_lock(|| {
use os::win32::as_utf16_p; libc::SetEnvironmentVariableW(n.as_ptr(), ptr::null());
as_utf16_p(n, |nbuf| {
libc::SetEnvironmentVariableW(nbuf, ptr::null());
})
}) })
} }
} }
@ -732,11 +715,12 @@ pub fn change_dir(p: &Path) -> bool {
#[cfg(windows)] #[cfg(windows)]
fn chdir(p: &Path) -> bool { fn chdir(p: &Path) -> bool {
let p = match p.as_str() {
Some(s) => s.to_utf16().append_one(0),
None => return false,
};
unsafe { unsafe {
use os::win32::as_utf16_p; libc::SetCurrentDirectoryW(p.as_ptr()) != (0 as libc::BOOL)
return as_utf16_p(p.as_str().unwrap(), |buf| {
libc::SetCurrentDirectoryW(buf) != (0 as libc::BOOL)
});
} }
} }

View File

@ -272,21 +272,21 @@ pub mod dl {
#[cfg(target_os = "win32")] #[cfg(target_os = "win32")]
pub mod dl { pub mod dl {
use c_str::ToCStr;
use libc; use libc;
use os; use os;
use ptr; use ptr;
use result::{Ok, Err, Result}; use result::{Ok, Err, Result};
use string::String; use str::StrAllocating;
use str; use str;
use c_str::ToCStr; use string::String;
pub unsafe fn open_external<T: ToCStr>(filename: T) -> *u8 { pub unsafe fn open_external<T: ToCStr>(filename: T) -> *u8 {
// Windows expects Unicode data // Windows expects Unicode data
let filename_cstr = filename.to_c_str(); let filename_cstr = filename.to_c_str();
let filename_str = str::from_utf8(filename_cstr.as_bytes_no_nul()).unwrap(); let filename_str = str::from_utf8(filename_cstr.as_bytes_no_nul()).unwrap();
os::win32::as_utf16_p(filename_str, |raw_name| { let filename_str = filename_str.to_utf16().append_one(0);
LoadLibraryW(raw_name as *libc::c_void) as *u8 LoadLibraryW(filename_str.as_ptr() as *libc::c_void) as *u8
})
} }
pub unsafe fn open_internal() -> *u8 { pub unsafe fn open_internal() -> *u8 {