blob: 087e4d5e603e96676e70c183908c069ee9d13276 [file] [log] [blame]
The Android Open Source Projectdd7bc332009-03-03 19:32:55 -08001#include <stdio.h>
2#include <stdlib.h>
3#include <string.h>
4#include <sys/types.h>
5#include <dirent.h>
6#include <errno.h>
7
8#include <sys/stat.h>
9#include <unistd.h>
10#include <time.h>
11
12#include <pwd.h>
13#include <grp.h>
14
15#include <linux/kdev_t.h>
16
17// bits for flags argument
Andy McFaddenb33d3412009-04-08 19:25:35 -070018#define LIST_LONG (1 << 0)
19#define LIST_ALL (1 << 1)
20#define LIST_RECURSIVE (1 << 2)
21#define LIST_DIRECTORIES (1 << 3)
The Android Open Source Projectdd7bc332009-03-03 19:32:55 -080022
23// fwd
24static int listpath(const char *name, int flags);
25
26static char mode2kind(unsigned mode)
27{
28 switch(mode & S_IFMT){
29 case S_IFSOCK: return 's';
30 case S_IFLNK: return 'l';
31 case S_IFREG: return '-';
32 case S_IFDIR: return 'd';
33 case S_IFBLK: return 'b';
34 case S_IFCHR: return 'c';
35 case S_IFIFO: return 'p';
36 default: return '?';
37 }
38}
39
40static void mode2str(unsigned mode, char *out)
41{
42 *out++ = mode2kind(mode);
43
44 *out++ = (mode & 0400) ? 'r' : '-';
45 *out++ = (mode & 0200) ? 'w' : '-';
46 if(mode & 04000) {
47 *out++ = (mode & 0100) ? 's' : 'S';
48 } else {
49 *out++ = (mode & 0100) ? 'x' : '-';
50 }
51 *out++ = (mode & 040) ? 'r' : '-';
52 *out++ = (mode & 020) ? 'w' : '-';
53 if(mode & 02000) {
54 *out++ = (mode & 010) ? 's' : 'S';
55 } else {
56 *out++ = (mode & 010) ? 'x' : '-';
57 }
58 *out++ = (mode & 04) ? 'r' : '-';
59 *out++ = (mode & 02) ? 'w' : '-';
60 if(mode & 01000) {
61 *out++ = (mode & 01) ? 't' : 'T';
62 } else {
63 *out++ = (mode & 01) ? 'x' : '-';
64 }
65 *out = 0;
66}
67
68static void user2str(unsigned uid, char *out)
69{
70 struct passwd *pw = getpwuid(uid);
71 if(pw) {
72 strcpy(out, pw->pw_name);
73 } else {
74 sprintf(out, "%d", uid);
75 }
76}
77
78static void group2str(unsigned gid, char *out)
79{
80 struct group *gr = getgrgid(gid);
81 if(gr) {
82 strcpy(out, gr->gr_name);
83 } else {
84 sprintf(out, "%d", gid);
85 }
86}
87
88static int listfile(const char *path, int flags)
89{
90 struct stat s;
91 char date[32];
92 char mode[16];
93 char user[16];
94 char group[16];
95 const char *name;
96
97 /* name is anything after the final '/', or the whole path if none*/
98 name = strrchr(path, '/');
99 if(name == 0) {
100 name = path;
101 } else {
102 name++;
103 }
104
105 if(lstat(path, &s) < 0) {
106 return -1;
107 }
108
109 mode2str(s.st_mode, mode);
110 user2str(s.st_uid, user);
111 group2str(s.st_gid, group);
112
113 strftime(date, 32, "%Y-%m-%d %H:%M", localtime((const time_t*)&s.st_mtime));
114 date[31] = 0;
115
116// 12345678901234567890123456789012345678901234567890123456789012345678901234567890
117// MMMMMMMM UUUUUUUU GGGGGGGGG XXXXXXXX YYYY-MM-DD HH:MM NAME (->LINK)
118
119 switch(s.st_mode & S_IFMT) {
120 case S_IFBLK:
121 case S_IFCHR:
122 printf("%s %-8s %-8s %3d, %3d %s %s\n",
123 mode, user, group,
124 (int) MAJOR(s.st_rdev), (int) MINOR(s.st_rdev),
125 date, name);
126 break;
127 case S_IFREG:
128 printf("%s %-8s %-8s %8d %s %s\n",
129 mode, user, group, (int) s.st_size, date, name);
130 break;
131 case S_IFLNK: {
132 char linkto[256];
133 int len;
134
135 len = readlink(path, linkto, 256);
136 if(len < 0) return -1;
137
138 if(len > 255) {
139 linkto[252] = '.';
140 linkto[253] = '.';
141 linkto[254] = '.';
142 linkto[255] = 0;
143 } else {
144 linkto[len] = 0;
145 }
146
147 printf("%s %-8s %-8s %s %s -> %s\n",
148 mode, user, group, date, name, linkto);
149 break;
150 }
151 default:
152 printf("%s %-8s %-8s %s %s\n",
153 mode, user, group, date, name);
154
155 }
156 return 0;
157}
158
159static int listdir(const char *name, int flags)
160{
161 char tmp[4096];
162 DIR *d;
163 struct dirent *de;
164
165 d = opendir(name);
166 if(d == 0) {
167 fprintf(stderr, "opendir failed, %s\n", strerror(errno));
168 return -1;
169 }
170
171 while((de = readdir(d)) != 0){
172 if (!strcmp(de->d_name, ".") || !strcmp(de->d_name, "..")) continue;
173 if(de->d_name[0] == '.' && (flags & LIST_ALL) == 0) continue;
174 if ((flags & LIST_LONG) != 0) {
175 sprintf(tmp, "%s/%s", name, de->d_name);
176 listfile(tmp, flags);
177 } else {
178 printf("%s\n", de->d_name);
179 }
180 }
181
182 if (flags & LIST_RECURSIVE) {
183 rewinddir(d);
184
185 while ((de = readdir(d)) != 0) {
186 struct stat s;
187 int err;
188
189 if (!strcmp(de->d_name, ".") || !strcmp(de->d_name, ".."))
190 continue;
191 if (de->d_name[0] == '.' && (flags & LIST_ALL) == 0)
192 continue;
193
194 if (!strcmp(name, "/")) sprintf(tmp, "/%s", de->d_name);
195 else sprintf(tmp, "%s/%s", name, de->d_name);
196
197 /*
198 * If the name ends in a '/', use stat() so we treat it like a
199 * directory even if it's a symlink.
200 */
201 if (tmp[strlen(tmp)-1] == '/')
202 err = stat(tmp, &s);
203 else
204 err = lstat(tmp, &s);
205
206 if (err < 0) {
207 perror(tmp);
208 closedir(d);
209 return -1;
210 }
211
212 if (S_ISDIR(s.st_mode)) {
213 printf("\n%s:\n", tmp);
214 listdir(tmp, flags);
215 }
216 }
217 }
218
219 closedir(d);
220 return 0;
221}
222
223static int listpath(const char *name, int flags)
224{
225 struct stat s;
226 int err;
227
228 /*
229 * If the name ends in a '/', use stat() so we treat it like a
230 * directory even if it's a symlink.
231 */
232 if (name[strlen(name)-1] == '/')
233 err = stat(name, &s);
234 else
235 err = lstat(name, &s);
236
237 if (err < 0) {
238 perror(name);
239 return -1;
240 }
241
Andy McFaddenb33d3412009-04-08 19:25:35 -0700242 if ((flags & LIST_DIRECTORIES) == 0 && S_ISDIR(s.st_mode)) {
The Android Open Source Projectdd7bc332009-03-03 19:32:55 -0800243 if (flags & LIST_RECURSIVE)
244 printf("\n%s:\n", name);
245 return listdir(name, flags);
246 } else {
247 if ((flags & LIST_LONG) != 0) {
248 /* yeah this calls stat() again*/
249 return listfile(name, flags);
250 } else {
251 printf("%s\n", name);
252 return 0;
253 }
254 }
255}
256
257int ls_main(int argc, char **argv)
258{
259 int flags = 0;
260 int listed = 0;
261
262 if(argc > 1) {
263 int i;
264 int err = 0;
265
266 for (i = 1; i < argc; i++) {
267 if(!strcmp(argv[i], "-l")) {
268 flags |= LIST_LONG;
269 } else if (!strcmp(argv[i], "-a")) {
270 flags |= LIST_ALL;
271 } else if (!strcmp(argv[i], "-R")) {
272 flags |= LIST_RECURSIVE;
Andy McFaddenb33d3412009-04-08 19:25:35 -0700273 } else if (!strcmp(argv[i], "-d")) {
274 flags |= LIST_DIRECTORIES;
The Android Open Source Projectdd7bc332009-03-03 19:32:55 -0800275 } else {
276 listed++;
277 if(listpath(argv[i], flags) != 0) {
278 err = EXIT_FAILURE;
279 }
280 }
281 }
282
283 if (listed > 0) return err;
284 }
285
286 // list working directory if no files or directories were specified
287 return listpath(".", flags);
288}