cmd_disk.c 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128
  1. /*
  2. * (C) Copyright 2000-2011
  3. * Wolfgang Denk, DENX Software Engineering, wd@denx.de.
  4. *
  5. * SPDX-License-Identifier: GPL-2.0+
  6. */
  7. #include <common.h>
  8. #include <command.h>
  9. #include <part.h>
  10. #if defined(CONFIG_CMD_IDE) || defined(CONFIG_CMD_SCSI) || \
  11. defined(CONFIG_USB_STORAGE)
  12. int common_diskboot(cmd_tbl_t *cmdtp, const char *intf, int argc,
  13. char *const argv[])
  14. {
  15. int dev, part;
  16. ulong addr = CONFIG_SYS_LOAD_ADDR;
  17. ulong cnt;
  18. disk_partition_t info;
  19. image_header_t *hdr;
  20. block_dev_desc_t *dev_desc;
  21. #if defined(CONFIG_FIT)
  22. const void *fit_hdr = NULL;
  23. #endif
  24. bootstage_mark(BOOTSTAGE_ID_IDE_START);
  25. if (argc > 3) {
  26. bootstage_error(BOOTSTAGE_ID_IDE_ADDR);
  27. return CMD_RET_USAGE;
  28. }
  29. bootstage_mark(BOOTSTAGE_ID_IDE_ADDR);
  30. if (argc > 1)
  31. addr = simple_strtoul(argv[1], NULL, 16);
  32. bootstage_mark(BOOTSTAGE_ID_IDE_BOOT_DEVICE);
  33. part = get_device_and_partition(intf, (argc == 3) ? argv[2] : NULL,
  34. &dev_desc, &info, 1);
  35. if (part < 0) {
  36. bootstage_error(BOOTSTAGE_ID_IDE_TYPE);
  37. return 1;
  38. }
  39. dev = dev_desc->dev;
  40. bootstage_mark(BOOTSTAGE_ID_IDE_TYPE);
  41. printf("\nLoading from %s device %d, partition %d: "
  42. "Name: %.32s Type: %.32s\n", intf, dev, part, info.name,
  43. info.type);
  44. debug("First Block: " LBAFU ", # of blocks: " LBAFU
  45. ", Block Size: %ld\n",
  46. info.start, info.size, info.blksz);
  47. if (dev_desc->block_read(dev, info.start, 1, (ulong *) addr) != 1) {
  48. printf("** Read error on %d:%d\n", dev, part);
  49. bootstage_error(BOOTSTAGE_ID_IDE_PART_READ);
  50. return 1;
  51. }
  52. bootstage_mark(BOOTSTAGE_ID_IDE_PART_READ);
  53. switch (genimg_get_format((void *) addr)) {
  54. case IMAGE_FORMAT_LEGACY:
  55. hdr = (image_header_t *) addr;
  56. bootstage_mark(BOOTSTAGE_ID_IDE_FORMAT);
  57. if (!image_check_hcrc(hdr)) {
  58. puts("\n** Bad Header Checksum **\n");
  59. bootstage_error(BOOTSTAGE_ID_IDE_CHECKSUM);
  60. return 1;
  61. }
  62. bootstage_mark(BOOTSTAGE_ID_IDE_CHECKSUM);
  63. image_print_contents(hdr);
  64. cnt = image_get_image_size(hdr);
  65. break;
  66. #if defined(CONFIG_FIT)
  67. case IMAGE_FORMAT_FIT:
  68. fit_hdr = (const void *) addr;
  69. puts("Fit image detected...\n");
  70. cnt = fit_get_size(fit_hdr);
  71. break;
  72. #endif
  73. default:
  74. bootstage_error(BOOTSTAGE_ID_IDE_FORMAT);
  75. puts("** Unknown image type\n");
  76. return 1;
  77. }
  78. cnt += info.blksz - 1;
  79. cnt /= info.blksz;
  80. cnt -= 1;
  81. if (dev_desc->block_read(dev, info.start + 1, cnt,
  82. (ulong *)(addr + info.blksz)) != cnt) {
  83. printf("** Read error on %d:%d\n", dev, part);
  84. bootstage_error(BOOTSTAGE_ID_IDE_READ);
  85. return 1;
  86. }
  87. bootstage_mark(BOOTSTAGE_ID_IDE_READ);
  88. #if defined(CONFIG_FIT)
  89. /* This cannot be done earlier,
  90. * we need complete FIT image in RAM first */
  91. if (genimg_get_format((void *) addr) == IMAGE_FORMAT_FIT) {
  92. if (!fit_check_format(fit_hdr)) {
  93. bootstage_error(BOOTSTAGE_ID_IDE_FIT_READ);
  94. puts("** Bad FIT image format\n");
  95. return 1;
  96. }
  97. bootstage_mark(BOOTSTAGE_ID_IDE_FIT_READ_OK);
  98. fit_print_contents(fit_hdr);
  99. }
  100. #endif
  101. flush_cache(addr, (cnt+1)*info.blksz);
  102. /* Loading ok, update default load address */
  103. load_addr = addr;
  104. return bootm_maybe_autostart(cmdtp, argv[0]);
  105. }
  106. #endif