@@ -23,28 +23,32 @@ use crate::str;
23
23
///
24
24
/// This type represents a borrowed reference to a nul-terminated
25
25
/// array of bytes. It can be constructed safely from a <code>&[[u8]]</code>
26
- /// slice, or unsafely from a raw `*const c_char`. It can then be
27
- /// converted to a Rust <code>&[str]</code> by performing UTF-8 validation, or
28
- /// into an owned `CString`.
26
+ /// slice, or unsafely from a raw `*const c_char`. It can be expressed as a
27
+ /// literal in the form `c"Hello world"`.
28
+ ///
29
+ /// The `CStr` can then be converted to a Rust <code>&[str]</code> by performing
30
+ /// UTF-8 validation, or into an owned `CString`.
29
31
///
30
32
/// `&CStr` is to `CString` as <code>&[str]</code> is to `String`: the former
31
33
/// in each pair are borrowed references; the latter are owned
32
34
/// strings.
33
35
///
34
36
/// Note that this structure does **not** have a guaranteed layout (the `repr(transparent)`
35
- /// notwithstanding) and is not recommended to be placed in the signatures of FFI functions.
36
- /// Instead, safe wrappers of FFI functions may leverage the unsafe [`CStr::from_ptr `] constructor
37
- /// to provide a safe interface to other consumers.
37
+ /// notwithstanding) and should not be placed in the signatures of FFI functions.
38
+ /// Instead, safe wrappers of FFI functions may leverage [`CStr::as_ptr `] and the unsafe
39
+ /// [`CStr::from_ptr`] constructor to provide a safe interface to other consumers.
38
40
///
39
41
/// # Examples
40
42
///
41
43
/// Inspecting a foreign C string:
42
44
///
43
- /// ```ignore (extern-declaration)
45
+ /// ```
44
46
/// use std::ffi::CStr;
45
47
/// use std::os::raw::c_char;
46
48
///
49
+ /// # #[cfg(any())] // Extern functions are awkward in doc comments - fake it instead
47
50
/// extern "C" { fn my_string() -> *const c_char; }
51
+ /// # unsafe extern "C" fn my_string() -> *const c_char { c"hello".as_ptr() }
48
52
///
49
53
/// unsafe {
50
54
/// let slice = CStr::from_ptr(my_string());
@@ -54,12 +58,14 @@ use crate::str;
54
58
///
55
59
/// Passing a Rust-originating C string:
56
60
///
57
- /// ```ignore (extern-declaration)
61
+ /// ```
58
62
/// use std::ffi::{CString, CStr};
59
63
/// use std::os::raw::c_char;
60
64
///
61
65
/// fn work(data: &CStr) {
66
+ /// # #[cfg(any())] // Extern functions are awkward in doc comments - fake it instead
62
67
/// extern "C" { fn work_with(data: *const c_char); }
68
+ /// # unsafe extern "C" fn work_with(s: *const c_char) {}
63
69
///
64
70
/// unsafe { work_with(data.as_ptr()) }
65
71
/// }
@@ -70,11 +76,13 @@ use crate::str;
70
76
///
71
77
/// Converting a foreign C string into a Rust `String`:
72
78
///
73
- /// ```ignore (extern-declaration)
79
+ /// ```
74
80
/// use std::ffi::CStr;
75
81
/// use std::os::raw::c_char;
76
82
///
83
+ /// # #[cfg(any())] // Extern functions are awkward in doc comments - fake it instead
77
84
/// extern "C" { fn my_string() -> *const c_char; }
85
+ /// # unsafe extern "C" fn my_string() -> *const c_char { c"hello".as_ptr() }
78
86
///
79
87
/// fn my_string_safe() -> String {
80
88
/// let cstr = unsafe { CStr::from_ptr(my_string()) };
@@ -241,11 +249,11 @@ impl CStr {
241
249
///
242
250
/// # Examples
243
251
///
244
- /// ```ignore (extern-declaration)
252
+ /// ```
245
253
/// use std::ffi::{c_char, CStr};
246
254
///
247
- /// extern "C" {
248
- /// fn my_string() -> *const c_char;
255
+ /// fn my_string() -> *const c_char {
256
+ /// c"hello".as_ptr()
249
257
/// }
250
258
///
251
259
/// unsafe {
@@ -264,6 +272,8 @@ impl CStr {
264
272
/// BYTES.as_ptr().cast()
265
273
/// };
266
274
/// const HELLO: &CStr = unsafe { CStr::from_ptr(HELLO_PTR) };
275
+ ///
276
+ /// assert_eq!(c"Hello, world!", HELLO);
267
277
/// ```
268
278
///
269
279
/// [valid]: core::ptr#safety
@@ -549,6 +559,7 @@ impl CStr {
549
559
///
550
560
/// let empty_cstr = CStr::from_bytes_with_nul(b"\0")?;
551
561
/// assert!(empty_cstr.is_empty());
562
+ /// assert!(c"".is_empty());
552
563
/// # Ok(())
553
564
/// # }
554
565
/// ```
0 commit comments