1 /* ls.c - command to list files and devices */
3 * GRUB -- GRand Unified Bootloader
4 * Copyright (C) 2003,2005,2007,2008,2009 Free Software Foundation, Inc.
6 * GRUB is free software: you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation, either version 3 of the License, or
9 * (at your option) any later version.
11 * GRUB is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 * You should have received a copy of the GNU General Public License
17 * along with GRUB. If not, see <http://www.gnu.org/licenses/>.
20 #include <grub/types.h>
21 #include <grub/misc.h>
25 #include <grub/disk.h>
26 #include <grub/device.h>
27 #include <grub/term.h>
28 #include <grub/partition.h>
29 #include <grub/file.h>
30 #include <grub/normal.h>
31 #include <grub/extcmd.h>
32 #include <grub/datetime.h>
33 #include <grub/i18n.h>
36 static const struct grub_arg_option options
[] =
38 {"long", 'l', 0, N_("Show a long list with more detailed information."), 0, 0},
39 {"human-readable", 'h', 0, N_("Print sizes in a human readable format."), 0, 0},
40 {"all", 'a', 0, N_("List all files."), 0, 0},
44 static const char grub_human_sizes
[] = {' ', 'K', 'M', 'G', 'T'};
47 grub_ls_list_devices (int longlist
)
49 grub_net_app_level_t proto
;
51 auto int grub_ls_print_devices (const char *name
);
52 int grub_ls_print_devices (const char *name
)
55 grub_normal_print_device_info (name
);
57 grub_printf ("(%s) ", name
);
62 grub_device_iterate (grub_ls_print_devices
);
65 grub_puts_ (N_ ("Network protocols:\n"));
67 FOR_NET_APP_LEVEL (proto
)
69 grub_printf ("%s ", proto
->name
);
80 grub_ls_list_files (char *dirname
, int longlist
, int all
, int human
)
87 auto int print_files (const char *filename
,
88 const struct grub_dirhook_info
*info
);
89 auto int print_files_long (const char *filename
,
90 const struct grub_dirhook_info
*info
);
92 int print_files (const char *filename
, const struct grub_dirhook_info
*info
)
94 if (all
|| filename
[0] != '.')
95 grub_printf ("%s%s ", filename
, info
->dir
? "/" : "");
100 int print_files_long (const char *filename
,
101 const struct grub_dirhook_info
*info
)
103 if ((! all
) && (filename
[0] == '.'))
111 if (dirname
[grub_strlen (dirname
) - 1] == '/')
112 pathname
= grub_xasprintf ("%s%s", dirname
, filename
);
114 pathname
= grub_xasprintf ("%s/%s", dirname
, filename
);
119 /* XXX: For ext2fs symlinks are detected as files while they
120 should be reported as directories. */
121 file
= grub_file_open (pathname
);
125 grub_free (pathname
);
130 grub_printf ("%-12llu", (unsigned long long) file
->size
);
133 grub_uint64_t fsize
= file
->size
* 100ULL;
134 int fsz
= file
->size
;
140 fsize
= (fsize
+ 512) / 1024;
147 grub_uint32_t whole
, fraction
;
149 whole
= grub_divmod64 (fsize
, 100, &fraction
);
150 grub_snprintf (buf
, sizeof (buf
),
151 "%u.%02u%c", whole
, fraction
,
152 grub_human_sizes
[units
]);
153 grub_printf ("%-12s", buf
);
156 grub_printf ("%-12llu", (unsigned long long) file
->size
);
159 grub_file_close (file
);
160 grub_free (pathname
);
163 grub_printf ("%-12s", "DIR");
167 struct grub_datetime datetime
;
168 grub_unixtime2datetime (info
->mtime
, &datetime
);
170 grub_printf (" %d-%02d-%02d %02d:%02d:%02d %-11s ",
171 datetime
.year
, datetime
.month
, datetime
.day
,
172 datetime
.hour
, datetime
.minute
,
174 grub_get_weekday_name (&datetime
));
176 grub_printf (" %04d%02d%02d%02d%02d%02d ",
177 datetime
.year
, datetime
.month
,
178 datetime
.day
, datetime
.hour
,
179 datetime
.minute
, datetime
.second
);
181 grub_printf ("%s%s\n", filename
, info
->dir
? "/" : "");
186 device_name
= grub_file_get_device_name (dirname
);
187 dev
= grub_device_open (device_name
);
191 fs
= grub_fs_probe (dev
);
192 path
= grub_strchr (dirname
, ')');
198 if (! path
&& ! device_name
)
200 grub_error (GRUB_ERR_BAD_ARGUMENT
, "invalid argument");
206 if (grub_errno
== GRUB_ERR_UNKNOWN_FS
)
207 grub_errno
= GRUB_ERR_NONE
;
209 grub_normal_print_device_info (device_name
);
214 (fs
->dir
) (dev
, path
, print_files_long
);
216 (fs
->dir
) (dev
, path
, print_files
);
218 if (grub_errno
== GRUB_ERR_BAD_FILE_TYPE
219 && path
[grub_strlen (path
) - 1] != '/')
221 /* PATH might be a regular file. */
224 struct grub_dirhook_info info
;
227 file
= grub_file_open (dirname
);
231 grub_file_close (file
);
233 p
= grub_strrchr (dirname
, '/') + 1;
234 dirname
= grub_strndup (dirname
, p
- dirname
);
239 grub_memset (&info
, 0, sizeof (info
));
241 print_files_long (p
, &info
);
243 print_files (p
, &info
);
248 if (grub_errno
== GRUB_ERR_NONE
)
256 grub_device_close (dev
);
258 grub_free (device_name
);
264 grub_cmd_ls (grub_extcmd_t cmd
, int argc
, char **args
)
266 struct grub_arg_list
*state
= cmd
->state
;
269 grub_ls_list_devices (state
[0].set
);
271 grub_ls_list_files (args
[0], state
[0].set
, state
[2].set
,
277 static grub_extcmd_t cmd
;
281 cmd
= grub_register_extcmd ("ls", grub_cmd_ls
, GRUB_COMMAND_FLAG_BOTH
,
282 N_("[-l|-h|-a] [FILE]"),
283 N_("List devices and files."), options
);
288 grub_unregister_extcmd (cmd
);