1 /*
2 * lsattr.c - List file attributes on an ext2 file system
3 *
4 * Copyright (C) 1993, 1994 Remy Card <card@masi.ibp.fr>
5 * Laboratoire MASI, Institut Blaise Pascal
6 * Universite Pierre et Marie Curie (Paris VI)
7 *
8 * This file can be redistributed under the terms of the GNU General
9 * Public License
10 */
11
12 /*
13 * History:
14 * 93/10/30 - Creation
15 * 93/11/13 - Replace stat() calls by lstat() to avoid loops
16 * 94/02/27 - Integrated in Ted's distribution
17 * 98/12/29 - Display version info only when -V specified (G M Sipe)
18 */
19
20 #define _LARGEFILE64_SOURCE
21
22 #include <sys/types.h>
23 #include <dirent.h>
24 #ifdef HAVE_ERRNO_H
25 #include <errno.h>
26 #endif
27 #include <fcntl.h>
28 #ifdef HAVE_GETOPT_H
29 #include <getopt.h>
30 #else
31 extern int optind;
32 extern char *optarg;
33 #endif
34 #include <stdio.h>
35 #include <unistd.h>
36 #include <stdlib.h>
37 #include <string.h>
38 #include <sys/param.h>
39 #include <sys/stat.h>
40
41 #include "ext2fs/ext2_fs.h"
42 #include "et/com_err.h"
43 #include "e2p/e2p.h"
44
45 #include "../version.h"
46 #include "nls-enable.h"
47
48 #ifdef __GNUC__
49 #define EXT2FS_ATTR(x) __attribute__(x)
50 #else
51 #define EXT2FS_ATTR(x)
52 #endif
53
54 static const char * program_name = "lsattr";
55
56 static int all;
57 static int dirs_opt;
58 static unsigned pf_options;
59 static int recursive;
60 static int verbose;
61 static int generation_opt;
62
63 #ifdef _LFS64_LARGEFILE
64 #define LSTAT lstat64
65 #define STRUCT_STAT struct stat64
66 #else
67 #define LSTAT lstat
68 #define STRUCT_STAT struct stat
69 #endif
70
usage(void)71 static void usage(void)
72 {
73 fprintf(stderr, _("Usage: %s [-RVadlv] [files...]\n"), program_name);
74 exit(1);
75 }
76
list_attributes(const char * name)77 static int list_attributes (const char * name)
78 {
79 unsigned long flags;
80 unsigned long generation;
81
82 if (fgetflags (name, &flags) == -1) {
83 com_err (program_name, errno, _("While reading flags on %s"),
84 name);
85 return -1;
86 }
87 if (generation_opt) {
88 if (fgetversion (name, &generation) == -1) {
89 com_err (program_name, errno,
90 _("While reading version on %s"),
91 name);
92 return -1;
93 }
94 printf ("%5lu ", generation);
95 }
96 if (pf_options & PFOPT_LONG) {
97 printf("%-28s ", name);
98 print_flags(stdout, flags, pf_options);
99 fputc('\n', stdout);
100 } else {
101 print_flags(stdout, flags, pf_options);
102 printf(" %s\n", name);
103 }
104 return 0;
105 }
106
107 static int lsattr_dir_proc (const char *, struct dirent *, void *);
108
lsattr_args(const char * name)109 static int lsattr_args (const char * name)
110 {
111 STRUCT_STAT st;
112 int retval = 0;
113
114 if (LSTAT (name, &st) == -1) {
115 com_err (program_name, errno, _("while trying to stat %s"),
116 name);
117 retval = -1;
118 } else {
119 if (S_ISDIR(st.st_mode) && !dirs_opt)
120 retval = iterate_on_dir (name, lsattr_dir_proc, NULL);
121 else
122 retval = list_attributes (name);
123 }
124 return retval;
125 }
126
lsattr_dir_proc(const char * dir_name,struct dirent * de,void * private EXT2FS_ATTR ((unused)))127 static int lsattr_dir_proc (const char * dir_name, struct dirent * de,
128 void * private EXT2FS_ATTR((unused)))
129 {
130 STRUCT_STAT st;
131 char *path;
132 int dir_len = strlen(dir_name);
133
134 path = malloc(dir_len + strlen (de->d_name) + 2);
135
136 if (dir_len && dir_name[dir_len-1] == '/')
137 sprintf (path, "%s%s", dir_name, de->d_name);
138 else
139 sprintf (path, "%s/%s", dir_name, de->d_name);
140 if (LSTAT (path, &st) == -1)
141 perror (path);
142 else {
143 if (de->d_name[0] != '.' || all) {
144 list_attributes (path);
145 if (S_ISDIR(st.st_mode) && recursive &&
146 strcmp(de->d_name, ".") &&
147 strcmp(de->d_name, "..")) {
148 printf ("\n%s:\n", path);
149 iterate_on_dir (path, lsattr_dir_proc, NULL);
150 printf ("\n");
151 }
152 }
153 }
154 free(path);
155 return 0;
156 }
157
main(int argc,char ** argv)158 int main (int argc, char ** argv)
159 {
160 int c;
161 int i;
162 int err, retval = 0;
163
164 #ifdef ENABLE_NLS
165 setlocale(LC_MESSAGES, "");
166 setlocale(LC_CTYPE, "");
167 bindtextdomain(NLS_CAT_NAME, LOCALEDIR);
168 textdomain(NLS_CAT_NAME);
169 set_com_err_gettext(gettext);
170 #endif
171 if (argc && *argv)
172 program_name = *argv;
173 while ((c = getopt (argc, argv, "RVadlv")) != EOF)
174 switch (c)
175 {
176 case 'R':
177 recursive = 1;
178 break;
179 case 'V':
180 verbose = 1;
181 break;
182 case 'a':
183 all = 1;
184 break;
185 case 'd':
186 dirs_opt = 1;
187 break;
188 case 'l':
189 pf_options = PFOPT_LONG;
190 break;
191 case 'v':
192 generation_opt = 1;
193 break;
194 default:
195 usage();
196 }
197
198 if (verbose)
199 fprintf (stderr, "lsattr %s (%s)\n",
200 E2FSPROGS_VERSION, E2FSPROGS_DATE);
201 if (optind > argc - 1) {
202 if (lsattr_args (".") == -1)
203 retval = 1;
204 } else {
205 for (i = optind; i < argc; i++) {
206 err = lsattr_args (argv[i]);
207 if (err)
208 retval = 1;
209 }
210 }
211 exit(retval);
212 }
213