Sync usage with man page.
[netbsd-mini2440.git] / sys / arch / ia64 / stand / efi / libefi / efifs_ls.c
blobbc070199033a70ea9f8446769420a42ce262ae7f
1 /* $NetBSD: efifs_ls.c,v 1.1 2006/04/07 14:21:32 cherry Exp $ */
3 /*
4 * Copyright (c) 1993
5 * The Regents of the University of California. All rights reserved.
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
9 * are met:
10 * 1. Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer.
12 * 2. Redistributions in binary form must reproduce the above copyright
13 * notice, this list of conditions and the following disclaimer in the
14 * documentation and/or other materials provided with the distribution.
15 * 3. Neither the name of the University nor the names of its contributors
16 * may be used to endorse or promote products derived from this software
17 * without specific prior written permission.
19 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
20 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
21 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
22 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
23 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
24 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
25 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
26 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
27 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
28 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
29 * SUCH DAMAGE.
33 * Copyright (c) 1996
34 * Matthias Drochner. All rights reserved.
36 * Redistribution and use in source and binary forms, with or without
37 * modification, are permitted provided that the following conditions
38 * are met:
39 * 1. Redistributions of source code must retain the above copyright
40 * notice, this list of conditions and the following disclaimer.
41 * 2. Redistributions in binary form must reproduce the above copyright
42 * notice, this list of conditions and the following disclaimer in the
43 * documentation and/or other materials provided with the distribution.
45 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
46 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
47 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
48 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
49 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
50 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
51 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
52 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
53 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
54 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
57 /* Based on libsa/ufs_ls.c */
59 #include <dirent.h>
60 #include <sys/param.h>
61 #include <sys/stat.h>
63 #include <lib/libkern/libkern.h>
65 #include "stand.h"
67 #include <efi.h>
68 #include <efilib.h>
70 #include "efifs.h"
72 #define NELEM(x) (sizeof (x) / sizeof(*x))
74 typedef struct entry_t entry_t;
75 struct entry_t {
76 entry_t *e_next;
77 ino_t e_ino;
78 uint8_t e_type;
79 char e_name[1];
82 static const char *const typestr[] = {
83 "unknown",
84 "FIFO",
85 "CHR",
87 "DIR",
89 "BLK",
91 "REG",
93 "LNK",
95 "SOCK",
97 "WHT"
100 static int
101 fn_match(const char *fname, const char *pattern)
103 char fc, pc;
105 do {
106 fc = *fname++;
107 pc = *pattern++;
108 if (!fc && !pc)
109 return 1;
110 if (pc == '?' && fc)
111 pc = fc;
112 } while (fc == pc);
114 if (pc != '*')
115 return 0;
116 /* Too hard (and unnecessary really) too check for "*?name" etc....
117 "**" will look for a '*' and "*?" a '?' */
118 pc = *pattern++;
119 if (!pc)
120 return 1;
121 while ((fname = strchr(fname, pc)))
122 if (fn_match(++fname, pattern))
123 return 1;
124 return 0;
127 void
128 efifs_ls(const char *path)
130 int fd;
131 struct stat sb;
132 size_t size;
133 char dirbuf[DIRBLKSIZ];
134 const char *fname = 0;
135 char *p;
136 entry_t *names = 0, *n, **np;
138 if ((fd = open(path, 0)) < 0
139 || fstat(fd, &sb) < 0
140 || (sb.st_mode & S_IFMT) != S_IFDIR) {
141 /* Path supplied isn't a directory, open parent
142 directory and list matching files. */
143 if (fd >= 0)
144 close(fd);
145 fname = strrchr(path, '/');
146 if (fname) {
147 size = fname - path;
148 p = alloc(size + 1);
149 if (!p)
150 goto out;
151 memcpy(p, path, size);
152 p[size] = 0;
153 fd = open(p, 0);
154 free(p, size + 1);
155 } else {
156 fd = open("", 0);
157 fname = path;
160 if (fd < 0) {
161 printf("ls: %s\n", strerror(errno));
162 return;
164 if (fstat(fd, &sb) < 0) {
165 printf("stat: %s\n", strerror(errno));
166 goto out;
168 if ((sb.st_mode & S_IFMT) != S_IFDIR) {
169 printf("%s: %s\n", path, strerror(ENOTDIR));
170 goto out;
174 while ((size = read(fd, dirbuf, DIRBLKSIZ)) == DIRBLKSIZ) {
175 struct dirent *dp, *edp;
177 dp = (struct dirent *) dirbuf;
178 edp = (struct dirent *) (dirbuf + size);
180 for (; dp < edp; dp = (void *)((char *)dp + dp->d_reclen)) {
181 const char *t;
182 if (dp->d_ino == 0)
183 continue;
185 if (dp->d_type >= NELEM(typestr) ||
186 !(t = typestr[dp->d_type])) {
188 * This does not handle "old"
189 * filesystems properly. On little
190 * endian machines, we get a bogus
191 * type name if the namlen matches a
192 * valid type identifier. We could
193 * check if we read namlen "0" and
194 * handle this case specially, if
195 * there were a pressing need...
197 printf("bad dir entry\n");
198 goto out;
200 if (fname && !fn_match(dp->d_name, fname))
201 continue;
202 n = alloc(sizeof *n + strlen(dp->d_name));
203 if (!n) {
204 printf("%d: %s (%s)\n",
205 dp->d_ino, dp->d_name, t);
206 continue;
208 n->e_ino = dp->d_ino;
209 n->e_type = dp->d_type;
210 strcpy(n->e_name, dp->d_name);
211 for (np = &names; *np; np = &(*np)->e_next) {
212 if (strcmp(n->e_name, (*np)->e_name) < 0)
213 break;
215 n->e_next = *np;
216 *np = n;
220 if (names) {
221 do {
222 n = names;
223 printf("%d: %s (%s)\n",
224 n->e_ino, n->e_name, typestr[n->e_type]);
225 names = n->e_next;
226 free(n, 0);
227 } while (names);
228 } else {
229 printf( "%s not found\n", path );
231 out:
232 close(fd);