1 /******************************************************************************
2 * vtoydump.c ---- Dump ventoy os parameters
4 * Copyright (c) 2020, longpanda <admin@ventoy.net>
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.
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.
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/>.
27 #include <sys/types.h>
29 #include <sys/ioctl.h>
31 #include <sys/types.h>
35 #define IS_DIGIT(x) ((x) >= '0' && (x) <= '9')
38 typedef unsigned long long uint64_t;
39 typedef unsigned int uint32_t;
40 typedef unsigned short uint16_t;
41 typedef unsigned char uint8_t;
44 #define VENTOY_GUID { 0x77772020, 0x2e77, 0x6576, { 0x6e, 0x74, 0x6f, 0x79, 0x2e, 0x6e, 0x65, 0x74 }}
46 typedef enum ventoy_fs_type
48 ventoy_fs_exfat
= 0, /* 0: exfat */
49 ventoy_fs_ntfs
, /* 1: NTFS */
50 ventoy_fs_ext
, /* 2: ext2/ext3/ext4 */
51 ventoy_fs_xfs
, /* 3: XFS */
52 ventoy_fs_udf
, /* 4: UDF */
53 ventoy_fs_fat
, /* 5: FAT */
60 typedef struct ventoy_guid
69 typedef struct ventoy_image_disk_region
71 uint32_t image_sector_count
; /* image sectors contained in this region */
72 uint32_t image_start_sector
; /* image sector start */
73 uint64_t disk_start_sector
; /* disk sector start */
74 }ventoy_image_disk_region
;
76 typedef struct ventoy_image_location
80 /* image sector size, currently this value is always 2048 */
81 uint32_t image_sector_size
;
83 /* disk sector size, normally the value is 512 */
84 uint32_t disk_sector_size
;
86 uint32_t region_count
;
90 * If the image file has more than one fragments in disk,
91 * there will be more than one region data here.
92 * You can calculate the region count by
94 ventoy_image_disk_region regions
[1];
96 /* ventoy_image_disk_region regions[2~region_count-1] */
97 }ventoy_image_location
;
99 typedef struct ventoy_os_param
101 ventoy_guid guid
; // VENTOY_GUID
102 uint8_t chksum
; // checksum
104 uint8_t vtoy_disk_guid
[16];
105 uint64_t vtoy_disk_size
; // disk size in bytes
106 uint16_t vtoy_disk_part_id
; // begin with 1
107 uint16_t vtoy_disk_part_type
; // 0:exfat 1:ntfs other: reserved
108 char vtoy_img_path
[384]; // It seems to be enough, utf-8 format
109 uint64_t vtoy_img_size
; // image file size in bytes
112 * Ventoy will write a copy of ventoy_image_location data into runtime memory
113 * this is the physically address and length of that memory.
114 * Address 0 means no such data exist.
115 * Address will be aligned by 4KB.
118 uint64_t vtoy_img_location_addr
;
119 uint32_t vtoy_img_location_len
;
121 uint64_t vtoy_reserved
[4]; // Internal use by ventoy
123 uint8_t vtoy_disk_signature
[4];
125 uint8_t reserved
[27];
133 #if defined(_dragon_fly) || defined(_free_BSD) || defined(_QNX)
134 #define MMAP_FLAGS MAP_SHARED
136 #define MMAP_FLAGS MAP_PRIVATE
139 #define SEARCH_MEM_START 0x80000
140 #define SEARCH_MEM_LEN 0x1c000
142 static int verbose
= 0;
143 #define debug(fmt, ...) if(verbose) printf(fmt, ##__VA_ARGS__)
145 static ventoy_guid vtoy_guid
= VENTOY_GUID
;
147 static const char *g_ventoy_fs
[ventoy_fs_max
] =
149 "exfat", "ntfs", "ext*", "xfs", "udf", "fat"
152 static int vtoy_check_os_param(ventoy_os_param
*param
)
156 uint8_t *buf
= (uint8_t *)param
;
158 if (memcmp(¶m
->guid
, &vtoy_guid
, sizeof(ventoy_guid
)))
160 uint8_t *data1
= (uint8_t *)(¶m
->guid
);
161 uint8_t *data2
= (uint8_t *)(&vtoy_guid
);
163 for (i
= 0; i
< 16; i
++)
165 if (data1
[i
] != data2
[i
])
167 debug("guid not equal i = %u, 0x%02x, 0x%02x\n", i
, data1
[i
], data2
[i
]);
173 for (i
= 0; i
< sizeof(ventoy_os_param
); i
++)
180 debug("Invalid checksum 0x%02x\n", chksum
);
187 static int vtoy_os_param_from_file(const char *filename
, ventoy_os_param
*param
)
192 fd
= open(filename
, O_RDONLY
| O_BINARY
);
195 fprintf(stderr
, "Failed to open file %s error %d\n", filename
, errno
);
199 read(fd
, param
, sizeof(ventoy_os_param
));
201 if (vtoy_check_os_param(param
) == 0)
203 debug("find ventoy os param in file %s\n", filename
);
207 debug("ventoy os pararm NOT found in file %s\n", filename
);
215 static void vtoy_dump_os_param(ventoy_os_param
*param
)
217 printf("################# dump os param ################\n");
219 printf("param->chksum = 0x%x\n", param
->chksum
);
220 printf("param->vtoy_disk_guid = %02x %02x %02x %02x\n",
221 param
->vtoy_disk_guid
[0], param
->vtoy_disk_guid
[1],
222 param
->vtoy_disk_guid
[2], param
->vtoy_disk_guid
[3]);
224 printf("param->vtoy_disk_signature = %02x %02x %02x %02x\n",
225 param
->vtoy_disk_signature
[0], param
->vtoy_disk_signature
[1],
226 param
->vtoy_disk_signature
[2], param
->vtoy_disk_signature
[3]);
228 printf("param->vtoy_disk_size = %llu\n", (unsigned long long)param
->vtoy_disk_size
);
229 printf("param->vtoy_disk_part_id = %u\n", param
->vtoy_disk_part_id
);
230 printf("param->vtoy_disk_part_type = %u\n", param
->vtoy_disk_part_type
);
231 printf("param->vtoy_img_path = <%s>\n", param
->vtoy_img_path
);
232 printf("param->vtoy_img_size = <%llu>\n", (unsigned long long)param
->vtoy_img_size
);
233 printf("param->vtoy_img_location_addr = <0x%llx>\n", (unsigned long long)param
->vtoy_img_location_addr
);
234 printf("param->vtoy_img_location_len = <%u>\n", param
->vtoy_img_location_len
);
235 printf("param->vtoy_reserved[0] = 0x%llx\n", (unsigned long long)param
->vtoy_reserved
[0]);
236 printf("param->vtoy_reserved[1] = 0x%llx\n", (unsigned long long)param
->vtoy_reserved
[1]);
241 static int vtoy_get_disk_guid(const char *diskname
, uint8_t *vtguid
, uint8_t *vtsig
)
245 char devdisk
[128] = {0};
247 snprintf(devdisk
, sizeof(devdisk
) - 1, "/dev/%s", diskname
);
249 fd
= open(devdisk
, O_RDONLY
| O_BINARY
);
252 lseek(fd
, 0x180, SEEK_SET
);
253 read(fd
, vtguid
, 16);
255 lseek(fd
, 0x1b8, SEEK_SET
);
259 debug("GUID for %s: <", devdisk
);
260 for (i
= 0; i
< 16; i
++)
262 debug("%02x", vtguid
[i
]);
270 debug("failed to open %s %d\n", devdisk
, errno
);
275 static unsigned long long vtoy_get_disk_size_in_byte(const char *disk
)
279 unsigned long long size
= 0;
280 char diskpath
[256] = {0};
281 char sizebuf
[64] = {0};
283 // Try 1: get size from sysfs
284 snprintf(diskpath
, sizeof(diskpath
) - 1, "/sys/block/%s/size", disk
);
285 if (access(diskpath
, F_OK
) >= 0)
287 debug("get disk size from sysfs for %s\n", disk
);
289 fd
= open(diskpath
, O_RDONLY
| O_BINARY
);
292 read(fd
, sizebuf
, sizeof(sizebuf
));
293 size
= strtoull(sizebuf
, NULL
, 10);
300 debug("%s not exist \n", diskpath
);
303 // Try 2: get size from ioctl
304 snprintf(diskpath
, sizeof(diskpath
) - 1, "/dev/%s", disk
);
305 fd
= open(diskpath
, O_RDONLY
);
308 debug("get disk size from ioctl for %s\n", disk
);
309 rc
= ioctl(fd
, BLKGETSIZE64
, &size
);
313 debug("failed to ioctl %d\n", rc
);
319 debug("failed to open %s %d\n", diskpath
, errno
);
322 debug("disk %s size %llu bytes\n", disk
, (unsigned long long)size
);
326 static int vtoy_is_possible_blkdev(const char *name
)
334 if (name
[0] == 'r' && name
[1] == 'a' && name
[2] == 'm')
340 if (name
[0] == 'l' && name
[1] == 'o' && name
[2] == 'o' && name
[3] == 'p')
346 if (name
[0] == 'd' && name
[1] == 'm' && name
[2] == '-' && IS_DIGIT(name
[3]))
352 if (name
[0] == 's' && name
[1] == 'r' && IS_DIGIT(name
[2]))
360 static int vtoy_find_disk_by_size(unsigned long long size
, char *diskname
)
362 unsigned long long cursize
= 0;
364 struct dirent
* p
= NULL
;
367 dir
= opendir("/sys/block");
373 while ((p
= readdir(dir
)) != NULL
)
375 if (!vtoy_is_possible_blkdev(p
->d_name
))
377 debug("disk %s is filted by name\n", p
->d_name
);
381 cursize
= vtoy_get_disk_size_in_byte(p
->d_name
);
382 debug("disk %s size %llu\n", p
->d_name
, (unsigned long long)cursize
);
385 sprintf(diskname
, "%s", p
->d_name
);
393 static int vtoy_find_disk_by_guid(ventoy_os_param
*param
, char *diskname
)
398 struct dirent
* p
= NULL
;
402 dir
= opendir("/sys/block");
408 while ((p
= readdir(dir
)) != NULL
)
410 if (!vtoy_is_possible_blkdev(p
->d_name
))
412 debug("disk %s is filted by name\n", p
->d_name
);
416 memset(vtguid
, 0, sizeof(vtguid
));
417 rc
= vtoy_get_disk_guid(p
->d_name
, vtguid
, vtsig
);
418 if (rc
== 0 && memcmp(vtguid
, param
->vtoy_disk_guid
, 16) == 0 &&
419 memcmp(vtsig
, param
->vtoy_disk_signature
, 4) == 0)
421 sprintf(diskname
, "%s", p
->d_name
);
430 static int vtoy_printf_iso_path(ventoy_os_param
*param
)
432 printf("%s\n", param
->vtoy_img_path
);
436 static int vtoy_print_os_param(ventoy_os_param
*param
, char *diskname
)
439 char *path
= param
->vtoy_img_path
;
442 cnt
= vtoy_find_disk_by_size(param
->vtoy_disk_size
, diskname
);
445 cnt
= vtoy_find_disk_by_guid(param
, diskname
);
449 cnt
= vtoy_find_disk_by_guid(param
, diskname
);
450 debug("find 0 disk by size, try with guid cnt=%d...\n", cnt
);
453 if (param
->vtoy_disk_part_type
< ventoy_fs_max
)
455 fs
= g_ventoy_fs
[param
->vtoy_disk_part_type
];
464 printf("/dev/%s#%s#%s\n", diskname
, fs
, path
);
473 static int vtoy_check_device(ventoy_os_param
*param
, const char *device
)
475 unsigned long long size
;
476 uint8_t vtguid
[16] = {0};
477 uint8_t vtsig
[4] = {0};
479 debug("vtoy_check_device for <%s>\n", device
);
481 size
= vtoy_get_disk_size_in_byte(device
);
482 vtoy_get_disk_guid(device
, vtguid
, vtsig
);
484 debug("param->vtoy_disk_size=%llu size=%llu\n",
485 (unsigned long long)param
->vtoy_disk_size
, (unsigned long long)size
);
487 if (memcmp(vtguid
, param
->vtoy_disk_guid
, 16) == 0 &&
488 memcmp(vtsig
, param
->vtoy_disk_signature
, 4) == 0)
490 debug("<%s> is right ventoy disk\n", device
);
495 debug("<%s> is NOT right ventoy disk\n", device
);
501 * Find disk and image path from ventoy runtime data.
502 * By default data is read from phymem(legacy bios) or efivar(UEFI), if -f is input, data is read from file.
504 * -f datafile os param data file.
505 * -c /dev/xxx check ventoy disk
507 * -l also print image disk location
509 int vtoydump_main(int argc
, char **argv
)
514 char filename
[256] = {0};
515 char diskname
[256] = {0};
516 char device
[64] = {0};
517 ventoy_os_param
*param
= NULL
;
519 while ((ch
= getopt(argc
, argv
, "c:f:p:v::")) != -1)
523 strncpy(filename
, optarg
, sizeof(filename
) - 1);
531 strncpy(device
, optarg
, sizeof(device
) - 1);
536 strncpy(filename
, optarg
, sizeof(filename
) - 1);
540 fprintf(stderr
, "Usage: %s -f datafile [ -v ] \n", argv
[0]);
545 if (filename
[0] == 0)
547 fprintf(stderr
, "Usage: %s -f datafile [ -v ] \n", argv
[0]);
551 param
= malloc(sizeof(ventoy_os_param
));
554 fprintf(stderr
, "failed to alloc memory with size %d error %d\n",
555 (int)sizeof(ventoy_os_param
), errno
);
559 memset(param
, 0, sizeof(ventoy_os_param
));
561 debug("get os pararm from file %s\n", filename
);
562 rc
= vtoy_os_param_from_file(filename
, param
);
565 debug("ventoy os param not found %d %d\n", rc
, ENOENT
);
568 debug("now try with file %s\n", "/ventoy/ventoy_os_param");
569 rc
= vtoy_os_param_from_file("/ventoy/ventoy_os_param", param
);
583 vtoy_dump_os_param(param
);
588 rc
= vtoy_printf_iso_path(param
);
592 rc
= vtoy_check_device(param
, device
);
596 // print os param, you can change the output format in the function
597 rc
= vtoy_print_os_param(param
, diskname
);
609 #ifndef BUILD_VTOY_TOOL
610 int main(int argc
, char **argv
)
612 return vtoydump_main(argc
, argv
);