1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
|
.Dd August 17, 2024
.Dt GETXATTR 2
.Os Camellia
.Sh NAME
.Nm getxattr
.Nd get file attribute
.Sh SYNOPSIS
.In camellia/syscalls.h
.Ft ssize_t
.Fo _sys_getxattr
.Fa "hid_t h"
.Fa "const char *name"
.Fa "void *buf"
.Fa "size_t len"
.Fa "int flags"
.Fc
.Sh DESCRIPTION
.Nm
reads the attribute of
.Fa h
named by
.Fa name ,
into
.Fa buf .
.Fa flags
is reserved for future use, for now it has to be 0.
.Pp
The attributes can have arbitrary names of up to
.Dv XATTRNAME_MAX
characters, but some are explicitly defined:
.Bl -tag -width virt.index
.It Sy virt.index
The list of all xattrs in arbitrary order as concatenated NUL-terminated names,
including exactly one NUL at the end.
.It Sy virt.mode
The Unix file mode in octal, with or without the trailing 0.
.It Sy virt.owner
The Unix file owner, as either a decimal UID or an username that doesn't begin
with a digit.
.It Sy virt.group
See
.Sy virt.owner .
.El
.Sh RETURN VALUES
.Nm
returns the length of the entire attribute (which might be larger than len),
or a negative value on failure.
The most common errors are:
.Bl -tag -width EGENERIC
.It Er ENOSYS
The filesystem doesn't support xattrs.
.It Er ENOENT
This xattr doesn't exist.
.It Er EGENERIC
The filesystem probably doesn't support xattrs,
and it wasn't written very well.
Sorry.
.El
.\" .Sh SEE ALSO
.Sh NOTES
The maximum size of the xattr value might become limited in the future,
as in Linux.
.Sh RATIONALE
The original idea for storing metadata was something like
Alternate Data Streams, where you'd be able to open another
.Dq file
that stores the metadata for any handle.
It would either be unstructured
.Pq gross ,
or you'd need to open a separate file for each attribute
.Pq slow .
It also seemed like it would be complex to implement.
.Pp
A generic key-value store thing for handles also happens to seem more useful.
I was thinking about making the API more similar to
.Xr read 2
and
.Xr write 2
with an offset argument, but that seems unnecessary.
I don't expect the xattr values to be particularly large,
and only allowing writes/reads of an entire xattr at once will simplify
future attempts to add wrapper filesystems that enforce Unix-style permissions.
This made me end up with a Linux-style interface
.Pq thus the name .
.Pp
Instead of a separate syscall for listing xattrs I just use
.Sy virt.index ,
as that seems simpler and without any downside.
It also lets me define paging using
.Sy virt.index.1
and so on, avoiding the issue Linux has with listxattrs.
|