+static uint64_t vtoydm_get_part_start(const char *diskname, int part)
+{
+ int fd;
+ unsigned long long size = 0;
+ char diskpath[256] = {0};
+ char sizebuf[64] = {0};
+
+ if (strstr(diskname, "nvme") || strstr(diskname, "mmc") || strstr(diskname, "nbd"))
+ {
+ snprintf(diskpath, sizeof(diskpath) - 1, "/sys/class/block/%sp%d/start", diskname, part);
+ }
+ else
+ {
+ snprintf(diskpath, sizeof(diskpath) - 1, "/sys/class/block/%s%d/start", diskname, part);
+ }
+
+ if (access(diskpath, F_OK) >= 0)
+ {
+ debug("get part start from sysfs for %s %d\n", diskname, part);
+
+ fd = open(diskpath, O_RDONLY | O_BINARY);
+ if (fd >= 0)
+ {
+ read(fd, sizebuf, sizeof(sizebuf));
+ size = strtoull(sizebuf, NULL, 10);
+ close(fd);
+ return size;
+ }
+ }
+ else
+ {
+ debug("%s not exist \n", diskpath);
+ }
+
+ return size;
+}
+
+static uint64_t vtoydm_get_part_secnum(const char *diskname, int part)
+{
+ int fd;
+ unsigned long long size = 0;
+ char diskpath[256] = {0};
+ char sizebuf[64] = {0};
+
+ diskname += 5; /* skip /dev/ */
+
+ if (strstr(diskname, "nvme") || strstr(diskname, "mmc") || strstr(diskname, "nbd"))
+ {
+ snprintf(diskpath, sizeof(diskpath) - 1, "/sys/class/block/%sp%d/size", diskname, part);
+ }
+ else
+ {
+ snprintf(diskpath, sizeof(diskpath) - 1, "/sys/class/block/%s%d/size", diskname, part);
+ }
+
+ if (access(diskpath, F_OK) >= 0)
+ {
+ debug("get part size from sysfs for %s %d\n", diskname, part);
+
+ fd = open(diskpath, O_RDONLY | O_BINARY);
+ if (fd >= 0)
+ {
+ read(fd, sizebuf, sizeof(sizebuf));
+ size = strtoull(sizebuf, NULL, 10);
+ close(fd);
+ return size;
+ }
+ }
+ else
+ {
+ debug("%s not exist \n", diskpath);
+ }
+
+ return size;
+}
+
+static int vtoydm_vlnk_convert(char *disk, int len, int *part, uint64_t *offset)
+{
+ int rc = 1;
+ int cnt = 0;
+ int rdlen;
+ FILE *fp = NULL;
+ ventoy_os_param param;
+ char diskname[128] = {0};
+
+ fp = fopen("/ventoy/ventoy_os_param", "rb");
+ if (!fp)
+ {
+ debug("dm vlnk convert not exist %d\n", errno);
+ goto end;
+ }
+
+ memset(¶m, 0, sizeof(param));
+ rdlen = (int)fread(¶m, 1, sizeof(param), fp);
+ if (rdlen != (int)sizeof(param))
+ {
+ debug("fread failed %d %d\n", rdlen, errno);
+ goto end;
+ }
+
+ debug("dm vlnk convert vtoy_reserved=%d\n", param.vtoy_reserved[6]);
+
+ if (param.vtoy_reserved[6])
+ {
+ cnt = vtoy_find_disk_by_guid(¶m, diskname);
+ debug("vtoy_find_disk_by_guid cnt=%d\n", cnt);
+ if (cnt == 1)
+ {
+ *part = param.vtoy_disk_part_id;
+ *offset = vtoydm_get_part_start(diskname, *part);
+
+ debug("VLNK <%s> <%s> <P%d> <%llu>\n", disk, diskname, *part, (unsigned long long)(*offset));
+
+ snprintf(disk, len, "/dev/%s", diskname);
+
+ rc = 0;
+ }
+ }
+
+end:
+ if (fp)
+ fclose(fp);
+ return rc;
+}
+
+static int vtoydm_print_raw_linear_table(const char *img_map_file, const char *diskname, int part)
+{
+ uint64_t disk_sector_num;
+
+ disk_sector_num = vtoydm_get_part_secnum(diskname, part);
+
+ if (strstr(diskname, "nvme") || strstr(diskname, "mmc") || strstr(diskname, "nbd"))
+ {
+ printf("0 %lu linear %sp%d 0\n", (unsigned long)disk_sector_num, diskname, part);
+ }
+ else
+ {
+ printf("0 %lu linear %s%d 0\n", (unsigned long)disk_sector_num, diskname, part);
+ }
+
+ return 0;
+}
+