1 /* ls.c - command to list files and devices */
3 * GRUB -- GRand Unified Bootloader
4 * Copyright (C) 2003,2005,2007,2008 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>
34 static const struct grub_arg_option options
[] =
36 {"long", 'l', 0, "show a long list with more detailed information", 0, 0},
37 {"human-readable", 'h', 0, "print sizes in a human readable format", 0, 0},
38 {"all", 'a', 0, "list all files", 0, 0},
42 static const char grub_human_sizes
[] = {' ', 'K', 'M', 'G', 'T'};
45 grub_ls_list_devices (int longlist
)
47 auto int grub_ls_print_devices (const char *name
);
48 int grub_ls_print_devices (const char *name
)
51 grub_normal_print_device_info (name
);
53 grub_printf ("(%s) ", name
);
58 grub_device_iterate (grub_ls_print_devices
);
66 grub_ls_list_files (char *dirname
, int longlist
, int all
, int human
)
73 auto int print_files (const char *filename
,
74 const struct grub_dirhook_info
*info
);
75 auto int print_files_long (const char *filename
,
76 const struct grub_dirhook_info
*info
);
78 int print_files (const char *filename
, const struct grub_dirhook_info
*info
)
80 if (all
|| filename
[0] != '.')
81 grub_printf ("%s%s ", filename
, info
->dir
? "/" : "");
86 int print_files_long (const char *filename
,
87 const struct grub_dirhook_info
*info
)
89 char pathname
[grub_strlen (dirname
) + grub_strlen (filename
) + 1];
91 if ((! all
) && (filename
[0] == '.'))
98 if (dirname
[grub_strlen (dirname
) - 1] == '/')
99 grub_sprintf (pathname
, "%s%s", dirname
, filename
);
101 grub_sprintf (pathname
, "%s/%s", dirname
, filename
);
103 /* XXX: For ext2fs symlinks are detected as files while they
104 should be reported as directories. */
105 file
= grub_file_open (pathname
);
113 grub_printf ("%-12llu", (unsigned long long) file
->size
);
116 grub_uint64_t fsize
= file
->size
* 100ULL;
117 int fsz
= file
->size
;
123 fsize
= (fsize
+ 512) / 1024;
130 grub_uint32_t whole
, fraction
;
132 whole
= grub_divmod64 (fsize
, 100, &fraction
);
133 grub_sprintf (buf
, "%u.%02u%c", whole
, fraction
,
134 grub_human_sizes
[units
]);
135 grub_printf ("%-12s", buf
);
138 grub_printf ("%-12llu", (unsigned long long) file
->size
);
141 grub_file_close (file
);
144 grub_printf ("%-12s", "DIR");
148 struct grub_datetime datetime
;
149 grub_unixtime2datetime (info
->mtime
, &datetime
);
151 grub_printf (" %d-%02d-%02d %02d:%02d:%02d %-11s ",
152 datetime
.year
, datetime
.month
, datetime
.day
,
153 datetime
.hour
, datetime
.minute
,
155 grub_get_weekday_name (&datetime
));
157 grub_printf (" %04d%02d%02d%02d%02d%02d ",
158 datetime
.year
, datetime
.month
,
159 datetime
.day
, datetime
.hour
,
160 datetime
.minute
, datetime
.second
);
162 grub_printf ("%s%s\n", filename
, info
->dir
? "/" : "");
167 device_name
= grub_file_get_device_name (dirname
);
168 dev
= grub_device_open (device_name
);
172 fs
= grub_fs_probe (dev
);
173 path
= grub_strchr (dirname
, ')');
179 if (! path
&& ! device_name
)
181 grub_error (GRUB_ERR_BAD_ARGUMENT
, "invalid argument");
187 if (grub_errno
== GRUB_ERR_UNKNOWN_FS
)
188 grub_errno
= GRUB_ERR_NONE
;
190 grub_normal_print_device_info (device_name
);
195 (fs
->dir
) (dev
, path
, print_files_long
);
197 (fs
->dir
) (dev
, path
, print_files
);
199 if (grub_errno
== GRUB_ERR_BAD_FILE_TYPE
200 && path
[grub_strlen (path
) - 1] != '/')
202 /* PATH might be a regular file. */
205 struct grub_dirhook_info info
;
208 file
= grub_file_open (dirname
);
212 grub_file_close (file
);
214 p
= grub_strrchr (dirname
, '/') + 1;
215 dirname
= grub_strndup (dirname
, p
- dirname
);
220 grub_memset (&info
, 0, sizeof (info
));
222 print_files_long (p
, &info
);
224 print_files (p
, &info
);
229 if (grub_errno
== GRUB_ERR_NONE
)
237 grub_device_close (dev
);
239 grub_free (device_name
);
245 grub_cmd_ls (grub_extcmd_t cmd
, int argc
, char **args
)
247 struct grub_arg_list
*state
= cmd
->state
;
250 grub_ls_list_devices (state
[0].set
);
252 grub_ls_list_files (args
[0], state
[0].set
, state
[2].set
,
258 static grub_extcmd_t cmd
;
262 cmd
= grub_register_extcmd ("ls", grub_cmd_ls
, GRUB_COMMAND_FLAG_BOTH
,
263 "ls [-l|-h|-a] [FILE]",
264 "List devices and files.", options
);
269 grub_unregister_extcmd (cmd
);