From 73024e4b858e6c2083f40e8c987acedc22e9672b Mon Sep 17 00:00:00 2001 From: Corey Richardson Date: Mon, 27 Jan 2014 06:12:59 -0500 Subject: [PATCH] impl Clone for CString Clone tests --- src/libstd/c_str.rs | 52 +++++++++++++++++++++++++++++++++++++++++---- 1 file changed, 48 insertions(+), 4 deletions(-) diff --git a/src/libstd/c_str.rs b/src/libstd/c_str.rs index b7374b6f15d..4940358ddf9 100644 --- a/src/libstd/c_str.rs +++ b/src/libstd/c_str.rs @@ -68,6 +68,7 @@ use libc; use kinds::marker; use ops::Drop; +use clone::Clone; use option::{Option, Some, None}; use ptr::RawPtr; use ptr; @@ -76,6 +77,7 @@ use vec::{CloneableVector, ImmutableVector, MutableVector}; use vec; use unstable::intrinsics; +use rt::global_heap::malloc_raw; /// Resolution options for the `null_byte` condition pub enum NullByteResolution { @@ -99,6 +101,21 @@ pub struct CString { priv owns_buffer_: bool, } +impl Clone for CString { + /// Clone this CString into a new, uniquely owned CString. For safety + /// reasons, this is always a deep clone, rather than the usual shallow + /// clone. + fn clone(&self) -> CString { + if self.buf.is_null() { + CString { buf: self.buf, owns_buffer_: self.owns_buffer_ } + } else { + let buf = unsafe { malloc_raw(self.len()) } as *mut libc::c_char; + unsafe { ptr::copy_nonoverlapping_memory(buf, self.buf, self.len()); } + CString { buf: buf as *libc::c_char, owns_buffer_: true } + } + } +} + impl CString { /// Create a C String from a pointer. pub unsafe fn new(buf: *libc::c_char, owns_buffer: bool) -> CString { @@ -287,10 +304,7 @@ fn to_c_str(&self) -> CString { unsafe fn to_c_str_unchecked(&self) -> CString { let self_len = self.len(); - let buf = libc::malloc(self_len as libc::size_t + 1) as *mut u8; - if buf.is_null() { - fail!("failed to allocate memory!"); - } + let buf = malloc_raw(self_len + 1); ptr::copy_memory(buf, self.as_ptr(), self_len); *ptr::mut_offset(buf, self_len as int) = 0; @@ -598,6 +612,36 @@ fn test_iter_fail() { let c_str = unsafe { CString::new(ptr::null(), false) }; c_str.iter(); } + + #[test] + fn test_clone() { + let c_str = "hello".to_c_str(); + assert!(c_str == c_str.clone()); + } + + #[test] + fn test_clone_noleak() { + fn foo(f: |c: &CString|) { + let s = ~"test"; + let c = s.to_c_str(); + // give the closure a non-owned CString + let mut c_ = c.with_ref(|c| unsafe { CString::new(c, false) } ); + f(&c_); + // muck with the buffer for later printing + c_.with_mut_ref(|c| unsafe { *c = 'X' as libc::c_char } ); + } + + let mut c_: Option = None; + foo(|c| { + c_ = Some(c.clone()); + c.clone(); + // force a copy, reading the memory + c.as_bytes().to_owned(); + }); + let c_ = c_.unwrap(); + // force a copy, reading the memory + c_.as_bytes().to_owned(); + } } #[cfg(test)]