rustix/fs/
xattr.rs

1//! Extended attribute functions.
2
3#![allow(unsafe_code)]
4
5use crate::buffer::Buffer;
6use crate::{backend, ffi, io, path};
7use backend::c;
8use backend::fd::AsFd;
9use bitflags::bitflags;
10
11bitflags! {
12    /// `XATTR_*` constants for use with [`setxattr`], and other `*setxattr`
13    /// functions.
14    #[repr(transparent)]
15    #[derive(Copy, Clone, Eq, PartialEq, Hash, Debug)]
16    pub struct XattrFlags: ffi::c_uint {
17        /// `XATTR_CREATE`
18        const CREATE = c::XATTR_CREATE as c::c_uint;
19
20        /// `XATTR_REPLACE`
21        const REPLACE = c::XATTR_REPLACE as c::c_uint;
22
23        /// <https://docs.rs/bitflags/*/bitflags/#externally-defined-flags>
24        const _ = !0;
25    }
26}
27
28/// `getxattr(path, name, value)`—Get extended filesystem attributes.
29///
30/// For a higher-level API to xattr functionality, see the [xattr] crate.
31///
32/// [xattr]: https://crates.io/crates/xattr
33///
34/// # References
35///  - [Linux]
36///  - [Apple]
37///
38/// [Linux]: https://man7.org/linux/man-pages/man2/getxattr.2.html
39/// [Apple]: https://developer.apple.com/library/archive/documentation/System/Conceptual/ManPages_iPhoneOS/man2/getxattr.2.html
40#[inline]
41pub fn getxattr<P: path::Arg, Name: path::Arg, Buf: Buffer<u8>>(
42    path: P,
43    name: Name,
44    mut value: Buf,
45) -> io::Result<Buf::Output> {
46    path.into_with_c_str(|path| {
47        name.into_with_c_str(|name| {
48            // SAFETY: `getxattr` behaves.
49            let len = unsafe { backend::fs::syscalls::getxattr(path, name, value.parts_mut())? };
50            // SAFETY: `getxattr` behaves.
51            unsafe { Ok(value.assume_init(len)) }
52        })
53    })
54}
55
56/// `lgetxattr(path, name, value.as_ptr(), value.len())`—Get extended
57/// filesystem attributes, without following symlinks in the last path
58/// component.
59///
60/// # References
61///  - [Linux]
62///
63/// [Linux]: https://man7.org/linux/man-pages/man2/lgetxattr.2.html
64#[inline]
65pub fn lgetxattr<P: path::Arg, Name: path::Arg, Buf: Buffer<u8>>(
66    path: P,
67    name: Name,
68    mut value: Buf,
69) -> io::Result<Buf::Output> {
70    path.into_with_c_str(|path| {
71        name.into_with_c_str(|name| {
72            // SAFETY: `lgetxattr` behaves.
73            let len = unsafe { backend::fs::syscalls::lgetxattr(path, name, value.parts_mut())? };
74            // SAFETY: `lgetxattr` behaves.
75            unsafe { Ok(value.assume_init(len)) }
76        })
77    })
78}
79
80/// `fgetxattr(fd, name, value.as_ptr(), value.len())`—Get extended
81/// filesystem attributes on an open file descriptor.
82///
83/// # References
84///  - [Linux]
85///  - [Apple]
86///
87/// [Linux]: https://man7.org/linux/man-pages/man2/fgetxattr.2.html
88/// [Apple]: https://developer.apple.com/library/archive/documentation/System/Conceptual/ManPages_iPhoneOS/man2/fgetxattr.2.html
89#[inline]
90pub fn fgetxattr<Fd: AsFd, Name: path::Arg, Buf: Buffer<u8>>(
91    fd: Fd,
92    name: Name,
93    mut value: Buf,
94) -> io::Result<Buf::Output> {
95    name.into_with_c_str(|name| {
96        // SAFETY: `fgetxattr` behaves.
97        let len = unsafe { backend::fs::syscalls::fgetxattr(fd.as_fd(), name, value.parts_mut())? };
98        // SAFETY: `fgetxattr` behaves.
99        unsafe { Ok(value.assume_init(len)) }
100    })
101}
102
103/// `setxattr(path, name, value.as_ptr(), value.len(), flags)`—Set extended
104/// filesystem attributes.
105///
106/// # References
107///  - [Linux]
108///  - [Apple]
109///
110/// [Linux]: https://man7.org/linux/man-pages/man2/setxattr.2.html
111/// [Apple]: https://developer.apple.com/library/archive/documentation/System/Conceptual/ManPages_iPhoneOS/man2/setxattr.2.html
112#[inline]
113pub fn setxattr<P: path::Arg, Name: path::Arg>(
114    path: P,
115    name: Name,
116    value: &[u8],
117    flags: XattrFlags,
118) -> io::Result<()> {
119    path.into_with_c_str(|path| {
120        name.into_with_c_str(|name| backend::fs::syscalls::setxattr(path, name, value, flags))
121    })
122}
123
124/// `setxattr(path, name, value.as_ptr(), value.len(), flags)`—Set extended
125/// filesystem attributes, without following symlinks in the last path
126/// component.
127///
128/// # References
129///  - [Linux]
130///
131/// [Linux]: https://man7.org/linux/man-pages/man2/lsetxattr.2.html
132#[inline]
133pub fn lsetxattr<P: path::Arg, Name: path::Arg>(
134    path: P,
135    name: Name,
136    value: &[u8],
137    flags: XattrFlags,
138) -> io::Result<()> {
139    path.into_with_c_str(|path| {
140        name.into_with_c_str(|name| backend::fs::syscalls::lsetxattr(path, name, value, flags))
141    })
142}
143
144/// `fsetxattr(fd, name, value.as_ptr(), value.len(), flags)`—Set extended
145/// filesystem attributes on an open file descriptor.
146///
147/// # References
148///  - [Linux]
149///  - [Apple]
150///
151/// [Linux]: https://man7.org/linux/man-pages/man2/fsetxattr.2.html
152/// [Apple]: https://developer.apple.com/library/archive/documentation/System/Conceptual/ManPages_iPhoneOS/man2/fsetxattr.2.html
153#[inline]
154pub fn fsetxattr<Fd: AsFd, Name: path::Arg>(
155    fd: Fd,
156    name: Name,
157    value: &[u8],
158    flags: XattrFlags,
159) -> io::Result<()> {
160    name.into_with_c_str(|name| backend::fs::syscalls::fsetxattr(fd.as_fd(), name, value, flags))
161}
162
163/// `listxattr(path, list.as_ptr(), list.len())`—List extended filesystem
164/// attributes.
165///
166/// # References
167///  - [Linux]
168///
169/// [Linux]: https://man7.org/linux/man-pages/man2/listxattr.2.html
170/// [Apple]: https://developer.apple.com/library/archive/documentation/System/Conceptual/ManPages_iPhoneOS/man2/listxattr.2.html
171#[inline]
172pub fn listxattr<P: path::Arg, Buf: Buffer<u8>>(path: P, mut list: Buf) -> io::Result<Buf::Output> {
173    path.into_with_c_str(|path| {
174        // SAFETY: `listxattr` behaves.
175        let len = unsafe { backend::fs::syscalls::listxattr(path, list.parts_mut())? };
176        // SAFETY: `listxattr` behaves.
177        unsafe { Ok(list.assume_init(len)) }
178    })
179}
180
181/// `llistxattr(path, list.as_ptr(), list.len())`—List extended filesystem
182/// attributes, without following symlinks in the last path component.
183///
184/// # References
185///  - [Linux]
186///
187/// [Linux]: https://man7.org/linux/man-pages/man2/llistxattr.2.html
188#[inline]
189pub fn llistxattr<P: path::Arg, Buf: Buffer<u8>>(
190    path: P,
191    mut list: Buf,
192) -> io::Result<Buf::Output> {
193    path.into_with_c_str(|path| {
194        // SAFETY: `flistxattr` behaves.
195        let len = unsafe { backend::fs::syscalls::llistxattr(path, list.parts_mut())? };
196        // SAFETY: `flistxattr` behaves.
197        unsafe { Ok(list.assume_init(len)) }
198    })
199}
200
201/// `flistxattr(fd, list.as_ptr(), list.len())`—List extended filesystem
202/// attributes on an open file descriptor.
203///
204/// # References
205///  - [Linux]
206///  - [Apple]
207///
208/// [Linux]: https://man7.org/linux/man-pages/man2/flistxattr.2.html
209/// [Apple]: https://developer.apple.com/library/archive/documentation/System/Conceptual/ManPages_iPhoneOS/man2/flistxattr.2.html
210#[inline]
211pub fn flistxattr<Fd: AsFd, Buf: Buffer<u8>>(fd: Fd, mut list: Buf) -> io::Result<Buf::Output> {
212    // SAFETY: `flistxattr` behaves.
213    let len = unsafe { backend::fs::syscalls::flistxattr(fd.as_fd(), list.parts_mut())? };
214    // SAFETY: `flistxattr` behaves.
215    unsafe { Ok(list.assume_init(len)) }
216}
217
218/// `removexattr(path, name)`—Remove an extended filesystem attribute.
219///
220/// # References
221///  - [Linux]
222///  - [Apple]
223///
224/// [Linux]: https://man7.org/linux/man-pages/man2/removexattr.2.html
225/// [Apple]: https://developer.apple.com/library/archive/documentation/System/Conceptual/ManPages_iPhoneOS/man2/removexattr.2.html
226pub fn removexattr<P: path::Arg, Name: path::Arg>(path: P, name: Name) -> io::Result<()> {
227    path.into_with_c_str(|path| {
228        name.into_with_c_str(|name| backend::fs::syscalls::removexattr(path, name))
229    })
230}
231
232/// `lremovexattr(path, name)`—Remove an extended filesystem attribute,
233/// without following symlinks in the last path component.
234///
235/// # References
236///  - [Linux]
237///
238/// [Linux]: https://man7.org/linux/man-pages/man2/lremovexattr.2.html
239pub fn lremovexattr<P: path::Arg, Name: path::Arg>(path: P, name: Name) -> io::Result<()> {
240    path.into_with_c_str(|path| {
241        name.into_with_c_str(|name| backend::fs::syscalls::lremovexattr(path, name))
242    })
243}
244
245/// `fremovexattr(fd, name)`—Remove an extended filesystem attribute on an
246/// open file descriptor.
247///
248/// # References
249///  - [Linux]
250///  - [Apple]
251///
252/// [Linux]: https://man7.org/linux/man-pages/man2/fremovexattr.2.html
253/// [Apple]: https://developer.apple.com/library/archive/documentation/System/Conceptual/ManPages_iPhoneOS/man2/fremovexattr.2.html
254pub fn fremovexattr<Fd: AsFd, Name: path::Arg>(fd: Fd, name: Name) -> io::Result<()> {
255    name.into_with_c_str(|name| backend::fs::syscalls::fremovexattr(fd.as_fd(), name))
256}