1 /* file.c - file I/O functions */
3 * GRUB -- GRand Unified Bootloader
4 * Copyright (C) 2002,2006,2007,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/misc.h>
22 #include <grub/file.h>
26 #include <grub/device.h>
27 #include <grub/i18n.h>
29 void (*EXPORT_VAR (grub_grubnet_fini
)) (void);
31 grub_file_filter_t grub_file_filters
[GRUB_FILE_FILTER_MAX
];
33 /* Get the device part of the filename NAME. It is enclosed by parentheses. */
35 grub_file_get_device_name (const char *name
)
39 char *p
= grub_strchr (name
, ')');
44 grub_error (GRUB_ERR_BAD_FILENAME
, N_("missing `%c' symbol"), ')');
48 ret
= (char *) grub_malloc (p
- name
);
52 grub_memcpy (ret
, name
+ 1, p
- name
- 1);
53 ret
[p
- name
- 1] = '\0';
60 /* Support mem:xxx:size:xxx format in chainloader */
61 grub_file_t
grub_memfile_open(const char *name
);
62 #define GRUB_MEMFILE_MEM "mem:"
63 #define GRUB_MEMFILE_SIZE "size:"
65 grub_file_t
grub_memfile_open(const char *name
)
70 file
= (grub_file_t
)grub_zalloc(sizeof(*file
));
76 file
->name
= grub_strdup(name
);
77 file
->data
= (void *)grub_strtoul(name
+ grub_strlen(GRUB_MEMFILE_MEM
), NULL
, 0);
79 size
= grub_strstr(name
, GRUB_MEMFILE_SIZE
);
80 file
->size
= (grub_off_t
)grub_strtoul(size
+ grub_strlen(GRUB_MEMFILE_SIZE
), NULL
, 0);
82 grub_errno
= GRUB_ERR_NONE
;
86 int ventoy_check_file_exist(const char * fmt
, ...)
90 char fullpath
[256] = {0};
93 grub_vsnprintf(fullpath
, 255, fmt
, ap
);
96 file
= grub_file_open(fullpath
, GRUB_FILE_TYPE_NONE
);
104 grub_file_close(file
);
109 typedef struct grub_vlnk
114 struct grub_vlnk
*next
;
117 static grub_vlnk
*g_vlnk_list
;
119 int grub_file_is_vlnk_suffix(const char *name
, int len
)
121 grub_uint32_t suffix
;
125 suffix
= *(grub_uint32_t
*)(name
+ len
- 4);
126 if (grub_strncmp(name
+ len
- 9, ".vlnk.", 6) == 0)
128 /* .iso .wim .img .vhd .efi .dat */
129 if (suffix
== 0x6F73692E || suffix
== 0x6D69772E ||
130 suffix
== 0x676D692E || suffix
== 0x6468762E ||
131 suffix
== 0x6966652E || suffix
== 0x7461642E)
136 else if (len
> 10 && grub_strncmp(name
+ len
- 10, ".vlnk.", 6) == 0)
139 if (suffix
== 0x78646876 || suffix
== 0x796F7476)
149 int grub_file_add_vlnk(const char *src
, const char *dst
)
151 grub_vlnk
*node
= NULL
;
155 node
= grub_zalloc(sizeof(grub_vlnk
));
158 node
->srclen
= (int)grub_strlen(src
);
159 grub_strncpy(node
->src
, src
, sizeof(node
->src
) - 1);
160 grub_strncpy(node
->dst
, dst
, sizeof(node
->dst
) - 1);
162 node
->next
= g_vlnk_list
;
171 const char *grub_file_get_vlnk(const char *name
, int *vlnk
)
174 grub_vlnk
*node
= g_vlnk_list
;
176 len
= grub_strlen(name
);
177 if (!grub_file_is_vlnk_suffix(name
, len
))
184 if (node
->srclen
== len
&& grub_strcmp(name
, node
->src
) == 0)
196 grub_file_open (const char *name
, enum grub_file_type type
)
199 grub_device_t device
= 0;
200 grub_file_t file
= 0, last_file
= 0;
202 const char *file_name
;
203 grub_file_filter_id_t filter
;
205 /* <DESC> : mem:xxx:size:xxx format in chainloader grub_strlen(GRUB_MEMFILE_MEM) */
206 if (grub_strncmp(name
, GRUB_MEMFILE_MEM
, 4) == 0) {
207 return grub_memfile_open(name
);
210 if (g_vlnk_list
&& (type
& GRUB_FILE_TYPE_NO_VLNK
) == 0)
211 name
= grub_file_get_vlnk(name
, &vlnk
);
213 device_name
= grub_file_get_device_name (name
);
217 /* Get the file part of NAME. */
218 file_name
= (name
[0] == '(') ? grub_strchr (name
, ')') : NULL
;
224 device
= grub_device_open (device_name
);
225 grub_free (device_name
);
229 file
= (grub_file_t
) grub_zalloc (sizeof (*file
));
233 file
->device
= device
;
236 /* In case of relative pathnames and non-Unix systems (like Windows)
237 * name of host files may not start with `/'. Blocklists for host files
238 * are meaningless as well (for a start, host disk does not allow any direct
239 * access - it is just a marker). So skip host disk in this case.
241 if (device
->disk
&& file_name
[0] != '/'
242 #if defined(GRUB_UTIL) || defined(GRUB_MACHINE_EMU)
243 && grub_strcmp (device
->disk
->name
, "host")
246 /* This is a block list. */
247 file
->fs
= &grub_fs_blocklist
;
250 file
->fs
= grub_fs_probe (device
);
255 if ((file
->fs
->fs_open
) (file
, file_name
) != GRUB_ERR_NONE
)
258 file
->name
= grub_strdup (name
);
259 grub_errno
= GRUB_ERR_NONE
;
261 for (filter
= 0; file
&& filter
< ARRAY_SIZE (grub_file_filters
);
263 if (grub_file_filters
[filter
])
266 file
= grub_file_filters
[filter
] (file
, type
);
267 if (file
&& file
!= last_file
)
269 file
->name
= grub_strdup (name
);
270 grub_errno
= GRUB_ERR_NONE
;
274 grub_file_close (last_file
);
280 grub_device_close (device
);
282 /* if (net) grub_net_close (net); */
289 grub_disk_read_hook_t grub_file_progress_hook
;
292 grub_file_read (grub_file_t file
, void *buf
, grub_size_t len
)
295 grub_disk_read_hook_t read_hook
;
296 void *read_hook_data
;
298 if (file
->offset
> file
->size
)
300 grub_error (GRUB_ERR_OUT_OF_RANGE
,
301 N_("attempt to read past the end of file"));
308 if (len
> file
->size
- file
->offset
)
309 len
= file
->size
- file
->offset
;
311 /* Prevent an overflow. */
312 if ((grub_ssize_t
) len
< 0)
318 if (grub_strncmp(file
->name
, GRUB_MEMFILE_MEM
, grub_strlen(GRUB_MEMFILE_MEM
)) == 0) {
319 grub_memcpy(buf
, (grub_uint8_t
*)(file
->data
) + file
->offset
, len
);
324 read_hook
= file
->read_hook
;
325 read_hook_data
= file
->read_hook_data
;
326 if (!file
->read_hook
)
328 file
->read_hook
= grub_file_progress_hook
;
329 file
->read_hook_data
= file
;
330 file
->progress_offset
= file
->offset
;
332 res
= (file
->fs
->fs_read
) (file
, buf
, len
);
333 file
->read_hook
= read_hook
;
334 file
->read_hook_data
= read_hook_data
;
342 grub_file_close (grub_file_t file
)
344 if (file
->fs
&& file
->fs
->fs_close
)
345 (file
->fs
->fs_close
) (file
);
348 grub_device_close (file
->device
);
349 grub_free (file
->name
);
355 grub_file_seek (grub_file_t file
, grub_off_t offset
)
359 if (offset
> file
->size
)
361 grub_error (GRUB_ERR_OUT_OF_RANGE
,
362 N_("attempt to seek outside of the file"));
367 file
->offset
= offset
;