Deleted Added
full compact
compare.c (121853) compare.c (122141)
1/*-
2 * Copyright (c) 1989, 1993
3 * The Regents of the University of California. All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 * 1. Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
13 * 3. Neither the name of the University nor the names of its contributors
14 * may be used to endorse or promote products derived from this software
15 * without specific prior written permission.
16 *
17 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
18 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
21 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27 * SUCH DAMAGE.
28 */
29
30#if 0
31#ifndef lint
32static char sccsid[] = "@(#)compare.c 8.1 (Berkeley) 6/6/93";
33#endif /* not lint */
34#endif
35#include <sys/cdefs.h>
1/*-
2 * Copyright (c) 1989, 1993
3 * The Regents of the University of California. All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 * 1. Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
13 * 3. Neither the name of the University nor the names of its contributors
14 * may be used to endorse or promote products derived from this software
15 * without specific prior written permission.
16 *
17 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
18 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
21 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27 * SUCH DAMAGE.
28 */
29
30#if 0
31#ifndef lint
32static char sccsid[] = "@(#)compare.c 8.1 (Berkeley) 6/6/93";
33#endif /* not lint */
34#endif
35#include <sys/cdefs.h>
36__FBSDID("$FreeBSD: head/usr.sbin/mtree/compare.c 121853 2003-11-01 08:43:54Z bde $");
36__FBSDID("$FreeBSD: head/usr.sbin/mtree/compare.c 122141 2003-11-05 22:26:08Z phk $");
37
38#include <sys/param.h>
39#include <sys/stat.h>
40#include <sys/time.h>
41
42#include <err.h>
43#include <errno.h>
44#include <fcntl.h>
45#include <fts.h>
46#ifdef MD5
47#include <md5.h>
48#endif
49#ifdef RMD160
50#include <ripemd.h>
51#endif
52#ifdef SHA1
53#include <sha.h>
54#endif
55#include <stdint.h>
56#include <stdio.h>
57#include <time.h>
58#include <unistd.h>
59#include <vis.h>
60
61#include "mtree.h"
62#include "extern.h"
63
64extern int uflag;
65extern int lineno;
66
37
38#include <sys/param.h>
39#include <sys/stat.h>
40#include <sys/time.h>
41
42#include <err.h>
43#include <errno.h>
44#include <fcntl.h>
45#include <fts.h>
46#ifdef MD5
47#include <md5.h>
48#endif
49#ifdef RMD160
50#include <ripemd.h>
51#endif
52#ifdef SHA1
53#include <sha.h>
54#endif
55#include <stdint.h>
56#include <stdio.h>
57#include <time.h>
58#include <unistd.h>
59#include <vis.h>
60
61#include "mtree.h"
62#include "extern.h"
63
64extern int uflag;
65extern int lineno;
66
67static const char *ftype(u_int);
68
69#define INDENTNAMELEN 8
70#define LABEL \
71 if (!label++) { \
72 len = printf("%s changed\n", RP(p)); \
73 tab = "\t"; \
74 }
75
76int
77compare(char *name __unused, NODE *s, FTSENT *p)
78{
79 struct timeval tv[2];
80 uint32_t val;
81 int fd, label;
82 off_t len;
83 char *cp;
84 const char *tab = "";
85 char *fflags;
86
87 label = 0;
88 switch(s->type) {
89 case F_BLOCK:
90 if (!S_ISBLK(p->fts_statp->st_mode))
91 goto typeerr;
92 break;
93 case F_CHAR:
94 if (!S_ISCHR(p->fts_statp->st_mode))
95 goto typeerr;
96 break;
97 case F_DIR:
98 if (!S_ISDIR(p->fts_statp->st_mode))
99 goto typeerr;
100 break;
101 case F_FIFO:
102 if (!S_ISFIFO(p->fts_statp->st_mode))
103 goto typeerr;
104 break;
105 case F_FILE:
106 if (!S_ISREG(p->fts_statp->st_mode))
107 goto typeerr;
108 break;
109 case F_LINK:
110 if (!S_ISLNK(p->fts_statp->st_mode))
111 goto typeerr;
112 break;
113 case F_SOCK:
114 if (!S_ISSOCK(p->fts_statp->st_mode)) {
115typeerr: LABEL;
116 (void)printf("\ttype expected %s found %s\n",
117 ftype(s->type), inotype(p->fts_statp->st_mode));
118 return (label);
119 }
120 break;
121 }
122 /* Set the uid/gid first, then set the mode. */
123 if (s->flags & (F_UID | F_UNAME) && s->st_uid != p->fts_statp->st_uid) {
124 LABEL;
125 (void)printf("%suser expected %lu found %lu",
126 tab, (u_long)s->st_uid, (u_long)p->fts_statp->st_uid);
127 if (uflag)
128 if (chown(p->fts_accpath, s->st_uid, -1))
129 (void)printf(" not modified: %s\n",
130 strerror(errno));
131 else
132 (void)printf(" modified\n");
133 else
134 (void)printf("\n");
135 tab = "\t";
136 }
137 if (s->flags & (F_GID | F_GNAME) && s->st_gid != p->fts_statp->st_gid) {
138 LABEL;
139 (void)printf("%sgid expected %lu found %lu",
140 tab, (u_long)s->st_gid, (u_long)p->fts_statp->st_gid);
141 if (uflag)
142 if (chown(p->fts_accpath, -1, s->st_gid))
143 (void)printf(" not modified: %s\n",
144 strerror(errno));
145 else
146 (void)printf(" modified\n");
147 else
148 (void)printf("\n");
149 tab = "\t";
150 }
151 if (s->flags & F_MODE &&
152 !S_ISLNK(p->fts_statp->st_mode) &&
153 s->st_mode != (p->fts_statp->st_mode & MBITS)) {
154 LABEL;
155 (void)printf("%spermissions expected %#o found %#o",
156 tab, s->st_mode, p->fts_statp->st_mode & MBITS);
157 if (uflag)
158 if (chmod(p->fts_accpath, s->st_mode))
159 (void)printf(" not modified: %s\n",
160 strerror(errno));
161 else
162 (void)printf(" modified\n");
163 else
164 (void)printf("\n");
165 tab = "\t";
166 }
167 if (s->flags & F_NLINK && s->type != F_DIR &&
168 s->st_nlink != p->fts_statp->st_nlink) {
169 LABEL;
170 (void)printf("%slink_count expected %u found %u\n",
171 tab, s->st_nlink, p->fts_statp->st_nlink);
172 tab = "\t";
173 }
174 if (s->flags & F_SIZE && s->st_size != p->fts_statp->st_size &&
175 !S_ISDIR(p->fts_statp->st_mode)) {
176 LABEL;
177 (void)printf("%ssize expected %jd found %jd\n", tab,
178 (intmax_t)s->st_size, (intmax_t)p->fts_statp->st_size);
179 tab = "\t";
180 }
181 /*
182 * XXX
183 * Catches nano-second differences, but doesn't display them.
184 */
185 if ((s->flags & F_TIME) &&
186 ((s->st_mtimespec.tv_sec != p->fts_statp->st_mtimespec.tv_sec) ||
187 (s->st_mtimespec.tv_nsec != p->fts_statp->st_mtimespec.tv_nsec))) {
188 LABEL;
189 (void)printf("%smodification time expected %.24s ",
190 tab, ctime(&s->st_mtimespec.tv_sec));
191 (void)printf("found %.24s",
192 ctime(&p->fts_statp->st_mtimespec.tv_sec));
193 if (uflag) {
194 tv[0].tv_sec = s->st_mtimespec.tv_sec;
195 tv[0].tv_usec = s->st_mtimespec.tv_nsec / 1000;
196 tv[1] = tv[0];
197 if (utimes(p->fts_accpath, tv))
198 (void)printf(" not modified: %s\n",
199 strerror(errno));
200 else
201 (void)printf(" modified\n");
202 } else
203 (void)printf("\n");
204 tab = "\t";
205 }
206 if (s->flags & F_CKSUM) {
207 if ((fd = open(p->fts_accpath, O_RDONLY, 0)) < 0) {
208 LABEL;
209 (void)printf("%scksum: %s: %s\n",
210 tab, p->fts_accpath, strerror(errno));
211 tab = "\t";
212 } else if (crc(fd, &val, &len)) {
213 (void)close(fd);
214 LABEL;
215 (void)printf("%scksum: %s: %s\n",
216 tab, p->fts_accpath, strerror(errno));
217 tab = "\t";
218 } else {
219 (void)close(fd);
220 if (s->cksum != val) {
221 LABEL;
222 (void)printf("%scksum expected %lu found %lu\n",
223 tab, s->cksum, (unsigned long)val);
224 tab = "\t";
225 }
226 }
227 }
228 if ((s->flags & F_FLAGS) && s->st_flags != p->fts_statp->st_flags) {
229 LABEL;
230 fflags = flags_to_string(s->st_flags);
231 (void)printf("%sflags expected \"%s\"", tab, fflags);
232 free(fflags);
233
234 fflags = flags_to_string(p->fts_statp->st_flags);
235 (void)printf(" found \"%s\"", fflags);
236 free(fflags);
237
238 if (uflag)
239 if (chflags(p->fts_accpath, s->st_flags))
240 (void)printf(" not modified: %s\n",
241 strerror(errno));
242 else
243 (void)printf(" modified\n");
244 else
245 (void)printf("\n");
246 tab = "\t";
247 }
248#ifdef MD5
249 if (s->flags & F_MD5) {
250 char *new_digest, buf[33];
251
252 new_digest = MD5File(p->fts_accpath, buf);
253 if (!new_digest) {
254 LABEL;
255 printf("%sMD5: %s: %s\n", tab, p->fts_accpath,
256 strerror(errno));
257 tab = "\t";
258 } else if (strcmp(new_digest, s->md5digest)) {
259 LABEL;
260 printf("%sMD5 expected %s found %s\n", tab, s->md5digest,
261 new_digest);
262 tab = "\t";
263 }
264 }
265#endif /* MD5 */
266#ifdef SHA1
267 if (s->flags & F_SHA1) {
268 char *new_digest, buf[41];
269
270 new_digest = SHA1_File(p->fts_accpath, buf);
271 if (!new_digest) {
272 LABEL;
273 printf("%sSHA-1: %s: %s\n", tab, p->fts_accpath,
274 strerror(errno));
275 tab = "\t";
276 } else if (strcmp(new_digest, s->sha1digest)) {
277 LABEL;
278 printf("%sSHA-1 expected %s found %s\n",
279 tab, s->sha1digest, new_digest);
280 tab = "\t";
281 }
282 }
283#endif /* SHA1 */
284#ifdef RMD160
285 if (s->flags & F_RMD160) {
286 char *new_digest, buf[41];
287
288 new_digest = RIPEMD160_File(p->fts_accpath, buf);
289 if (!new_digest) {
290 LABEL;
291 printf("%sRIPEMD160: %s: %s\n", tab,
292 p->fts_accpath, strerror(errno));
293 tab = "\t";
294 } else if (strcmp(new_digest, s->rmd160digest)) {
295 LABEL;
296 printf("%sRIPEMD160 expected %s found %s\n",
297 tab, s->rmd160digest, new_digest);
298 tab = "\t";
299 }
300 }
301#endif /* RMD160 */
302
303 if (s->flags & F_SLINK &&
304 strcmp(cp = rlink(p->fts_accpath), s->slink)) {
305 LABEL;
306 (void)printf("%slink_ref expected %s found %s\n",
307 tab, s->slink, cp);
308 }
309 return (label);
310}
311
312const char *
313inotype(u_int type)
314{
315 switch(type & S_IFMT) {
316 case S_IFBLK:
317 return ("block");
318 case S_IFCHR:
319 return ("char");
320 case S_IFDIR:
321 return ("dir");
322 case S_IFIFO:
323 return ("fifo");
324 case S_IFREG:
325 return ("file");
326 case S_IFLNK:
327 return ("link");
328 case S_IFSOCK:
329 return ("socket");
330 default:
331 return ("unknown");
332 }
333 /* NOTREACHED */
334}
335
67#define INDENTNAMELEN 8
68#define LABEL \
69 if (!label++) { \
70 len = printf("%s changed\n", RP(p)); \
71 tab = "\t"; \
72 }
73
74int
75compare(char *name __unused, NODE *s, FTSENT *p)
76{
77 struct timeval tv[2];
78 uint32_t val;
79 int fd, label;
80 off_t len;
81 char *cp;
82 const char *tab = "";
83 char *fflags;
84
85 label = 0;
86 switch(s->type) {
87 case F_BLOCK:
88 if (!S_ISBLK(p->fts_statp->st_mode))
89 goto typeerr;
90 break;
91 case F_CHAR:
92 if (!S_ISCHR(p->fts_statp->st_mode))
93 goto typeerr;
94 break;
95 case F_DIR:
96 if (!S_ISDIR(p->fts_statp->st_mode))
97 goto typeerr;
98 break;
99 case F_FIFO:
100 if (!S_ISFIFO(p->fts_statp->st_mode))
101 goto typeerr;
102 break;
103 case F_FILE:
104 if (!S_ISREG(p->fts_statp->st_mode))
105 goto typeerr;
106 break;
107 case F_LINK:
108 if (!S_ISLNK(p->fts_statp->st_mode))
109 goto typeerr;
110 break;
111 case F_SOCK:
112 if (!S_ISSOCK(p->fts_statp->st_mode)) {
113typeerr: LABEL;
114 (void)printf("\ttype expected %s found %s\n",
115 ftype(s->type), inotype(p->fts_statp->st_mode));
116 return (label);
117 }
118 break;
119 }
120 /* Set the uid/gid first, then set the mode. */
121 if (s->flags & (F_UID | F_UNAME) && s->st_uid != p->fts_statp->st_uid) {
122 LABEL;
123 (void)printf("%suser expected %lu found %lu",
124 tab, (u_long)s->st_uid, (u_long)p->fts_statp->st_uid);
125 if (uflag)
126 if (chown(p->fts_accpath, s->st_uid, -1))
127 (void)printf(" not modified: %s\n",
128 strerror(errno));
129 else
130 (void)printf(" modified\n");
131 else
132 (void)printf("\n");
133 tab = "\t";
134 }
135 if (s->flags & (F_GID | F_GNAME) && s->st_gid != p->fts_statp->st_gid) {
136 LABEL;
137 (void)printf("%sgid expected %lu found %lu",
138 tab, (u_long)s->st_gid, (u_long)p->fts_statp->st_gid);
139 if (uflag)
140 if (chown(p->fts_accpath, -1, s->st_gid))
141 (void)printf(" not modified: %s\n",
142 strerror(errno));
143 else
144 (void)printf(" modified\n");
145 else
146 (void)printf("\n");
147 tab = "\t";
148 }
149 if (s->flags & F_MODE &&
150 !S_ISLNK(p->fts_statp->st_mode) &&
151 s->st_mode != (p->fts_statp->st_mode & MBITS)) {
152 LABEL;
153 (void)printf("%spermissions expected %#o found %#o",
154 tab, s->st_mode, p->fts_statp->st_mode & MBITS);
155 if (uflag)
156 if (chmod(p->fts_accpath, s->st_mode))
157 (void)printf(" not modified: %s\n",
158 strerror(errno));
159 else
160 (void)printf(" modified\n");
161 else
162 (void)printf("\n");
163 tab = "\t";
164 }
165 if (s->flags & F_NLINK && s->type != F_DIR &&
166 s->st_nlink != p->fts_statp->st_nlink) {
167 LABEL;
168 (void)printf("%slink_count expected %u found %u\n",
169 tab, s->st_nlink, p->fts_statp->st_nlink);
170 tab = "\t";
171 }
172 if (s->flags & F_SIZE && s->st_size != p->fts_statp->st_size &&
173 !S_ISDIR(p->fts_statp->st_mode)) {
174 LABEL;
175 (void)printf("%ssize expected %jd found %jd\n", tab,
176 (intmax_t)s->st_size, (intmax_t)p->fts_statp->st_size);
177 tab = "\t";
178 }
179 /*
180 * XXX
181 * Catches nano-second differences, but doesn't display them.
182 */
183 if ((s->flags & F_TIME) &&
184 ((s->st_mtimespec.tv_sec != p->fts_statp->st_mtimespec.tv_sec) ||
185 (s->st_mtimespec.tv_nsec != p->fts_statp->st_mtimespec.tv_nsec))) {
186 LABEL;
187 (void)printf("%smodification time expected %.24s ",
188 tab, ctime(&s->st_mtimespec.tv_sec));
189 (void)printf("found %.24s",
190 ctime(&p->fts_statp->st_mtimespec.tv_sec));
191 if (uflag) {
192 tv[0].tv_sec = s->st_mtimespec.tv_sec;
193 tv[0].tv_usec = s->st_mtimespec.tv_nsec / 1000;
194 tv[1] = tv[0];
195 if (utimes(p->fts_accpath, tv))
196 (void)printf(" not modified: %s\n",
197 strerror(errno));
198 else
199 (void)printf(" modified\n");
200 } else
201 (void)printf("\n");
202 tab = "\t";
203 }
204 if (s->flags & F_CKSUM) {
205 if ((fd = open(p->fts_accpath, O_RDONLY, 0)) < 0) {
206 LABEL;
207 (void)printf("%scksum: %s: %s\n",
208 tab, p->fts_accpath, strerror(errno));
209 tab = "\t";
210 } else if (crc(fd, &val, &len)) {
211 (void)close(fd);
212 LABEL;
213 (void)printf("%scksum: %s: %s\n",
214 tab, p->fts_accpath, strerror(errno));
215 tab = "\t";
216 } else {
217 (void)close(fd);
218 if (s->cksum != val) {
219 LABEL;
220 (void)printf("%scksum expected %lu found %lu\n",
221 tab, s->cksum, (unsigned long)val);
222 tab = "\t";
223 }
224 }
225 }
226 if ((s->flags & F_FLAGS) && s->st_flags != p->fts_statp->st_flags) {
227 LABEL;
228 fflags = flags_to_string(s->st_flags);
229 (void)printf("%sflags expected \"%s\"", tab, fflags);
230 free(fflags);
231
232 fflags = flags_to_string(p->fts_statp->st_flags);
233 (void)printf(" found \"%s\"", fflags);
234 free(fflags);
235
236 if (uflag)
237 if (chflags(p->fts_accpath, s->st_flags))
238 (void)printf(" not modified: %s\n",
239 strerror(errno));
240 else
241 (void)printf(" modified\n");
242 else
243 (void)printf("\n");
244 tab = "\t";
245 }
246#ifdef MD5
247 if (s->flags & F_MD5) {
248 char *new_digest, buf[33];
249
250 new_digest = MD5File(p->fts_accpath, buf);
251 if (!new_digest) {
252 LABEL;
253 printf("%sMD5: %s: %s\n", tab, p->fts_accpath,
254 strerror(errno));
255 tab = "\t";
256 } else if (strcmp(new_digest, s->md5digest)) {
257 LABEL;
258 printf("%sMD5 expected %s found %s\n", tab, s->md5digest,
259 new_digest);
260 tab = "\t";
261 }
262 }
263#endif /* MD5 */
264#ifdef SHA1
265 if (s->flags & F_SHA1) {
266 char *new_digest, buf[41];
267
268 new_digest = SHA1_File(p->fts_accpath, buf);
269 if (!new_digest) {
270 LABEL;
271 printf("%sSHA-1: %s: %s\n", tab, p->fts_accpath,
272 strerror(errno));
273 tab = "\t";
274 } else if (strcmp(new_digest, s->sha1digest)) {
275 LABEL;
276 printf("%sSHA-1 expected %s found %s\n",
277 tab, s->sha1digest, new_digest);
278 tab = "\t";
279 }
280 }
281#endif /* SHA1 */
282#ifdef RMD160
283 if (s->flags & F_RMD160) {
284 char *new_digest, buf[41];
285
286 new_digest = RIPEMD160_File(p->fts_accpath, buf);
287 if (!new_digest) {
288 LABEL;
289 printf("%sRIPEMD160: %s: %s\n", tab,
290 p->fts_accpath, strerror(errno));
291 tab = "\t";
292 } else if (strcmp(new_digest, s->rmd160digest)) {
293 LABEL;
294 printf("%sRIPEMD160 expected %s found %s\n",
295 tab, s->rmd160digest, new_digest);
296 tab = "\t";
297 }
298 }
299#endif /* RMD160 */
300
301 if (s->flags & F_SLINK &&
302 strcmp(cp = rlink(p->fts_accpath), s->slink)) {
303 LABEL;
304 (void)printf("%slink_ref expected %s found %s\n",
305 tab, s->slink, cp);
306 }
307 return (label);
308}
309
310const char *
311inotype(u_int type)
312{
313 switch(type & S_IFMT) {
314 case S_IFBLK:
315 return ("block");
316 case S_IFCHR:
317 return ("char");
318 case S_IFDIR:
319 return ("dir");
320 case S_IFIFO:
321 return ("fifo");
322 case S_IFREG:
323 return ("file");
324 case S_IFLNK:
325 return ("link");
326 case S_IFSOCK:
327 return ("socket");
328 default:
329 return ("unknown");
330 }
331 /* NOTREACHED */
332}
333
336static const char *
334const char *
337ftype(u_int type)
338{
339 switch(type) {
340 case F_BLOCK:
341 return ("block");
342 case F_CHAR:
343 return ("char");
344 case F_DIR:
345 return ("dir");
346 case F_FIFO:
347 return ("fifo");
348 case F_FILE:
349 return ("file");
350 case F_LINK:
351 return ("link");
352 case F_SOCK:
353 return ("socket");
354 default:
355 return ("unknown");
356 }
357 /* NOTREACHED */
358}
359
360char *
361rlink(char *name)
362{
363 static char lbuf[MAXPATHLEN * 4];
364 int len;
365 char tbuf[MAXPATHLEN];
366
367 if ((len = readlink(name, tbuf, sizeof(tbuf) - 1)) == -1)
368 err(1, "line %d: %s", lineno, name);
369 tbuf[len] = '\0';
370 strvis(lbuf, tbuf, VIS_WHITE | VIS_OCTAL);
371 return (lbuf);
372}
335ftype(u_int type)
336{
337 switch(type) {
338 case F_BLOCK:
339 return ("block");
340 case F_CHAR:
341 return ("char");
342 case F_DIR:
343 return ("dir");
344 case F_FIFO:
345 return ("fifo");
346 case F_FILE:
347 return ("file");
348 case F_LINK:
349 return ("link");
350 case F_SOCK:
351 return ("socket");
352 default:
353 return ("unknown");
354 }
355 /* NOTREACHED */
356}
357
358char *
359rlink(char *name)
360{
361 static char lbuf[MAXPATHLEN * 4];
362 int len;
363 char tbuf[MAXPATHLEN];
364
365 if ((len = readlink(name, tbuf, sizeof(tbuf) - 1)) == -1)
366 err(1, "line %d: %s", lineno, name);
367 tbuf[len] = '\0';
368 strvis(lbuf, tbuf, VIS_WHITE | VIS_OCTAL);
369 return (lbuf);
370}