1a068c194SPoul-Henning Kamp /*-
21de7b4b8SPedro F. Giffuni * SPDX-License-Identifier: BSD-3-Clause
31de7b4b8SPedro F. Giffuni *
452271f82SRobert Watson * Copyright (c) 2002, 2003 Networks Associates Technology, Inc.
5a068c194SPoul-Henning Kamp * Copyright (c) 2002 Poul-Henning Kamp.
6a068c194SPoul-Henning Kamp * Copyright (c) 1999, 2000, 2001, 2002 Robert N. M. Watson
7a068c194SPoul-Henning Kamp * All rights reserved.
8a068c194SPoul-Henning Kamp *
9a068c194SPoul-Henning Kamp * This software was developed for the FreeBSD Project by Poul-Henning
10a068c194SPoul-Henning Kamp * Kamp and Network Associates Laboratories, the Security Research Division
11a068c194SPoul-Henning Kamp * of Network Associates, Inc. under DARPA/SPAWAR contract N66001-01-C-8035
12a068c194SPoul-Henning Kamp * ("CBOSS"), as part of the DARPA CHATS research program
13a068c194SPoul-Henning Kamp *
14a068c194SPoul-Henning Kamp * Redistribution and use in source and binary forms, with or without
15a068c194SPoul-Henning Kamp * modification, are permitted provided that the following conditions
16a068c194SPoul-Henning Kamp * are met:
17a068c194SPoul-Henning Kamp * 1. Redistributions of source code must retain the above copyright
18a068c194SPoul-Henning Kamp * notice, this list of conditions and the following disclaimer.
19a068c194SPoul-Henning Kamp * 2. Redistributions in binary form must reproduce the above copyright
20a068c194SPoul-Henning Kamp * notice, this list of conditions and the following disclaimer in the
21a068c194SPoul-Henning Kamp * documentation and/or other materials provided with the distribution.
22a068c194SPoul-Henning Kamp * 3. The names of the authors may not be used to endorse or promote
23a068c194SPoul-Henning Kamp * products derived from this software without specific prior written
24a068c194SPoul-Henning Kamp * permission.
25a068c194SPoul-Henning Kamp *
26a068c194SPoul-Henning Kamp * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
27a068c194SPoul-Henning Kamp * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
28a068c194SPoul-Henning Kamp * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
29a068c194SPoul-Henning Kamp * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
30a068c194SPoul-Henning Kamp * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
31a068c194SPoul-Henning Kamp * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
32a068c194SPoul-Henning Kamp * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
33a068c194SPoul-Henning Kamp * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
34a068c194SPoul-Henning Kamp * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
35a068c194SPoul-Henning Kamp * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
36a068c194SPoul-Henning Kamp * SUCH DAMAGE.
37a068c194SPoul-Henning Kamp */
38a068c194SPoul-Henning Kamp
39a068c194SPoul-Henning Kamp #include <sys/types.h>
40c29930cfSAlan Somers #include <sys/sbuf.h>
41a068c194SPoul-Henning Kamp #include <sys/uio.h>
42a068c194SPoul-Henning Kamp #include <sys/extattr.h>
43a068c194SPoul-Henning Kamp
447522ecb3SRobert Watson #include <libgen.h>
45a068c194SPoul-Henning Kamp #include <libutil.h>
46a068c194SPoul-Henning Kamp #include <stdio.h>
47a068c194SPoul-Henning Kamp #include <stdlib.h>
48a068c194SPoul-Henning Kamp #include <string.h>
49a068c194SPoul-Henning Kamp #include <unistd.h>
50a068c194SPoul-Henning Kamp #include <vis.h>
51a068c194SPoul-Henning Kamp #include <err.h>
52a068c194SPoul-Henning Kamp #include <errno.h>
53a068c194SPoul-Henning Kamp
54a068c194SPoul-Henning Kamp static enum { EADUNNO, EAGET, EASET, EARM, EALS } what = EADUNNO;
55a068c194SPoul-Henning Kamp
56a068c194SPoul-Henning Kamp static void __dead2
usage(void)57a068c194SPoul-Henning Kamp usage(void)
58a068c194SPoul-Henning Kamp {
59a068c194SPoul-Henning Kamp
60a068c194SPoul-Henning Kamp switch (what) {
61a068c194SPoul-Henning Kamp case EAGET:
626c3fb112SRobert Watson fprintf(stderr, "usage: getextattr [-fhqsx] attrnamespace");
63a068c194SPoul-Henning Kamp fprintf(stderr, " attrname filename ...\n");
64a068c194SPoul-Henning Kamp exit(-1);
65a068c194SPoul-Henning Kamp case EASET:
66cb0187a0SBrian Feldman fprintf(stderr, "usage: setextattr [-fhnq] attrnamespace");
67a068c194SPoul-Henning Kamp fprintf(stderr, " attrname attrvalue filename ...\n");
68c29930cfSAlan Somers fprintf(stderr, " or setextattr -i [-fhnq] attrnamespace");
69c29930cfSAlan Somers fprintf(stderr, " attrname filename ...\n");
70a068c194SPoul-Henning Kamp exit(-1);
71a068c194SPoul-Henning Kamp case EARM:
726c3fb112SRobert Watson fprintf(stderr, "usage: rmextattr [-fhq] attrnamespace");
73a068c194SPoul-Henning Kamp fprintf(stderr, " attrname filename ...\n");
74a068c194SPoul-Henning Kamp exit(-1);
75a068c194SPoul-Henning Kamp case EALS:
766c3fb112SRobert Watson fprintf(stderr, "usage: lsextattr [-fhq] attrnamespace");
77a068c194SPoul-Henning Kamp fprintf(stderr, " filename ...\n");
78a068c194SPoul-Henning Kamp exit(-1);
79a068c194SPoul-Henning Kamp case EADUNNO:
80a068c194SPoul-Henning Kamp default:
81a068c194SPoul-Henning Kamp fprintf(stderr, "usage: (getextattr|lsextattr|rmextattr");
82a068c194SPoul-Henning Kamp fprintf(stderr, "|setextattr)\n");
83a068c194SPoul-Henning Kamp exit (-1);
84a068c194SPoul-Henning Kamp }
85a068c194SPoul-Henning Kamp }
86a068c194SPoul-Henning Kamp
87a068c194SPoul-Henning Kamp static void
mkbuf(char ** buf,int * oldlen,int newlen)88a068c194SPoul-Henning Kamp mkbuf(char **buf, int *oldlen, int newlen)
89a068c194SPoul-Henning Kamp {
90a068c194SPoul-Henning Kamp
91a068c194SPoul-Henning Kamp if (*oldlen >= newlen)
92a068c194SPoul-Henning Kamp return;
93a068c194SPoul-Henning Kamp if (*buf != NULL)
94a068c194SPoul-Henning Kamp free(*buf);
95a068c194SPoul-Henning Kamp *buf = malloc(newlen);
96a068c194SPoul-Henning Kamp if (*buf == NULL)
97a068c194SPoul-Henning Kamp err(1, "malloc");
98a068c194SPoul-Henning Kamp *oldlen = newlen;
99a068c194SPoul-Henning Kamp }
100a068c194SPoul-Henning Kamp
101a068c194SPoul-Henning Kamp int
main(int argc,char * argv[])102a068c194SPoul-Henning Kamp main(int argc, char *argv[])
103a068c194SPoul-Henning Kamp {
104*1137d1a7SAlan Somers #define STDIN_BUF_SZ 4096
105c29930cfSAlan Somers char stdin_data[STDIN_BUF_SZ];
106c29930cfSAlan Somers char *p;
107a068c194SPoul-Henning Kamp
108a068c194SPoul-Henning Kamp const char *options, *attrname;
109e324bf91SMatthew D Fleming size_t len;
110e324bf91SMatthew D Fleming ssize_t ret;
111c29930cfSAlan Somers int ch, error, i, arg_counter, attrnamespace, minargc;
112a068c194SPoul-Henning Kamp
113c29930cfSAlan Somers char *visbuf = NULL;
114c29930cfSAlan Somers int visbuflen = 0;
115c29930cfSAlan Somers char *buf = NULL;
116c29930cfSAlan Somers int buflen = 0;
117c29930cfSAlan Somers struct sbuf *attrvalue = NULL;
118a068c194SPoul-Henning Kamp int flag_force = 0;
1196c3fb112SRobert Watson int flag_nofollow = 0;
120cb0187a0SBrian Feldman int flag_null = 0;
121c29930cfSAlan Somers int count_quiet = 0;
122c29930cfSAlan Somers int flag_from_stdin = 0;
123a068c194SPoul-Henning Kamp int flag_string = 0;
124a068c194SPoul-Henning Kamp int flag_hex = 0;
125a068c194SPoul-Henning Kamp
1267522ecb3SRobert Watson p = basename(argv[0]);
127a068c194SPoul-Henning Kamp if (p == NULL)
128a068c194SPoul-Henning Kamp p = argv[0];
129a068c194SPoul-Henning Kamp if (!strcmp(p, "getextattr")) {
130a068c194SPoul-Henning Kamp what = EAGET;
1316c3fb112SRobert Watson options = "fhqsx";
132878382faSBrian Feldman minargc = 3;
133a068c194SPoul-Henning Kamp } else if (!strcmp(p, "setextattr")) {
134a068c194SPoul-Henning Kamp what = EASET;
135c29930cfSAlan Somers options = "fhinq";
136c29930cfSAlan Somers minargc = 3;
137a068c194SPoul-Henning Kamp } else if (!strcmp(p, "rmextattr")) {
138a068c194SPoul-Henning Kamp what = EARM;
1396c3fb112SRobert Watson options = "fhq";
140878382faSBrian Feldman minargc = 3;
141a068c194SPoul-Henning Kamp } else if (!strcmp(p, "lsextattr")) {
142a068c194SPoul-Henning Kamp what = EALS;
1436c3fb112SRobert Watson options = "fhq";
144878382faSBrian Feldman minargc = 2;
145a068c194SPoul-Henning Kamp } else {
146a068c194SPoul-Henning Kamp usage();
147a068c194SPoul-Henning Kamp }
148a068c194SPoul-Henning Kamp
149a068c194SPoul-Henning Kamp while ((ch = getopt(argc, argv, options)) != -1) {
150a068c194SPoul-Henning Kamp switch (ch) {
151a068c194SPoul-Henning Kamp case 'f':
152a068c194SPoul-Henning Kamp flag_force = 1;
153a068c194SPoul-Henning Kamp break;
1546c3fb112SRobert Watson case 'h':
1556c3fb112SRobert Watson flag_nofollow = 1;
1566c3fb112SRobert Watson break;
157c29930cfSAlan Somers case 'i':
158c29930cfSAlan Somers flag_from_stdin = 1;
159c29930cfSAlan Somers break;
160cb0187a0SBrian Feldman case 'n':
161cb0187a0SBrian Feldman flag_null = 1;
162cb0187a0SBrian Feldman break;
163a068c194SPoul-Henning Kamp case 'q':
164c29930cfSAlan Somers count_quiet += 1;
165a068c194SPoul-Henning Kamp break;
166a068c194SPoul-Henning Kamp case 's':
167a068c194SPoul-Henning Kamp flag_string = 1;
168a068c194SPoul-Henning Kamp break;
169a068c194SPoul-Henning Kamp case 'x':
170a068c194SPoul-Henning Kamp flag_hex = 1;
171a068c194SPoul-Henning Kamp break;
172a068c194SPoul-Henning Kamp case '?':
173a068c194SPoul-Henning Kamp default:
174a068c194SPoul-Henning Kamp usage();
175a068c194SPoul-Henning Kamp }
176a068c194SPoul-Henning Kamp }
177a068c194SPoul-Henning Kamp
178a068c194SPoul-Henning Kamp argc -= optind;
179a068c194SPoul-Henning Kamp argv += optind;
180a068c194SPoul-Henning Kamp
181c29930cfSAlan Somers if (what == EASET && flag_from_stdin == 0)
182c29930cfSAlan Somers minargc++;
183c29930cfSAlan Somers
184878382faSBrian Feldman if (argc < minargc)
185a068c194SPoul-Henning Kamp usage();
186a068c194SPoul-Henning Kamp
187a068c194SPoul-Henning Kamp error = extattr_string_to_namespace(argv[0], &attrnamespace);
188a068c194SPoul-Henning Kamp if (error)
189b689e6a8SRobert Watson err(-1, "%s", argv[0]);
190a068c194SPoul-Henning Kamp argc--; argv++;
191a068c194SPoul-Henning Kamp
19252271f82SRobert Watson if (what != EALS) {
193a068c194SPoul-Henning Kamp attrname = argv[0];
194a068c194SPoul-Henning Kamp argc--; argv++;
19552271f82SRobert Watson } else
19652271f82SRobert Watson attrname = NULL;
197a068c194SPoul-Henning Kamp
198a068c194SPoul-Henning Kamp if (what == EASET) {
199c29930cfSAlan Somers attrvalue = sbuf_new_auto();
200c29930cfSAlan Somers if (flag_from_stdin) {
201c29930cfSAlan Somers while ((error = read(0, stdin_data, STDIN_BUF_SZ)) > 0)
202c29930cfSAlan Somers sbuf_bcat(attrvalue, stdin_data, error);
203c29930cfSAlan Somers } else {
204c29930cfSAlan Somers sbuf_cpy(attrvalue, argv[0]);
205a068c194SPoul-Henning Kamp argc--; argv++;
206a068c194SPoul-Henning Kamp }
207c29930cfSAlan Somers sbuf_finish(attrvalue);
208c29930cfSAlan Somers }
209a068c194SPoul-Henning Kamp
210a068c194SPoul-Henning Kamp for (arg_counter = 0; arg_counter < argc; arg_counter++) {
211a068c194SPoul-Henning Kamp switch (what) {
212a068c194SPoul-Henning Kamp case EARM:
2136c3fb112SRobert Watson if (flag_nofollow)
2146c3fb112SRobert Watson error = extattr_delete_link(argv[arg_counter],
2156c3fb112SRobert Watson attrnamespace, attrname);
2166c3fb112SRobert Watson else
217a068c194SPoul-Henning Kamp error = extattr_delete_file(argv[arg_counter],
218a068c194SPoul-Henning Kamp attrnamespace, attrname);
219a068c194SPoul-Henning Kamp if (error >= 0)
220a068c194SPoul-Henning Kamp continue;
221a068c194SPoul-Henning Kamp break;
222a068c194SPoul-Henning Kamp case EASET:
223c29930cfSAlan Somers len = sbuf_len(attrvalue) + flag_null;
2246c3fb112SRobert Watson if (flag_nofollow)
225e324bf91SMatthew D Fleming ret = extattr_set_link(argv[arg_counter],
226c29930cfSAlan Somers attrnamespace, attrname,
227c29930cfSAlan Somers sbuf_data(attrvalue), len);
2286c3fb112SRobert Watson else
229e324bf91SMatthew D Fleming ret = extattr_set_file(argv[arg_counter],
230c29930cfSAlan Somers attrnamespace, attrname,
231c29930cfSAlan Somers sbuf_data(attrvalue), len);
232e324bf91SMatthew D Fleming if (ret >= 0) {
233c29930cfSAlan Somers if ((size_t)ret != len && !count_quiet) {
234e324bf91SMatthew D Fleming warnx("Set %zd bytes of %zu for %s",
235e324bf91SMatthew D Fleming ret, len, attrname);
236e324bf91SMatthew D Fleming }
237a068c194SPoul-Henning Kamp continue;
238e324bf91SMatthew D Fleming }
239a068c194SPoul-Henning Kamp break;
240a068c194SPoul-Henning Kamp case EALS:
24152271f82SRobert Watson if (flag_nofollow)
242e324bf91SMatthew D Fleming ret = extattr_list_link(argv[arg_counter],
24352271f82SRobert Watson attrnamespace, NULL, 0);
24452271f82SRobert Watson else
245e324bf91SMatthew D Fleming ret = extattr_list_file(argv[arg_counter],
24652271f82SRobert Watson attrnamespace, NULL, 0);
247e324bf91SMatthew D Fleming if (ret < 0)
24852271f82SRobert Watson break;
249e324bf91SMatthew D Fleming mkbuf(&buf, &buflen, ret);
25052271f82SRobert Watson if (flag_nofollow)
251e324bf91SMatthew D Fleming ret = extattr_list_link(argv[arg_counter],
25252271f82SRobert Watson attrnamespace, buf, buflen);
25352271f82SRobert Watson else
254e324bf91SMatthew D Fleming ret = extattr_list_file(argv[arg_counter],
25552271f82SRobert Watson attrnamespace, buf, buflen);
256e324bf91SMatthew D Fleming if (ret < 0)
25752271f82SRobert Watson break;
258c29930cfSAlan Somers if (!count_quiet)
25952271f82SRobert Watson printf("%s\t", argv[arg_counter]);
260e324bf91SMatthew D Fleming for (i = 0; i < ret; i += ch + 1) {
26107c19910SZachary Loafman /* The attribute name length is unsigned. */
26207c19910SZachary Loafman ch = (unsigned char)buf[i];
26352271f82SRobert Watson printf("%s%*.*s", i ? "\t" : "",
26407c19910SZachary Loafman ch, ch, buf + i + 1);
26507c19910SZachary Loafman }
266c29930cfSAlan Somers if (!count_quiet || ret > 0)
26752271f82SRobert Watson printf("\n");
26852271f82SRobert Watson continue;
269a068c194SPoul-Henning Kamp case EAGET:
2706c3fb112SRobert Watson if (flag_nofollow)
271e324bf91SMatthew D Fleming ret = extattr_get_link(argv[arg_counter],
2726c3fb112SRobert Watson attrnamespace, attrname, NULL, 0);
2736c3fb112SRobert Watson else
274e324bf91SMatthew D Fleming ret = extattr_get_file(argv[arg_counter],
275a068c194SPoul-Henning Kamp attrnamespace, attrname, NULL, 0);
276e324bf91SMatthew D Fleming if (ret < 0)
277a068c194SPoul-Henning Kamp break;
278e324bf91SMatthew D Fleming mkbuf(&buf, &buflen, ret);
2796c3fb112SRobert Watson if (flag_nofollow)
280e324bf91SMatthew D Fleming ret = extattr_get_link(argv[arg_counter],
2816c3fb112SRobert Watson attrnamespace, attrname, buf, buflen);
2826c3fb112SRobert Watson else
283e324bf91SMatthew D Fleming ret = extattr_get_file(argv[arg_counter],
284a068c194SPoul-Henning Kamp attrnamespace, attrname, buf, buflen);
285e324bf91SMatthew D Fleming if (ret < 0)
286a068c194SPoul-Henning Kamp break;
287c29930cfSAlan Somers if (!count_quiet)
288a068c194SPoul-Henning Kamp printf("%s\t", argv[arg_counter]);
289a068c194SPoul-Henning Kamp if (flag_string) {
290e324bf91SMatthew D Fleming mkbuf(&visbuf, &visbuflen, ret * 4 + 1);
291e324bf91SMatthew D Fleming strvisx(visbuf, buf, ret,
292a068c194SPoul-Henning Kamp VIS_SAFE | VIS_WHITE);
293c29930cfSAlan Somers printf("\"%s\"", visbuf);
294a068c194SPoul-Henning Kamp } else if (flag_hex) {
295e324bf91SMatthew D Fleming for (i = 0; i < ret; i++)
29679e29f95SAlan Somers printf("%s%02x", i ? " " : "",
29779e29f95SAlan Somers (unsigned char)buf[i]);
298a068c194SPoul-Henning Kamp } else {
299e324bf91SMatthew D Fleming fwrite(buf, ret, 1, stdout);
300c29930cfSAlan Somers }
301c29930cfSAlan Somers if (count_quiet < 2)
302a068c194SPoul-Henning Kamp printf("\n");
303a068c194SPoul-Henning Kamp continue;
304a068c194SPoul-Henning Kamp default:
305a068c194SPoul-Henning Kamp break;
306a068c194SPoul-Henning Kamp }
307c29930cfSAlan Somers if (!count_quiet)
308a068c194SPoul-Henning Kamp warn("%s: failed", argv[arg_counter]);
309a068c194SPoul-Henning Kamp if (flag_force)
310a068c194SPoul-Henning Kamp continue;
311a068c194SPoul-Henning Kamp return(1);
312a068c194SPoul-Henning Kamp }
313a068c194SPoul-Henning Kamp return (0);
314a068c194SPoul-Henning Kamp }
315