amiga.c 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154
  1. /* amiga.c - Read amiga partition tables (RDB). */
  2. /*
  3. * GRUB -- GRand Unified Bootloader
  4. * Copyright (C) 2002,2004,2005,2006,2007 Free Software Foundation, Inc.
  5. *
  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.
  10. *
  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.
  15. *
  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/>.
  18. */
  19. #include <grub/disk.h>
  20. #include <grub/misc.h>
  21. #include <grub/mm.h>
  22. #include <grub/partition.h>
  23. #include <grub/dl.h>
  24. struct grub_amiga_rdsk
  25. {
  26. /* "RDSK". */
  27. grub_uint8_t magic[4];
  28. grub_uint32_t size;
  29. grub_int32_t checksum;
  30. grub_uint32_t scsihost;
  31. grub_uint32_t blksz;
  32. grub_uint32_t flags;
  33. grub_uint32_t badblcklst;
  34. grub_uint32_t partitionlst;
  35. grub_uint32_t fslst;
  36. /* The other information is not important for us. */
  37. } __attribute__ ((packed));
  38. struct grub_amiga_partition
  39. {
  40. /* "PART". */
  41. grub_uint8_t magic[4];
  42. grub_int32_t size;
  43. grub_int32_t checksum;
  44. grub_uint32_t scsihost;
  45. grub_uint32_t next;
  46. grub_uint32_t flags;
  47. grub_uint32_t unused1[2];
  48. grub_uint32_t devflags;
  49. grub_uint8_t namelen;
  50. grub_uint8_t name[31];
  51. grub_uint32_t unused2[15];
  52. grub_uint32_t unused3[3];
  53. grub_uint32_t heads;
  54. grub_uint32_t unused4;
  55. grub_uint32_t block_per_track;
  56. grub_uint32_t unused5[3];
  57. grub_uint32_t lowcyl;
  58. grub_uint32_t highcyl;
  59. grub_uint32_t firstcyl;
  60. } __attribute__ ((packed));
  61. static struct grub_partition_map grub_amiga_partition_map;
  62. static grub_err_t
  63. amiga_partition_map_iterate (grub_disk_t disk,
  64. int (*hook) (grub_disk_t disk,
  65. const grub_partition_t partition,
  66. void *closure),
  67. void *closure)
  68. {
  69. struct grub_partition part;
  70. struct grub_amiga_rdsk rdsk;
  71. int partno = 0;
  72. int next = -1;
  73. unsigned pos;
  74. /* The RDSK block is one of the first 15 blocks. */
  75. for (pos = 0; pos < 15; pos++)
  76. {
  77. /* Read the RDSK block which is a descriptor for the entire disk. */
  78. if (grub_disk_read (disk, pos, 0, sizeof (rdsk), &rdsk))
  79. return grub_errno;
  80. if (grub_strcmp ((char *) rdsk.magic, "RDSK") == 0)
  81. {
  82. /* Found the first PART block. */
  83. next = grub_be_to_cpu32 (rdsk.partitionlst);
  84. break;
  85. }
  86. }
  87. if (next == -1)
  88. return grub_error (GRUB_ERR_BAD_PART_TABLE,
  89. "Amiga partition map not found");
  90. /* The end of the partition list is marked using "-1". */
  91. while (next != -1)
  92. {
  93. struct grub_amiga_partition apart;
  94. /* Read the RDSK block which is a descriptor for the entire disk. */
  95. if (grub_disk_read (disk, next, 0, sizeof (apart), &apart))
  96. return grub_errno;
  97. /* Calculate the first block and the size of the partition. */
  98. part.start = (grub_be_to_cpu32 (apart.lowcyl)
  99. * grub_be_to_cpu32 (apart.heads)
  100. * grub_be_to_cpu32 (apart.block_per_track));
  101. part.len = ((grub_be_to_cpu32 (apart.highcyl)
  102. - grub_be_to_cpu32 (apart.lowcyl) + 1)
  103. * grub_be_to_cpu32 (apart.heads)
  104. * grub_be_to_cpu32 (apart.block_per_track));
  105. part.offset = (grub_off_t) next * 512;
  106. part.number = partno;
  107. part.index = 0;
  108. part.partmap = &grub_amiga_partition_map;
  109. if (hook (disk, &part, closure))
  110. return grub_errno;
  111. next = grub_be_to_cpu32 (apart.next);
  112. partno++;
  113. }
  114. return 0;
  115. }
  116. /* Partition map type. */
  117. static struct grub_partition_map grub_amiga_partition_map =
  118. {
  119. .name = "amiga",
  120. .iterate = amiga_partition_map_iterate,
  121. };
  122. GRUB_MOD_INIT(part_amiga)
  123. {
  124. grub_partition_map_register (&grub_amiga_partition_map);
  125. }
  126. GRUB_MOD_FINI(part_amiga)
  127. {
  128. grub_partition_map_unregister (&grub_amiga_partition_map);
  129. }