]> glassweightruler.freedombox.rocks Git - Ventoy.git/blob - GRUB2/MOD_SRC/grub-2.04/grub-core/ventoy/ventoy_windows.c
5d08fa68c0b91b00b9dbfe375ec03f34754162f6
[Ventoy.git] / GRUB2 / MOD_SRC / grub-2.04 / grub-core / ventoy / ventoy_windows.c
1 /******************************************************************************
2 * ventoy_windows.c
3 *
4 * Copyright (c) 2020, longpanda <admin@ventoy.net>
5 *
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License as
8 * published by the Free Software Foundation; either version 3 of the
9 * License, or (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful, but
12 * WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, see <http://www.gnu.org/licenses/>.
18 *
19 */
20
21 #include <grub/types.h>
22 #include <grub/misc.h>
23 #include <grub/mm.h>
24 #include <grub/err.h>
25 #include <grub/dl.h>
26 #include <grub/disk.h>
27 #include <grub/device.h>
28 #include <grub/term.h>
29 #include <grub/partition.h>
30 #include <grub/file.h>
31 #include <grub/normal.h>
32 #include <grub/extcmd.h>
33 #include <grub/datetime.h>
34 #include <grub/i18n.h>
35 #include <grub/net.h>
36 #include <grub/time.h>
37 #include <grub/crypto.h>
38 #include <grub/ventoy.h>
39 #include "ventoy_def.h"
40
41 GRUB_MOD_LICENSE ("GPLv3+");
42
43 static int g_iso_fs_type = 0;
44 static int g_wim_total_patch_count = 0;
45 static int g_wim_valid_patch_count = 0;
46 static wim_patch *g_wim_patch_head = NULL;
47
48 static grub_uint64_t g_suppress_wincd_override_offset = 0;
49 static grub_uint32_t g_suppress_wincd_override_data = 0;
50
51 grub_uint8_t g_temp_buf[512];
52
53 grub_ssize_t lzx_decompress ( const void *data, grub_size_t len, void *buf );
54 grub_ssize_t xca_decompress ( const void *data, grub_size_t len, void *buf );
55
56 static wim_patch *ventoy_find_wim_patch(const char *path)
57 {
58 int len = (int)grub_strlen(path);
59 wim_patch *node = g_wim_patch_head;
60
61 while (node)
62 {
63 if (len == node->pathlen && 0 == grub_strcmp(path, node->path))
64 {
65 return node;
66 }
67 node = node->next;
68 }
69
70 return NULL;
71 }
72
73 static int ventoy_collect_wim_patch(const char *bcdfile)
74 {
75 int i, j, k;
76 int rc = 1;
77 grub_uint64_t magic;
78 grub_file_t file = NULL;
79 char *buf = NULL;
80 wim_patch *node = NULL;
81 char c;
82 grub_uint8_t byte;
83 char valid;
84 char path[256];
85
86 g_ventoy_case_insensitive = 1;
87 file = grub_file_open(bcdfile, VENTOY_FILE_TYPE);
88 g_ventoy_case_insensitive = 0;
89 if (!file)
90 {
91 debug("Failed to open file %s\n", bcdfile);
92 grub_errno = 0;
93 goto end;
94 }
95
96 buf = grub_malloc(file->size + 8);
97 if (!buf)
98 {
99 goto end;
100 }
101
102 grub_file_read(file, buf, file->size);
103
104 for (i = 0; i < (int)file->size - 8; i++)
105 {
106 if (buf[i + 8] != 0)
107 {
108 continue;
109 }
110
111 magic = *(grub_uint64_t *)(buf + i);
112
113 /* .wim .WIM .Wim */
114 if ((magic == 0x006D00690077002EULL) ||
115 (magic == 0x004D00490057002EULL) ||
116 (magic == 0x006D00690057002EULL))
117 {
118 for (j = i; j > 0; j-= 2)
119 {
120 if (*(grub_uint16_t *)(buf + j) == 0)
121 {
122 break;
123 }
124 }
125
126 if (j > 0)
127 {
128 byte = (grub_uint8_t)(*(grub_uint16_t *)(buf + j + 2));
129 if (byte != '/' && byte != '\\')
130 {
131 continue;
132 }
133
134 valid = 1;
135 for (k = 0, j += 2; k < (int)sizeof(path) - 1 && j < i + 8; j += 2)
136 {
137 byte = (grub_uint8_t)(*(grub_uint16_t *)(buf + j));
138 c = (char)byte;
139 if (byte > '~' || byte < ' ') /* not printable */
140 {
141 valid = 0;
142 break;
143 }
144 else if (c == '\\')
145 {
146 c = '/';
147 }
148
149 path[k++] = c;
150 }
151 path[k++] = 0;
152
153 debug("@@@@ Find wim flag:<%s>\n", path);
154
155 if (0 == valid)
156 {
157 debug("Invalid wim file %d\n", k);
158 }
159 else if (NULL == ventoy_find_wim_patch(path))
160 {
161 node = grub_zalloc(sizeof(wim_patch));
162 if (node)
163 {
164 node->pathlen = grub_snprintf(node->path, sizeof(node->path), "%s", path);
165
166 debug("add patch <%s>\n", path);
167
168 if (g_wim_patch_head)
169 {
170 node->next = g_wim_patch_head;
171 }
172 g_wim_patch_head = node;
173
174 g_wim_total_patch_count++;
175 }
176 }
177 else
178 {
179 debug("wim <%s> already exist\n", path);
180 }
181 }
182 }
183 }
184
185 end:
186 check_free(file, grub_file_close);
187 grub_check_free(buf);
188 return rc;
189 }
190
191 grub_err_t ventoy_cmd_wim_patch_count(grub_extcmd_context_t ctxt, int argc, char **args)
192 {
193 char buf[32];
194
195 (void)ctxt;
196 (void)argc;
197 (void)args;
198
199 if (argc == 1)
200 {
201 grub_snprintf(buf, sizeof(buf), "%d", g_wim_total_patch_count);
202 ventoy_set_env(args[0], buf);
203 }
204
205 return 0;
206 }
207
208 grub_err_t ventoy_cmd_collect_wim_patch(grub_extcmd_context_t ctxt, int argc, char **args)
209 {
210 wim_patch *node = NULL;
211
212 (void)ctxt;
213 (void)argc;
214 (void)args;
215
216 if (argc != 2)
217 {
218 return 1;
219 }
220
221 debug("ventoy_cmd_collect_wim_patch %s %s\n", args[0], args[1]);
222
223 if (grub_strcmp(args[0], "bcd") == 0)
224 {
225 ventoy_collect_wim_patch(args[1]);
226 return 0;
227 }
228
229 if (NULL == ventoy_find_wim_patch(args[1]))
230 {
231 node = grub_zalloc(sizeof(wim_patch));
232 if (node)
233 {
234 node->pathlen = grub_snprintf(node->path, sizeof(node->path), "%s", args[1]);
235
236 debug("add patch <%s>\n", args[1]);
237
238 if (g_wim_patch_head)
239 {
240 node->next = g_wim_patch_head;
241 }
242 g_wim_patch_head = node;
243
244 g_wim_total_patch_count++;
245 }
246 }
247
248 return 0;
249 }
250
251
252 grub_err_t ventoy_cmd_dump_wim_patch(grub_extcmd_context_t ctxt, int argc, char **args)
253 {
254 int i = 0;
255 wim_patch *node = NULL;
256
257 (void)ctxt;
258 (void)argc;
259 (void)args;
260
261 for (node = g_wim_patch_head; node; node = node->next)
262 {
263 grub_printf("%d %s [%s]\n", i++, node->path, node->valid ? "SUCCESS" : "FAIL");
264 }
265
266 return 0;
267 }
268
269
270 static int wim_name_cmp(const char *search, grub_uint16_t *name, grub_uint16_t namelen)
271 {
272 char c1 = vtoy_to_upper(*search);
273 char c2 = vtoy_to_upper(*name);
274
275 while (namelen > 0 && (c1 == c2))
276 {
277 search++;
278 name++;
279 namelen--;
280
281 c1 = vtoy_to_upper(*search);
282 c2 = vtoy_to_upper(*name);
283 }
284
285 if (namelen == 0 && *search == 0)
286 {
287 return 0;
288 }
289
290 return 1;
291 }
292
293 static int ventoy_is_pe64(grub_uint8_t *buffer)
294 {
295 grub_uint32_t pe_off;
296
297 if (buffer[0] != 'M' || buffer[1] != 'Z')
298 {
299 return 0;
300 }
301
302 pe_off = *(grub_uint32_t *)(buffer + 60);
303
304 if (buffer[pe_off] != 'P' || buffer[pe_off + 1] != 'E')
305 {
306 return 0;
307 }
308
309 if (*(grub_uint16_t *)(buffer + pe_off + 24) == 0x020b)
310 {
311 return 1;
312 }
313
314 return 0;
315 }
316
317 grub_err_t ventoy_cmd_is_pe64(grub_extcmd_context_t ctxt, int argc, char **args)
318 {
319 int ret = 1;
320 grub_file_t file;
321 grub_uint8_t buf[512];
322
323 (void)ctxt;
324 (void)argc;
325
326 file = grub_file_open(args[0], VENTOY_FILE_TYPE);
327 if (!file)
328 {
329 return 1;
330 }
331
332 grub_memset(buf, 0, 512);
333 grub_file_read(file, buf, 512);
334 if (ventoy_is_pe64(buf))
335 {
336 debug("%s is PE64\n", args[0]);
337 ret = 0;
338 }
339 else
340 {
341 debug("%s is PE32\n", args[0]);
342 }
343 grub_file_close(file);
344
345 return ret;
346 }
347
348 grub_err_t ventoy_cmd_sel_wimboot(grub_extcmd_context_t ctxt, int argc, char **args)
349 {
350 int size;
351 char *buf = NULL;
352 char configfile[128];
353
354 (void)ctxt;
355 (void)argc;
356 (void)args;
357
358 debug("select wimboot argc:%d\n", argc);
359
360 buf = (char *)grub_malloc(8192);
361 if (!buf)
362 {
363 return 0;
364 }
365
366 size = (int)grub_snprintf(buf, 8192,
367 "menuentry \"Windows Setup (32-bit)\" {\n"
368 " set vtoy_wimboot_sel=32\n"
369 "}\n"
370 "menuentry \"Windows Setup (64-bit)\" {\n"
371 " set vtoy_wimboot_sel=64\n"
372 "}\n"
373 );
374 buf[size] = 0;
375
376 g_ventoy_menu_esc = 1;
377 g_ventoy_suppress_esc = 1;
378
379 grub_snprintf(configfile, sizeof(configfile), "configfile mem:0x%llx:size:%d", (ulonglong)(ulong)buf, size);
380 grub_script_execute_sourcecode(configfile);
381
382 g_ventoy_menu_esc = 0;
383 g_ventoy_suppress_esc = 0;
384
385 grub_free(buf);
386
387 if (g_ventoy_last_entry == 0)
388 {
389 debug("last entry=%d %s=32\n", g_ventoy_last_entry, args[0]);
390 grub_env_set(args[0], "32");
391 }
392 else
393 {
394 debug("last entry=%d %s=64\n", g_ventoy_last_entry, args[0]);
395 grub_env_set(args[0], "64");
396 }
397
398 VENTOY_CMD_RETURN(GRUB_ERR_NONE);
399 }
400
401 grub_err_t ventoy_cmd_wimdows_reset(grub_extcmd_context_t ctxt, int argc, char **args)
402 {
403 wim_patch *next = NULL;
404 wim_patch *node = g_wim_patch_head;
405
406 (void)ctxt;
407 (void)argc;
408 (void)args;
409
410 while (node)
411 {
412 next = node->next;
413 grub_free(node);
414 node = next;
415 }
416
417 g_wim_patch_head = NULL;
418 g_wim_total_patch_count = 0;
419 g_wim_valid_patch_count = 0;
420
421 return 0;
422 }
423
424 static int ventoy_load_jump_exe(const char *path, grub_uint8_t **data, grub_uint32_t *size, wim_hash *hash)
425 {
426 grub_uint32_t i;
427 grub_uint32_t align;
428 grub_file_t file;
429
430 debug("windows load jump %s\n", path);
431
432 file = ventoy_grub_file_open(VENTOY_FILE_TYPE, "%s", path);
433 if (!file)
434 {
435 debug("Can't open file %s\n", path);
436 return 1;
437 }
438
439 align = ventoy_align((int)file->size, 2048);
440
441 debug("file %s size:%d align:%u\n", path, (int)file->size, align);
442
443 *size = (grub_uint32_t)file->size;
444 *data = (grub_uint8_t *)grub_malloc(align);
445 if ((*data) == NULL)
446 {
447 debug("Failed to alloc memory size %u\n", align);
448 goto end;
449 }
450
451 grub_file_read(file, (*data), file->size);
452
453 if (hash)
454 {
455 grub_crypto_hash(GRUB_MD_SHA1, hash->sha1, (*data), file->size);
456
457 if (g_ventoy_debug)
458 {
459 debug("%s", "jump bin 64 hash: ");
460 for (i = 0; i < sizeof(hash->sha1); i++)
461 {
462 ventoy_debug("%02x ", hash->sha1[i]);
463 }
464 ventoy_debug("\n");
465 }
466 }
467
468 end:
469
470 grub_file_close(file);
471 return 0;
472 }
473
474 static int ventoy_get_override_info(grub_file_t file, wim_tail *wim_data)
475 {
476 grub_uint32_t start_block;
477 grub_uint64_t file_offset;
478 grub_uint64_t override_offset;
479 grub_uint32_t override_len;
480 grub_uint64_t fe_entry_size_offset;
481
482 if (grub_strcmp(file->fs->name, "iso9660") == 0)
483 {
484 g_iso_fs_type = wim_data->iso_type = 0;
485 override_len = sizeof(ventoy_iso9660_override);
486 override_offset = grub_iso9660_get_last_file_dirent_pos(file) + 2;
487
488 grub_file_read(file, &start_block, 1); // just read for hook trigger
489 file_offset = grub_iso9660_get_last_read_pos(file);
490
491 debug("iso9660 wim size:%llu override_offset:%llu file_offset:%llu\n",
492 (ulonglong)file->size, (ulonglong)override_offset, (ulonglong)file_offset);
493 }
494 else
495 {
496 g_iso_fs_type = wim_data->iso_type = 1;
497 override_len = sizeof(ventoy_udf_override);
498 override_offset = grub_udf_get_last_file_attr_offset(file, &start_block, &fe_entry_size_offset);
499
500 file_offset = grub_udf_get_file_offset(file);
501
502 debug("UDF wim size:%llu override_offset:%llu file_offset:%llu start_block=%u\n",
503 (ulonglong)file->size, (ulonglong)override_offset, (ulonglong)file_offset, start_block);
504 }
505
506 wim_data->file_offset = file_offset;
507 wim_data->udf_start_block = start_block;
508 wim_data->fe_entry_size_offset = fe_entry_size_offset;
509 wim_data->override_offset = override_offset;
510 wim_data->override_len = override_len;
511
512 return 0;
513 }
514
515 static int ventoy_read_resource(grub_file_t fp, wim_header *wimhdr, wim_resource_header *head, void **buffer)
516 {
517 int decompress_len = 0;
518 int total_decompress = 0;
519 grub_uint32_t i = 0;
520 grub_uint32_t chunk_num = 0;
521 grub_uint32_t chunk_size = 0;
522 grub_uint32_t last_chunk_size = 0;
523 grub_uint32_t last_decompress_size = 0;
524 grub_uint32_t cur_offset = 0;
525 grub_uint8_t *cur_dst = NULL;
526 grub_uint8_t *buffer_compress = NULL;
527 grub_uint8_t *buffer_decompress = NULL;
528 grub_uint32_t *chunk_offset = NULL;
529
530 buffer_decompress = (grub_uint8_t *)grub_malloc(head->raw_size + head->size_in_wim);
531 if (NULL == buffer_decompress)
532 {
533 return 0;
534 }
535
536 grub_file_seek(fp, head->offset);
537
538 if (head->size_in_wim == head->raw_size)
539 {
540 grub_file_read(fp, buffer_decompress, head->size_in_wim);
541 *buffer = buffer_decompress;
542 return 0;
543 }
544
545 buffer_compress = buffer_decompress + head->raw_size;
546 grub_file_read(fp, buffer_compress, head->size_in_wim);
547
548 chunk_num = (head->raw_size + WIM_CHUNK_LEN - 1) / WIM_CHUNK_LEN;
549 cur_offset = (chunk_num - 1) * 4;
550 chunk_offset = (grub_uint32_t *)buffer_compress;
551
552 cur_dst = buffer_decompress;
553
554 for (i = 0; i < chunk_num - 1; i++)
555 {
556 chunk_size = (i == 0) ? chunk_offset[i] : chunk_offset[i] - chunk_offset[i - 1];
557
558 if (WIM_CHUNK_LEN == chunk_size)
559 {
560 grub_memcpy(cur_dst, buffer_compress + cur_offset, chunk_size);
561 decompress_len = (int)chunk_size;
562 }
563 else
564 {
565 if (wimhdr->flags & FLAG_HEADER_COMPRESS_XPRESS)
566 {
567 decompress_len = (int)xca_decompress(buffer_compress + cur_offset, chunk_size, cur_dst);
568 }
569 else
570 {
571 decompress_len = (int)lzx_decompress(buffer_compress + cur_offset, chunk_size, cur_dst);
572 }
573 }
574
575 //debug("chunk_size:%u decompresslen:%d\n", chunk_size, decompress_len);
576
577 total_decompress += decompress_len;
578 cur_dst += decompress_len;
579 cur_offset += chunk_size;
580 }
581
582 /* last chunk */
583 last_chunk_size = (grub_uint32_t)(head->size_in_wim - cur_offset);
584 last_decompress_size = head->raw_size - total_decompress;
585
586 if (last_chunk_size < WIM_CHUNK_LEN && last_chunk_size == last_decompress_size)
587 {
588 debug("Last chunk %u uncompressed\n", last_chunk_size);
589 grub_memcpy(cur_dst, buffer_compress + cur_offset, last_chunk_size);
590 decompress_len = (int)last_chunk_size;
591 }
592 else
593 {
594 if (wimhdr->flags & FLAG_HEADER_COMPRESS_XPRESS)
595 {
596 decompress_len = (int)xca_decompress(buffer_compress + cur_offset, head->size_in_wim - cur_offset, cur_dst);
597 }
598 else
599 {
600 decompress_len = (int)lzx_decompress(buffer_compress + cur_offset, head->size_in_wim - cur_offset, cur_dst);
601 }
602 }
603
604 cur_dst += decompress_len;
605 total_decompress += decompress_len;
606
607 if (cur_dst != buffer_decompress + head->raw_size)
608 {
609 debug("head->size_in_wim:%llu head->raw_size:%llu cur_dst:%p buffer_decompress:%p total_decompress:%d\n",
610 (ulonglong)head->size_in_wim, (ulonglong)head->raw_size, cur_dst, buffer_decompress, total_decompress);
611 grub_free(buffer_decompress);
612 return 1;
613 }
614
615 *buffer = buffer_decompress;
616 return 0;
617 }
618
619
620 static wim_directory_entry * search_wim_dirent(wim_directory_entry *dir, const char *search_name)
621 {
622 do
623 {
624 if (dir->len && dir->name_len)
625 {
626 if (wim_name_cmp(search_name, (grub_uint16_t *)(dir + 1), dir->name_len / 2) == 0)
627 {
628 return dir;
629 }
630 }
631 dir = (wim_directory_entry *)((grub_uint8_t *)dir + dir->len);
632 } while(dir->len);
633
634 return NULL;
635 }
636
637 static wim_directory_entry * search_full_wim_dirent
638 (
639 void *meta_data,
640 wim_directory_entry *dir,
641 const char **path
642 )
643 {
644 wim_directory_entry *subdir = NULL;
645 wim_directory_entry *search = dir;
646
647 while (*path)
648 {
649 subdir = (wim_directory_entry *)((char *)meta_data + search->subdir);
650 search = search_wim_dirent(subdir, *path);
651 path++;
652 }
653
654 return search;
655 }
656
657 static wim_directory_entry * search_replace_wim_dirent(void *meta_data, wim_directory_entry *dir)
658 {
659 wim_directory_entry *wim_dirent = NULL;
660 const char *winpeshl_path[] = { "Windows", "System32", "winpeshl.exe", NULL };
661 //const char *native_path[] = { "Windows", "System32", "native.exe", NULL };
662
663 wim_dirent = search_full_wim_dirent(meta_data, dir, winpeshl_path);
664 debug("search winpeshl.exe %p\n", wim_dirent);
665 if (wim_dirent)
666 {
667 return wim_dirent;
668 }
669
670 #if 0
671 wim_dirent = search_full_wim_dirent(meta_data, dir, native_path);
672 debug("search native.exe %p\n", wim_dirent);
673 if (wim_dirent)
674 {
675 return wim_dirent;
676 }
677 #endif
678
679 return NULL;
680 }
681
682
683 static wim_lookup_entry * ventoy_find_look_entry(wim_header *header, wim_lookup_entry *lookup, wim_hash *hash)
684 {
685 grub_uint32_t i = 0;
686
687 for (i = 0; i < (grub_uint32_t)header->lookup.raw_size / sizeof(wim_lookup_entry); i++)
688 {
689 if (grub_memcmp(&lookup[i].hash, hash, sizeof(wim_hash)) == 0)
690 {
691 return lookup + i;
692 }
693 }
694
695 return NULL;
696 }
697
698 static wim_lookup_entry * ventoy_find_meta_entry(wim_header *header, wim_lookup_entry *lookup)
699 {
700 grub_uint32_t i = 0;
701 grub_uint32_t index = 0;;
702
703 if ((header == NULL) || (lookup == NULL))
704 {
705 return NULL;
706 }
707
708 for (i = 0; i < (grub_uint32_t)header->lookup.raw_size / sizeof(wim_lookup_entry); i++)
709 {
710 if (lookup[i].resource.flags & RESHDR_FLAG_METADATA)
711 {
712 index++;
713 if (index == header->boot_index)
714 {
715 return lookup + i;
716 }
717 }
718 }
719
720 return NULL;
721 }
722
723 static grub_uint64_t ventoy_get_stream_len(wim_directory_entry *dir)
724 {
725 grub_uint16_t i;
726 grub_uint64_t offset = 0;
727 wim_stream_entry *stream = (wim_stream_entry *)((char *)dir + dir->len);
728
729 for (i = 0; i < dir->streams; i++)
730 {
731 offset += stream->len;
732 stream = (wim_stream_entry *)((char *)stream + stream->len);
733 }
734
735 return offset;
736 }
737
738 static int ventoy_update_all_hash(wim_patch *patch, void *meta_data, wim_directory_entry *dir)
739 {
740 if ((meta_data == NULL) || (dir == NULL))
741 {
742 return 0;
743 }
744
745 if (dir->len < sizeof(wim_directory_entry))
746 {
747 return 0;
748 }
749
750 do
751 {
752 if (dir->subdir == 0 && grub_memcmp(dir->hash.sha1, patch->old_hash.sha1, sizeof(wim_hash)) == 0)
753 {
754 debug("find target file, name_len:%u upadte hash\n", dir->name_len);
755 grub_memcpy(dir->hash.sha1, &(patch->wim_data.bin_hash), sizeof(wim_hash));
756 }
757
758 if (dir->subdir)
759 {
760 ventoy_update_all_hash(patch, meta_data, (wim_directory_entry *)((char *)meta_data + dir->subdir));
761 }
762
763 if (dir->streams)
764 {
765 dir = (wim_directory_entry *)((char *)dir + dir->len + ventoy_get_stream_len(dir));
766 }
767 else
768 {
769 dir = (wim_directory_entry *)((char *)dir + dir->len);
770 }
771 } while (dir->len >= sizeof(wim_directory_entry));
772
773 return 0;
774 }
775
776 static int ventoy_cat_exe_file_data(wim_tail *wim_data, grub_uint32_t exe_len, grub_uint8_t *exe_data)
777 {
778 int pe64 = 0;
779 char file[256];
780 grub_uint32_t jump_len = 0;
781 grub_uint32_t jump_align = 0;
782 grub_uint8_t *jump_data = NULL;
783
784 pe64 = ventoy_is_pe64(exe_data);
785
786 grub_snprintf(file, sizeof(file), "%s/vtoyjump%d.exe", grub_env_get("vtoy_path"), pe64 ? 64 : 32);
787 ventoy_load_jump_exe(file, &jump_data, &jump_len, NULL);
788 jump_align = ventoy_align(jump_len, 16);
789
790 wim_data->jump_exe_len = jump_len;
791 wim_data->bin_raw_len = jump_align + sizeof(ventoy_os_param) + sizeof(ventoy_windows_data) + exe_len;
792 wim_data->bin_align_len = ventoy_align(wim_data->bin_raw_len, 2048);
793
794 wim_data->jump_bin_data = grub_malloc(wim_data->bin_align_len);
795 if (wim_data->jump_bin_data)
796 {
797 grub_memcpy(wim_data->jump_bin_data, jump_data, jump_len);
798 grub_memcpy(wim_data->jump_bin_data + jump_align + sizeof(ventoy_os_param) + sizeof(ventoy_windows_data), exe_data, exe_len);
799 }
800
801 debug("jump_exe_len:%u bin_raw_len:%u bin_align_len:%u\n",
802 wim_data->jump_exe_len, wim_data->bin_raw_len, wim_data->bin_align_len);
803
804 return 0;
805 }
806
807 int ventoy_fill_windows_rtdata(void *buf, char *isopath)
808 {
809 char *pos = NULL;
810 char *script = NULL;
811 ventoy_windows_data *data = (ventoy_windows_data *)buf;
812
813 grub_memset(data, 0, sizeof(ventoy_windows_data));
814
815 pos = grub_strstr(isopath, "/");
816 if (!pos)
817 {
818 return 1;
819 }
820
821 script = ventoy_plugin_get_cur_install_template(pos);
822 if (script)
823 {
824 debug("auto install script <%s>\n", script);
825 grub_snprintf(data->auto_install_script, sizeof(data->auto_install_script) - 1, "%s", script);
826 }
827 else
828 {
829 debug("auto install script skipped or not configed %s\n", pos);
830 }
831
832 script = (char *)ventoy_plugin_get_injection(pos);
833 if (script)
834 {
835 if (ventoy_check_file_exist("%s%s", ventoy_get_env("vtoy_iso_part"), script))
836 {
837 debug("injection archive <%s> OK\n", script);
838 grub_snprintf(data->injection_archive, sizeof(data->injection_archive) - 1, "%s", script);
839 }
840 else
841 {
842 debug("injection archive <%s> NOT exist\n", script);
843 }
844 }
845 else
846 {
847 debug("injection archive not configed %s\n", pos);
848 }
849
850 return 0;
851 }
852
853 static int ventoy_update_before_chain(ventoy_os_param *param, char *isopath)
854 {
855 grub_uint32_t jump_align = 0;
856 wim_lookup_entry *meta_look = NULL;
857 wim_security_header *security = NULL;
858 wim_directory_entry *rootdir = NULL;
859 wim_header *head = NULL;
860 wim_lookup_entry *lookup = NULL;
861 wim_patch *node = NULL;
862 wim_tail *wim_data = NULL;
863
864 for (node = g_wim_patch_head; node; node = node->next)
865 {
866 if (0 == node->valid)
867 {
868 continue;
869 }
870
871 wim_data = &node->wim_data;
872 head = &wim_data->wim_header;
873 lookup = (wim_lookup_entry *)wim_data->new_lookup_data;
874
875 jump_align = ventoy_align(wim_data->jump_exe_len, 16);
876 if (wim_data->jump_bin_data)
877 {
878 grub_memcpy(wim_data->jump_bin_data + jump_align, param, sizeof(ventoy_os_param));
879 ventoy_fill_windows_rtdata(wim_data->jump_bin_data + jump_align + sizeof(ventoy_os_param), isopath);
880 }
881
882 grub_crypto_hash(GRUB_MD_SHA1, wim_data->bin_hash.sha1, wim_data->jump_bin_data, wim_data->bin_raw_len);
883
884 security = (wim_security_header *)wim_data->new_meta_data;
885 if (security->len > 0)
886 {
887 rootdir = (wim_directory_entry *)(wim_data->new_meta_data + ((security->len + 7) & 0xFFFFFFF8U));
888 }
889 else
890 {
891 rootdir = (wim_directory_entry *)(wim_data->new_meta_data + 8);
892 }
893
894 /* update all winpeshl.exe dirent entry's hash */
895 ventoy_update_all_hash(node, wim_data->new_meta_data, rootdir);
896
897 /* update winpeshl.exe lookup entry data (hash/offset/length) */
898 if (node->replace_look)
899 {
900 debug("update replace lookup entry_id:%ld\n", ((long)node->replace_look - (long)lookup) / sizeof(wim_lookup_entry));
901 node->replace_look->resource.raw_size = wim_data->bin_raw_len;
902 node->replace_look->resource.size_in_wim = wim_data->bin_raw_len;
903 node->replace_look->resource.flags = 0;
904 node->replace_look->resource.offset = wim_data->wim_align_size;
905
906 grub_memcpy(node->replace_look->hash.sha1, wim_data->bin_hash.sha1, sizeof(wim_hash));
907 }
908
909 /* update metadata's hash */
910 meta_look = ventoy_find_meta_entry(head, lookup);
911 if (meta_look)
912 {
913 debug("find meta lookup entry_id:%ld\n", ((long)meta_look - (long)lookup) / sizeof(wim_lookup_entry));
914 grub_memcpy(&meta_look->resource, &head->metadata, sizeof(wim_resource_header));
915 grub_crypto_hash(GRUB_MD_SHA1, meta_look->hash.sha1, wim_data->new_meta_data, wim_data->new_meta_len);
916 }
917 }
918
919 return 0;
920 }
921
922 static int ventoy_wimdows_locate_wim(const char *disk, wim_patch *patch)
923 {
924 int rc;
925 grub_file_t file;
926 grub_uint32_t exe_len;
927 grub_uint8_t *exe_data = NULL;
928 grub_uint8_t *decompress_data = NULL;
929 wim_lookup_entry *lookup = NULL;
930 wim_security_header *security = NULL;
931 wim_directory_entry *rootdir = NULL;
932 wim_directory_entry *search = NULL;
933 wim_header *head = &(patch->wim_data.wim_header);
934 wim_tail *wim_data = &patch->wim_data;
935
936 debug("windows locate wim start %s\n", patch->path);
937
938 g_ventoy_case_insensitive = 1;
939 file = ventoy_grub_file_open(VENTOY_FILE_TYPE, "%s%s", disk, patch->path);
940 g_ventoy_case_insensitive = 0;
941
942 if (!file)
943 {
944 debug("File %s%s NOT exist\n", disk, patch->path);
945 return 1;
946 }
947
948 ventoy_get_override_info(file, &patch->wim_data);
949
950 grub_file_seek(file, 0);
951 grub_file_read(file, head, sizeof(wim_header));
952
953 if (grub_memcmp(head->signature, WIM_HEAD_SIGNATURE, sizeof(head->signature)))
954 {
955 debug("Not a valid wim file %s\n", (char *)head->signature);
956 grub_file_close(file);
957 return 1;
958 }
959
960 if (head->flags & FLAG_HEADER_COMPRESS_LZMS)
961 {
962 debug("LZMS compress is not supported 0x%x\n", head->flags);
963 grub_file_close(file);
964 return 1;
965 }
966
967 rc = ventoy_read_resource(file, head, &head->metadata, (void **)&decompress_data);
968 if (rc)
969 {
970 grub_printf("failed to read meta data %d\n", rc);
971 grub_file_close(file);
972 return 1;
973 }
974
975 security = (wim_security_header *)decompress_data;
976 if (security->len > 0)
977 {
978 rootdir = (wim_directory_entry *)(decompress_data + ((security->len + 7) & 0xFFFFFFF8U));
979 }
980 else
981 {
982 rootdir = (wim_directory_entry *)(decompress_data + 8);
983 }
984
985 /* search winpeshl.exe dirent entry */
986 search = search_replace_wim_dirent(decompress_data, rootdir);
987 if (!search)
988 {
989 debug("Failed to find replace file %p\n", search);
990 grub_file_close(file);
991 return 1;
992 }
993
994 debug("find replace file at %p\n", search);
995
996 grub_memcpy(&patch->old_hash, search->hash.sha1, sizeof(wim_hash));
997
998 debug("read lookup offset:%llu size:%llu\n", (ulonglong)head->lookup.offset, (ulonglong)head->lookup.raw_size);
999 lookup = grub_malloc(head->lookup.raw_size);
1000 grub_file_seek(file, head->lookup.offset);
1001 grub_file_read(file, lookup, head->lookup.raw_size);
1002
1003 /* find and extact winpeshl.exe */
1004 patch->replace_look = ventoy_find_look_entry(head, lookup, &patch->old_hash);
1005 if (patch->replace_look)
1006 {
1007 exe_len = (grub_uint32_t)patch->replace_look->resource.raw_size;
1008 debug("find replace lookup entry_id:%ld raw_size:%u\n",
1009 ((long)patch->replace_look - (long)lookup) / sizeof(wim_lookup_entry), exe_len);
1010
1011 if (0 == ventoy_read_resource(file, head, &(patch->replace_look->resource), (void **)&(exe_data)))
1012 {
1013 ventoy_cat_exe_file_data(wim_data, exe_len, exe_data);
1014 grub_free(exe_data);
1015 }
1016 else
1017 {
1018 debug("failed to read replace file meta data %u\n", exe_len);
1019 }
1020 }
1021 else
1022 {
1023 debug("failed to find lookup entry for replace file 0x%02x 0x%02x\n",
1024 patch->old_hash.sha1[0], patch->old_hash.sha1[1]);
1025 }
1026
1027 wim_data->wim_raw_size = (grub_uint32_t)file->size;
1028 wim_data->wim_align_size = ventoy_align(wim_data->wim_raw_size, 2048);
1029
1030 grub_check_free(wim_data->new_meta_data);
1031 wim_data->new_meta_data = decompress_data;
1032 wim_data->new_meta_len = head->metadata.raw_size;
1033 wim_data->new_meta_align_len = ventoy_align(wim_data->new_meta_len, 2048);
1034
1035 grub_check_free(wim_data->new_lookup_data);
1036 wim_data->new_lookup_data = (grub_uint8_t *)lookup;
1037 wim_data->new_lookup_len = (grub_uint32_t)head->lookup.raw_size;
1038 wim_data->new_lookup_align_len = ventoy_align(wim_data->new_lookup_len, 2048);
1039
1040 head->metadata.flags = RESHDR_FLAG_METADATA;
1041 head->metadata.offset = wim_data->wim_align_size + wim_data->bin_align_len;
1042 head->metadata.size_in_wim = wim_data->new_meta_len;
1043 head->metadata.raw_size = wim_data->new_meta_len;
1044
1045 head->lookup.flags = 0;
1046 head->lookup.offset = head->metadata.offset + wim_data->new_meta_align_len;
1047 head->lookup.size_in_wim = wim_data->new_lookup_len;
1048 head->lookup.raw_size = wim_data->new_lookup_len;
1049
1050 grub_file_close(file);
1051
1052 debug("%s", "windows locate wim finish\n");
1053 return 0;
1054 }
1055
1056 grub_err_t ventoy_cmd_locate_wim_patch(grub_extcmd_context_t ctxt, int argc, char **args)
1057 {
1058 wim_patch *node = g_wim_patch_head;
1059
1060 (void)ctxt;
1061 (void)argc;
1062 (void)args;
1063
1064 while (node)
1065 {
1066 if (0 == ventoy_wimdows_locate_wim(args[0], node))
1067 {
1068 node->valid = 1;
1069 g_wim_valid_patch_count++;
1070 }
1071
1072 node = node->next;
1073 }
1074
1075 return 0;
1076 }
1077
1078 static grub_uint32_t ventoy_get_override_chunk_num(void)
1079 {
1080 grub_uint32_t chunk_num = 0;
1081
1082 if (g_iso_fs_type == 0)
1083 {
1084 /* ISO9660: */
1085 /* per wim */
1086 /* 1: file_size and file_offset */
1087 /* 2: new wim file header */
1088 chunk_num = g_wim_valid_patch_count * 2;
1089 }
1090 else
1091 {
1092 /* UDF: */
1093 /* global: */
1094 /* 1: block count in Partition Descriptor */
1095
1096 /* per wim */
1097 /* 1: file_size in file_entry or extend_file_entry */
1098 /* 2: data_size and position in extend data short ad */
1099 /* 3: new wim file header */
1100 chunk_num = g_wim_valid_patch_count * 3 + 1;
1101 }
1102
1103 if (g_suppress_wincd_override_offset > 0)
1104 {
1105 chunk_num++;
1106 }
1107
1108 return chunk_num;
1109 }
1110
1111 static void ventoy_fill_suppress_wincd_override_data(void *override)
1112 {
1113 ventoy_override_chunk *cur = (ventoy_override_chunk *)override;
1114
1115 cur->override_size = 4;
1116 cur->img_offset = g_suppress_wincd_override_offset;
1117 grub_memcpy(cur->override_data, &g_suppress_wincd_override_data, cur->override_size);
1118 }
1119
1120 static void ventoy_windows_fill_override_data_iso9660( grub_uint64_t isosize, void *override)
1121 {
1122 grub_uint64_t sector;
1123 grub_uint32_t new_wim_size;
1124 ventoy_override_chunk *cur;
1125 wim_patch *node = NULL;
1126 wim_tail *wim_data = NULL;
1127 ventoy_iso9660_override *dirent = NULL;
1128
1129 sector = (isosize + 2047) / 2048;
1130
1131 cur = (ventoy_override_chunk *)override;
1132
1133 if (g_suppress_wincd_override_offset > 0)
1134 {
1135 ventoy_fill_suppress_wincd_override_data(cur);
1136 cur++;
1137 }
1138
1139 debug("ventoy_windows_fill_override_data_iso9660 %lu\n", (ulong)isosize);
1140
1141 for (node = g_wim_patch_head; node; node = node->next)
1142 {
1143 wim_data = &node->wim_data;
1144 if (0 == node->valid)
1145 {
1146 continue;
1147 }
1148
1149 new_wim_size = wim_data->wim_align_size + wim_data->bin_align_len +
1150 wim_data->new_meta_align_len + wim_data->new_lookup_align_len;
1151
1152 dirent = (ventoy_iso9660_override *)wim_data->override_data;
1153
1154 dirent->first_sector = (grub_uint32_t)sector;
1155 dirent->size = new_wim_size;
1156 dirent->first_sector_be = grub_swap_bytes32(dirent->first_sector);
1157 dirent->size_be = grub_swap_bytes32(dirent->size);
1158
1159 sector += (new_wim_size / 2048);
1160
1161 /* override 1: position and length in dirent */
1162 cur->img_offset = wim_data->override_offset;
1163 cur->override_size = wim_data->override_len;
1164 grub_memcpy(cur->override_data, wim_data->override_data, cur->override_size);
1165 cur++;
1166
1167 /* override 2: new wim file header */
1168 cur->img_offset = wim_data->file_offset;
1169 cur->override_size = sizeof(wim_header);
1170 grub_memcpy(cur->override_data, &(wim_data->wim_header), cur->override_size);
1171 cur++;
1172 }
1173
1174 return;
1175 }
1176
1177 static int ventoy_windows_fill_udf_short_ad(grub_file_t isofile, grub_uint32_t curpos,
1178 wim_tail *wim_data, grub_uint32_t new_wim_size)
1179 {
1180 int i;
1181 grub_uint32_t total = 0;
1182 grub_uint32_t left_size = 0;
1183 ventoy_udf_override *udf = NULL;
1184 ventoy_udf_override tmp[4];
1185
1186 grub_memset(tmp, 0, sizeof(tmp));
1187 grub_file_seek(isofile, wim_data->override_offset);
1188 grub_file_read(isofile, tmp, sizeof(tmp));
1189
1190 left_size = new_wim_size;
1191 udf = (ventoy_udf_override *)wim_data->override_data;
1192
1193 for (i = 0; i < 4; i++)
1194 {
1195 total += tmp[i].length;
1196 if (total >= wim_data->wim_raw_size)
1197 {
1198 udf->length = left_size;
1199 udf->position = curpos;
1200 return 0;
1201 }
1202 else
1203 {
1204 udf->length = tmp[i].length;
1205 udf->position = curpos;
1206 }
1207
1208 left_size -= tmp[i].length;
1209 curpos += udf->length / 2048;
1210 udf++;
1211 wim_data->override_len += sizeof(ventoy_udf_override);
1212 }
1213
1214 debug("######## Too many udf ad ######\n");
1215 return 1;
1216 }
1217
1218 static void ventoy_windows_fill_override_data_udf(grub_file_t isofile, void *override)
1219 {
1220 grub_uint32_t data32;
1221 grub_uint64_t data64;
1222 grub_uint64_t sector;
1223 grub_uint32_t new_wim_size;
1224 grub_uint64_t total_wim_size = 0;
1225 grub_uint32_t udf_start_block = 0;
1226 ventoy_override_chunk *cur;
1227 wim_patch *node = NULL;
1228 wim_tail *wim_data = NULL;
1229
1230 sector = (isofile->size + 2047) / 2048;
1231
1232 cur = (ventoy_override_chunk *)override;
1233
1234 if (g_suppress_wincd_override_offset > 0)
1235 {
1236 ventoy_fill_suppress_wincd_override_data(cur);
1237 cur++;
1238 }
1239
1240 debug("ventoy_windows_fill_override_data_udf %lu\n", (ulong)isofile->size);
1241
1242 for (node = g_wim_patch_head; node; node = node->next)
1243 {
1244 wim_data = &node->wim_data;
1245 if (node->valid)
1246 {
1247 if (udf_start_block == 0)
1248 {
1249 udf_start_block = wim_data->udf_start_block;
1250 }
1251 new_wim_size = wim_data->wim_align_size + wim_data->bin_align_len +
1252 wim_data->new_meta_align_len + wim_data->new_lookup_align_len;
1253 total_wim_size += new_wim_size;
1254 }
1255 }
1256
1257 //override 1: sector number in pd data
1258 cur->img_offset = grub_udf_get_last_pd_size_offset();
1259 cur->override_size = 4;
1260 data32 = sector - udf_start_block + (total_wim_size / 2048);
1261 grub_memcpy(cur->override_data, &(data32), 4);
1262
1263 for (node = g_wim_patch_head; node; node = node->next)
1264 {
1265 wim_data = &node->wim_data;
1266 if (0 == node->valid)
1267 {
1268 continue;
1269 }
1270
1271 new_wim_size = wim_data->wim_align_size + wim_data->bin_align_len +
1272 wim_data->new_meta_align_len + wim_data->new_lookup_align_len;
1273
1274 //override 2: filesize in file_entry
1275 cur++;
1276 cur->img_offset = wim_data->fe_entry_size_offset;
1277 cur->override_size = 8;
1278 data64 = new_wim_size;
1279 grub_memcpy(cur->override_data, &(data64), 8);
1280
1281 /* override 3: position and length in extend data */
1282 ventoy_windows_fill_udf_short_ad(isofile, (grub_uint32_t)sector - udf_start_block, wim_data, new_wim_size);
1283
1284 sector += (new_wim_size / 2048);
1285
1286 cur++;
1287 cur->img_offset = wim_data->override_offset;
1288 cur->override_size = wim_data->override_len;
1289 grub_memcpy(cur->override_data, wim_data->override_data, cur->override_size);
1290
1291 /* override 4: new wim file header */
1292 cur++;
1293 cur->img_offset = wim_data->file_offset;
1294 cur->override_size = sizeof(wim_header);
1295 grub_memcpy(cur->override_data, &(wim_data->wim_header), cur->override_size);
1296 }
1297
1298 return;
1299 }
1300
1301 static grub_uint32_t ventoy_windows_get_virt_data_size(void)
1302 {
1303 grub_uint32_t size = 0;
1304 wim_tail *wim_data = NULL;
1305 wim_patch *node = g_wim_patch_head;
1306
1307 while (node)
1308 {
1309 if (node->valid)
1310 {
1311 wim_data = &node->wim_data;
1312 size += sizeof(ventoy_virt_chunk) + wim_data->bin_align_len +
1313 wim_data->new_meta_align_len + wim_data->new_lookup_align_len;
1314 }
1315 node = node->next;
1316 }
1317
1318 return size;
1319 }
1320
1321 static void ventoy_windows_fill_virt_data( grub_uint64_t isosize, ventoy_chain_head *chain)
1322 {
1323 grub_uint64_t sector;
1324 grub_uint32_t offset;
1325 grub_uint32_t wim_secs;
1326 grub_uint32_t mem_secs;
1327 char *override = NULL;
1328 ventoy_virt_chunk *cur = NULL;
1329 wim_tail *wim_data = NULL;
1330 wim_patch *node = NULL;
1331
1332 sector = (isosize + 2047) / 2048;
1333 offset = sizeof(ventoy_virt_chunk) * g_wim_valid_patch_count;
1334
1335 override = (char *)chain + chain->virt_chunk_offset;
1336 cur = (ventoy_virt_chunk *)override;
1337
1338 for (node = g_wim_patch_head; node; node = node->next)
1339 {
1340 if (0 == node->valid)
1341 {
1342 continue;
1343 }
1344
1345 wim_data = &node->wim_data;
1346
1347 wim_secs = wim_data->wim_align_size / 2048;
1348 mem_secs = (wim_data->bin_align_len + wim_data->new_meta_align_len + wim_data->new_lookup_align_len) / 2048;
1349
1350 cur->remap_sector_start = sector;
1351 cur->remap_sector_end = cur->remap_sector_start + wim_secs;
1352 cur->org_sector_start = (grub_uint32_t)(wim_data->file_offset / 2048);
1353
1354 cur->mem_sector_start = cur->remap_sector_end;
1355 cur->mem_sector_end = cur->mem_sector_start + mem_secs;
1356 cur->mem_sector_offset = offset;
1357
1358 sector += wim_secs + mem_secs;
1359 cur++;
1360
1361 grub_memcpy(override + offset, wim_data->jump_bin_data, wim_data->bin_raw_len);
1362 offset += wim_data->bin_align_len;
1363
1364 grub_memcpy(override + offset, wim_data->new_meta_data, wim_data->new_meta_len);
1365 offset += wim_data->new_meta_align_len;
1366
1367 grub_memcpy(override + offset, wim_data->new_lookup_data, wim_data->new_lookup_len);
1368 offset += wim_data->new_lookup_align_len;
1369
1370 chain->virt_img_size_in_bytes += wim_data->wim_align_size +
1371 wim_data->bin_align_len +
1372 wim_data->new_meta_align_len +
1373 wim_data->new_lookup_align_len;
1374 }
1375
1376 return;
1377 }
1378
1379 static int ventoy_windows_drive_map(ventoy_chain_head *chain)
1380 {
1381 grub_disk_t disk;
1382
1383 debug("drive map begin <%p> ...\n", chain);
1384
1385 if (chain->disk_drive == 0x80)
1386 {
1387 disk = grub_disk_open("hd1");
1388 if (disk)
1389 {
1390 grub_disk_close(disk);
1391 debug("drive map needed %p\n", disk);
1392 chain->drive_map = 0x81;
1393 }
1394 else
1395 {
1396 debug("failed to open disk %s\n", "hd1");
1397 }
1398 }
1399 else
1400 {
1401 debug("no need to map 0x%x\n", chain->disk_drive);
1402 }
1403
1404 return 0;
1405 }
1406
1407 static int ventoy_suppress_windows_cd_prompt(void)
1408 {
1409 int rc = 1;
1410 const char *cdprompt = NULL;
1411 grub_uint64_t readpos = 0;
1412 grub_file_t file = NULL;
1413 grub_uint8_t data[32];
1414
1415 cdprompt = ventoy_get_env("VTOY_WINDOWS_CD_PROMPT");
1416 if (cdprompt && cdprompt[0] == '1' && cdprompt[1] == 0)
1417 {
1418 debug("VTOY_WINDOWS_CD_PROMPT:<%s>\n", cdprompt);
1419 return 0;
1420 }
1421
1422 g_ventoy_case_insensitive = 1;
1423 file = ventoy_grub_file_open(VENTOY_FILE_TYPE, "%s/boot/bootfix.bin", "(loop)");
1424 g_ventoy_case_insensitive = 0;
1425
1426 if (!file)
1427 {
1428 debug("Failed to open %s\n", "bootfix.bin");
1429 goto end;
1430 }
1431
1432 grub_file_read(file, data, 32);
1433
1434 if (file->fs && file->fs->name && grub_strcmp(file->fs->name, "udf") == 0)
1435 {
1436 readpos = grub_udf_get_file_offset(file);
1437 }
1438 else
1439 {
1440 readpos = grub_iso9660_get_last_read_pos(file);
1441 }
1442
1443 debug("bootfix.bin readpos:%lu (sector:%lu) data: %02x %02x %02x %02x\n",
1444 (ulong)readpos, (ulong)readpos / 2048, data[24], data[25], data[26], data[27]);
1445
1446 if (*(grub_uint32_t *)(data + 24) == 0x13cd0080)
1447 {
1448 g_suppress_wincd_override_offset = readpos + 24;
1449 g_suppress_wincd_override_data = 0x13cd00fd;
1450
1451 rc = 0;
1452 }
1453
1454 debug("g_suppress_wincd_override_offset:%lu\n", (ulong)g_suppress_wincd_override_offset);
1455
1456 end:
1457 check_free(file, grub_file_close);
1458
1459 return rc;
1460 }
1461
1462 grub_err_t ventoy_cmd_windows_wimboot_data(grub_extcmd_context_t ctxt, int argc, char **args)
1463 {
1464 grub_uint32_t size = 0;
1465 const char *addr = NULL;
1466 ventoy_chain_head *chain = NULL;
1467 ventoy_os_param *param = NULL;
1468 char envbuf[64];
1469
1470 (void)ctxt;
1471 (void)argc;
1472 (void)args;
1473
1474 addr = grub_env_get("vtoy_chain_mem_addr");
1475 if (!addr)
1476 {
1477 debug("Failed to find vtoy_chain_mem_addr\n");
1478 return 1;
1479 }
1480
1481 chain = (ventoy_chain_head *)(void *)grub_strtoul(addr, NULL, 16);
1482
1483 if (grub_memcmp(&g_ventoy_guid, &chain->os_param.guid, 16) != 0)
1484 {
1485 debug("os_param.guid not match\n");
1486 return 1;
1487 }
1488
1489 size = sizeof(ventoy_os_param) + sizeof(ventoy_windows_data);
1490 param = (ventoy_os_param *)grub_zalloc(size);
1491 if (!param)
1492 {
1493 return 1;
1494 }
1495
1496 grub_memcpy(param, &chain->os_param, sizeof(ventoy_os_param));
1497 ventoy_fill_windows_rtdata(param + 1, param->vtoy_img_path);
1498
1499 grub_snprintf(envbuf, sizeof(envbuf), "0x%lx", (unsigned long)param);
1500 grub_env_set("vtoy_wimboot_mem_addr", envbuf);
1501 debug("vtoy_wimboot_mem_addr: %s\n", envbuf);
1502
1503 grub_snprintf(envbuf, sizeof(envbuf), "%u", size);
1504 grub_env_set("vtoy_wimboot_mem_size", envbuf);
1505 debug("vtoy_wimboot_mem_size: %s\n", envbuf);
1506
1507 VENTOY_CMD_RETURN(GRUB_ERR_NONE);
1508 }
1509
1510 grub_err_t ventoy_cmd_windows_chain_data(grub_extcmd_context_t ctxt, int argc, char **args)
1511 {
1512 int unknown_image = 0;
1513 int ventoy_compatible = 0;
1514 grub_uint32_t size = 0;
1515 grub_uint64_t isosize = 0;
1516 grub_uint32_t boot_catlog = 0;
1517 grub_uint32_t img_chunk_size = 0;
1518 grub_uint32_t override_size = 0;
1519 grub_uint32_t virt_chunk_size = 0;
1520 grub_file_t file;
1521 grub_disk_t disk;
1522 const char *pLastChain = NULL;
1523 const char *compatible;
1524 ventoy_chain_head *chain;
1525 char envbuf[64];
1526
1527 (void)ctxt;
1528 (void)argc;
1529
1530 debug("chain data begin <%s> ...\n", args[0]);
1531
1532 compatible = grub_env_get("ventoy_compatible");
1533 if (compatible && compatible[0] == 'Y')
1534 {
1535 ventoy_compatible = 1;
1536 }
1537
1538 if (NULL == g_img_chunk_list.chunk)
1539 {
1540 grub_printf("ventoy not ready\n");
1541 return 1;
1542 }
1543
1544 if (0 == ventoy_compatible && g_wim_valid_patch_count == 0)
1545 {
1546 unknown_image = 1;
1547 debug("Warning: %s was not recognized by Ventoy\n", args[0]);
1548 }
1549
1550 file = ventoy_grub_file_open(VENTOY_FILE_TYPE, "%s", args[0]);
1551 if (!file)
1552 {
1553 return 1;
1554 }
1555
1556 isosize = file->size;
1557
1558 boot_catlog = ventoy_get_iso_boot_catlog(file);
1559 if (boot_catlog)
1560 {
1561 if (ventoy_is_efi_os() && (!ventoy_has_efi_eltorito(file, boot_catlog)))
1562 {
1563 grub_env_set("LoadIsoEfiDriver", "on");
1564 }
1565 }
1566 else
1567 {
1568 if (ventoy_is_efi_os())
1569 {
1570 grub_env_set("LoadIsoEfiDriver", "on");
1571 }
1572 else
1573 {
1574 return grub_error(GRUB_ERR_BAD_ARGUMENT, "File %s is not bootable", args[0]);
1575 }
1576 }
1577
1578 g_suppress_wincd_override_offset = 0;
1579 if (!ventoy_is_efi_os()) /* legacy mode */
1580 {
1581 ventoy_suppress_windows_cd_prompt();
1582 }
1583
1584 img_chunk_size = g_img_chunk_list.cur_chunk * sizeof(ventoy_img_chunk);
1585
1586 if (ventoy_compatible || unknown_image)
1587 {
1588 override_size = g_suppress_wincd_override_offset > 0 ? sizeof(ventoy_override_chunk) : 0;
1589 size = sizeof(ventoy_chain_head) + img_chunk_size + override_size;
1590 }
1591 else
1592 {
1593 override_size = ventoy_get_override_chunk_num() * sizeof(ventoy_override_chunk);
1594 virt_chunk_size = ventoy_windows_get_virt_data_size();
1595 size = sizeof(ventoy_chain_head) + img_chunk_size + override_size + virt_chunk_size;
1596 }
1597
1598 pLastChain = grub_env_get("vtoy_chain_mem_addr");
1599 if (pLastChain)
1600 {
1601 chain = (ventoy_chain_head *)grub_strtoul(pLastChain, NULL, 16);
1602 if (chain)
1603 {
1604 debug("free last chain memory %p\n", chain);
1605 grub_free(chain);
1606 }
1607 }
1608
1609 chain = grub_malloc(size);
1610 if (!chain)
1611 {
1612 grub_printf("Failed to alloc chain memory size %u\n", size);
1613 grub_file_close(file);
1614 return 1;
1615 }
1616
1617 grub_snprintf(envbuf, sizeof(envbuf), "0x%lx", (unsigned long)chain);
1618 grub_env_set("vtoy_chain_mem_addr", envbuf);
1619 grub_snprintf(envbuf, sizeof(envbuf), "%u", size);
1620 grub_env_set("vtoy_chain_mem_size", envbuf);
1621
1622 grub_memset(chain, 0, sizeof(ventoy_chain_head));
1623
1624 /* part 1: os parameter */
1625 g_ventoy_chain_type = ventoy_chain_windows;
1626 ventoy_fill_os_param(file, &(chain->os_param));
1627
1628 if (0 == unknown_image)
1629 {
1630 ventoy_update_before_chain(&(chain->os_param), args[0]);
1631 }
1632
1633 /* part 2: chain head */
1634 disk = file->device->disk;
1635 chain->disk_drive = disk->id;
1636 chain->disk_sector_size = (1 << disk->log_sector_size);
1637 chain->real_img_size_in_bytes = file->size;
1638 chain->virt_img_size_in_bytes = (file->size + 2047) / 2048 * 2048;
1639 chain->boot_catalog = boot_catlog;
1640
1641 if (!ventoy_is_efi_os())
1642 {
1643 grub_file_seek(file, boot_catlog * 2048);
1644 grub_file_read(file, chain->boot_catalog_sector, sizeof(chain->boot_catalog_sector));
1645 }
1646
1647 /* part 3: image chunk */
1648 chain->img_chunk_offset = sizeof(ventoy_chain_head);
1649 chain->img_chunk_num = g_img_chunk_list.cur_chunk;
1650 grub_memcpy((char *)chain + chain->img_chunk_offset, g_img_chunk_list.chunk, img_chunk_size);
1651
1652 if (ventoy_compatible || unknown_image)
1653 {
1654 if (g_suppress_wincd_override_offset > 0)
1655 {
1656 chain->override_chunk_offset = chain->img_chunk_offset + img_chunk_size;
1657 chain->override_chunk_num = 1;
1658 ventoy_fill_suppress_wincd_override_data((char *)chain + chain->override_chunk_offset);
1659 }
1660
1661 return 0;
1662 }
1663
1664 if (0 == g_wim_valid_patch_count)
1665 {
1666 return 0;
1667 }
1668
1669 /* part 4: override chunk */
1670 chain->override_chunk_offset = chain->img_chunk_offset + img_chunk_size;
1671 chain->override_chunk_num = ventoy_get_override_chunk_num();
1672
1673 if (g_iso_fs_type == 0)
1674 {
1675 ventoy_windows_fill_override_data_iso9660(isosize, (char *)chain + chain->override_chunk_offset);
1676 }
1677 else
1678 {
1679 ventoy_windows_fill_override_data_udf(file, (char *)chain + chain->override_chunk_offset);
1680 }
1681
1682 /* part 5: virt chunk */
1683 chain->virt_chunk_offset = chain->override_chunk_offset + override_size;
1684 chain->virt_chunk_num = g_wim_valid_patch_count;
1685 ventoy_windows_fill_virt_data(isosize, chain);
1686
1687 if (ventoy_is_efi_os() == 0)
1688 {
1689 ventoy_windows_drive_map(chain);
1690 }
1691
1692 VENTOY_CMD_RETURN(GRUB_ERR_NONE);
1693 }
1694
1695 static grub_uint32_t ventoy_get_wim_iso_offset(const char *filepath)
1696 {
1697 grub_uint32_t imgoffset;
1698 grub_file_t file;
1699 char cmdbuf[128];
1700
1701 grub_snprintf(cmdbuf, sizeof(cmdbuf), "loopback wimiso \"%s\"", filepath);
1702 grub_script_execute_sourcecode(cmdbuf);
1703
1704 file = ventoy_grub_file_open(VENTOY_FILE_TYPE, "%s", "(wimiso)/boot/boot.wim");
1705 if (!file)
1706 {
1707 grub_printf("Failed to open boot.wim file in the image file\n");
1708 return 0;
1709 }
1710
1711 imgoffset = (grub_uint32_t)grub_iso9660_get_last_file_dirent_pos(file) + 2;
1712
1713 debug("wimiso wim direct offset: %u\n", imgoffset);
1714
1715 grub_file_close(file);
1716
1717 grub_script_execute_sourcecode("loopback -d wimiso");
1718
1719 return imgoffset;
1720 }
1721
1722 static int ventoy_get_wim_chunklist(const char *filename, ventoy_img_chunk_list *wimchunk, grub_uint64_t *wimsize)
1723 {
1724 grub_file_t wimfile;
1725
1726 wimfile = ventoy_grub_file_open(VENTOY_FILE_TYPE, "%s", filename);
1727 if (!wimfile)
1728 {
1729 return 1;
1730 }
1731
1732 grub_memset(wimchunk, 0, sizeof(ventoy_img_chunk_list));
1733 wimchunk->chunk = grub_malloc(sizeof(ventoy_img_chunk) * DEFAULT_CHUNK_NUM);
1734 if (NULL == wimchunk->chunk)
1735 {
1736 grub_file_close(wimfile);
1737 return grub_error(GRUB_ERR_OUT_OF_MEMORY, "Can't allocate image chunk memoty\n");
1738 }
1739
1740 wimchunk->max_chunk = DEFAULT_CHUNK_NUM;
1741 wimchunk->cur_chunk = 0;
1742
1743 ventoy_get_block_list(wimfile, wimchunk, wimfile->device->disk->partition->start);
1744
1745 *wimsize = wimfile->size;
1746 grub_file_close(wimfile);
1747
1748 return 0;
1749 }
1750
1751 grub_err_t ventoy_cmd_wim_check_bootable(grub_extcmd_context_t ctxt, int argc, char **args)
1752 {
1753 grub_uint32_t boot_index;
1754 grub_file_t file = NULL;
1755 wim_header *wimhdr = NULL;
1756
1757 (void)ctxt;
1758 (void)argc;
1759
1760 wimhdr = grub_zalloc(sizeof(wim_header));
1761 if (!wimhdr)
1762 {
1763 return 1;
1764 }
1765
1766 file = ventoy_grub_file_open(VENTOY_FILE_TYPE, "%s", args[0]);
1767 if (!file)
1768 {
1769 grub_free(wimhdr);
1770 return 1;
1771 }
1772
1773 grub_file_read(file, wimhdr, sizeof(wim_header));
1774 grub_file_close(file);
1775 boot_index = wimhdr->boot_index;
1776 grub_free(wimhdr);
1777
1778 if (boot_index == 0)
1779 {
1780 return 1;
1781 }
1782
1783 VENTOY_CMD_RETURN(GRUB_ERR_NONE);
1784 }
1785
1786 grub_err_t ventoy_cmd_wim_chain_data(grub_extcmd_context_t ctxt, int argc, char **args)
1787 {
1788 grub_uint32_t i = 0;
1789 grub_uint32_t imgoffset = 0;
1790 grub_uint32_t size = 0;
1791 grub_uint32_t isosector = 0;
1792 grub_uint64_t wimsize = 0;
1793 grub_uint32_t boot_catlog = 0;
1794 grub_uint32_t img_chunk1_size = 0;
1795 grub_uint32_t img_chunk2_size = 0;
1796 grub_uint32_t override_size = 0;
1797 grub_file_t file;
1798 grub_disk_t disk;
1799 const char *pLastChain = NULL;
1800 ventoy_chain_head *chain;
1801 ventoy_iso9660_override *dirent;
1802 ventoy_img_chunk *chunknode;
1803 ventoy_override_chunk *override;
1804 ventoy_img_chunk_list wimchunk;
1805 char envbuf[128];
1806
1807 (void)ctxt;
1808 (void)argc;
1809
1810 debug("wim chain data begin <%s> ...\n", args[0]);
1811
1812 if (NULL == g_wimiso_chunk_list.chunk || NULL == g_wimiso_path)
1813 {
1814 grub_printf("ventoy not ready\n");
1815 return 1;
1816 }
1817
1818 imgoffset = ventoy_get_wim_iso_offset(g_wimiso_path);
1819 if (imgoffset == 0)
1820 {
1821 grub_printf("image offset not found\n");
1822 return 1;
1823 }
1824
1825 if (0 != ventoy_get_wim_chunklist(args[0], &wimchunk, &wimsize))
1826 {
1827 grub_printf("Failed to get wim chunklist\n");
1828 return 1;
1829 }
1830
1831 file = ventoy_grub_file_open(VENTOY_FILE_TYPE, "%s", g_wimiso_path);
1832 if (!file)
1833 {
1834 return 1;
1835 }
1836
1837 boot_catlog = ventoy_get_iso_boot_catlog(file);
1838
1839 img_chunk1_size = g_wimiso_chunk_list.cur_chunk * sizeof(ventoy_img_chunk);
1840 img_chunk2_size = wimchunk.cur_chunk * sizeof(ventoy_img_chunk);
1841 override_size = sizeof(ventoy_override_chunk);
1842
1843 size = sizeof(ventoy_chain_head) + img_chunk1_size + img_chunk2_size + override_size;
1844
1845 pLastChain = grub_env_get("vtoy_chain_mem_addr");
1846 if (pLastChain)
1847 {
1848 chain = (ventoy_chain_head *)grub_strtoul(pLastChain, NULL, 16);
1849 if (chain)
1850 {
1851 debug("free last chain memory %p\n", chain);
1852 grub_free(chain);
1853 }
1854 }
1855
1856 chain = grub_malloc(size);
1857 if (!chain)
1858 {
1859 grub_printf("Failed to alloc chain memory size %u\n", size);
1860 grub_file_close(file);
1861 return 1;
1862 }
1863
1864 grub_snprintf(envbuf, sizeof(envbuf), "0x%lx", (unsigned long)chain);
1865 grub_env_set("vtoy_chain_mem_addr", envbuf);
1866 grub_snprintf(envbuf, sizeof(envbuf), "%u", size);
1867 grub_env_set("vtoy_chain_mem_size", envbuf);
1868
1869 grub_memset(chain, 0, sizeof(ventoy_chain_head));
1870
1871 /* part 1: os parameter */
1872 g_ventoy_chain_type = ventoy_chain_wim;
1873 ventoy_fill_os_param(file, &(chain->os_param));
1874
1875 /* part 2: chain head */
1876 disk = file->device->disk;
1877 chain->disk_drive = disk->id;
1878 chain->disk_sector_size = (1 << disk->log_sector_size);
1879 chain->real_img_size_in_bytes = ventoy_align_2k(file->size) + ventoy_align_2k(wimsize);
1880 chain->virt_img_size_in_bytes = chain->real_img_size_in_bytes;
1881 chain->boot_catalog = boot_catlog;
1882
1883 if (!ventoy_is_efi_os())
1884 {
1885 grub_file_seek(file, boot_catlog * 2048);
1886 grub_file_read(file, chain->boot_catalog_sector, sizeof(chain->boot_catalog_sector));
1887 }
1888
1889 /* part 3: image chunk */
1890 chain->img_chunk_offset = sizeof(ventoy_chain_head);
1891 chain->img_chunk_num = g_wimiso_chunk_list.cur_chunk + wimchunk.cur_chunk;
1892 grub_memcpy((char *)chain + chain->img_chunk_offset, g_wimiso_chunk_list.chunk, img_chunk1_size);
1893
1894 /* fs cluster size >= 2048, so don't need to proc align */
1895
1896 /* align by 2048 */
1897 chunknode = wimchunk.chunk + wimchunk.cur_chunk - 1;
1898 i = (chunknode->disk_end_sector + 1 - chunknode->disk_start_sector) % 4;
1899 if (i)
1900 {
1901 chunknode->disk_end_sector += 4 - i;
1902 }
1903
1904 isosector = (grub_uint32_t)((file->size + 2047) / 2048);
1905 for (i = 0; i < wimchunk.cur_chunk; i++)
1906 {
1907 chunknode = wimchunk.chunk + i;
1908 chunknode->img_start_sector = isosector;
1909 chunknode->img_end_sector = chunknode->img_start_sector +
1910 ((chunknode->disk_end_sector + 1 - chunknode->disk_start_sector) / 4) - 1;
1911 isosector = chunknode->img_end_sector + 1;
1912 }
1913
1914 grub_memcpy((char *)chain + chain->img_chunk_offset + img_chunk1_size, wimchunk.chunk, img_chunk2_size);
1915
1916 /* part 4: override chunk */
1917 chain->override_chunk_offset = chain->img_chunk_offset + img_chunk1_size + img_chunk2_size;
1918 chain->override_chunk_num = 1;
1919
1920 override = (ventoy_override_chunk *)((char *)chain + chain->override_chunk_offset);
1921 override->img_offset = imgoffset;
1922 override->override_size = sizeof(ventoy_iso9660_override);
1923
1924 dirent = (ventoy_iso9660_override *)(override->override_data);
1925 dirent->first_sector = (grub_uint32_t)((file->size + 2047) / 2048);
1926 dirent->size = (grub_uint32_t)(wimsize);
1927 dirent->first_sector_be = grub_swap_bytes32(dirent->first_sector);
1928 dirent->size_be = grub_swap_bytes32(dirent->size);
1929
1930 debug("imgoffset=%u first_sector=0x%x size=0x%x\n", imgoffset, dirent->first_sector, dirent->size);
1931
1932 if (ventoy_is_efi_os() == 0)
1933 {
1934 ventoy_windows_drive_map(chain);
1935 }
1936
1937 grub_file_close(file);
1938
1939 VENTOY_CMD_RETURN(GRUB_ERR_NONE);
1940 }
1941
1942 int ventoy_chain_file_size(const char *path)
1943 {
1944 int size;
1945 grub_file_t file;
1946
1947 file = grub_file_open(path, VENTOY_FILE_TYPE);
1948 size = (int)(file->size);
1949
1950 grub_file_close(file);
1951
1952 return size;
1953 }
1954
1955 int ventoy_chain_file_read(const char *path, int offset, int len, void *buf)
1956 {
1957 int size;
1958 grub_file_t file;
1959
1960 file = grub_file_open(path, VENTOY_FILE_TYPE);
1961 grub_file_seek(file, offset);
1962 size = grub_file_read(file, buf, len);
1963 grub_file_close(file);
1964
1965 return size;
1966 }
1967