1 /* amiga.c - Read amiga partition tables (RDB). */
3 * GRUB -- GRand Unified Bootloader
4 * Copyright (C) 2002,2004,2005,2006,2007 Free Software Foundation, Inc.
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.
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.
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/>.
20 #include <grub/disk.h>
21 #include <grub/misc.h>
23 #include <grub/partition.h>
26 GRUB_MOD_LICENSE ("GPLv3+");
28 #define AMIGA_CHECKSUM_WORDS 128
30 struct grub_amiga_rdsk
33 grub_uint8_t magic
[4];
34 #define GRUB_AMIGA_RDSK_MAGIC "RDSK"
36 grub_int32_t checksum
;
37 grub_uint32_t scsihost
;
40 grub_uint32_t badblcklst
;
41 grub_uint32_t partitionlst
;
44 grub_uint32_t unused
[AMIGA_CHECKSUM_WORDS
- 9];
47 struct grub_amiga_partition
50 grub_uint8_t magic
[4];
51 #define GRUB_AMIGA_PART_MAGIC "PART"
53 grub_int32_t checksum
;
54 grub_uint32_t scsihost
;
57 grub_uint32_t unused1
[2];
58 grub_uint32_t devflags
;
60 grub_uint8_t name
[31];
61 grub_uint32_t unused2
[15];
63 grub_uint32_t unused3
[3];
65 grub_uint32_t unused4
;
66 grub_uint32_t block_per_track
;
67 grub_uint32_t unused5
[3];
69 grub_uint32_t highcyl
;
71 grub_uint32_t firstcyl
;
72 grub_uint32_t unused
[AMIGA_CHECKSUM_WORDS
- 44];
75 static struct grub_partition_map grub_amiga_partition_map
;
80 amiga_partition_map_checksum (void *buf
)
82 grub_uint32_t
*ptr
= buf
;
85 /* Fancy and quick way of checking sz >= 512 / 4 = 128. */
86 if (ptr
[1] & ~grub_cpu_to_be32_compile_time (AMIGA_CHECKSUM_WORDS
- 1))
87 sz
= AMIGA_CHECKSUM_WORDS
;
89 sz
= grub_be_to_cpu32 (ptr
[1]);
91 for (; sz
; sz
--, ptr
++)
92 r
+= grub_be_to_cpu32 (*ptr
);
98 amiga_partition_map_iterate (grub_disk_t disk
,
99 grub_partition_iterate_hook_t hook
,
102 struct grub_partition part
;
103 struct grub_amiga_rdsk rdsk
;
108 /* The RDSK block is one of the first 15 blocks. */
109 for (pos
= 0; pos
< 15; pos
++)
111 /* Read the RDSK block which is a descriptor for the entire disk. */
112 if (grub_disk_read (disk
, pos
, 0, sizeof (rdsk
), &rdsk
))
115 if (grub_memcmp (rdsk
.magic
, GRUB_AMIGA_RDSK_MAGIC
,
116 sizeof (rdsk
.magic
)) == 0
117 && amiga_partition_map_checksum (&rdsk
) == 0)
119 /* Found the first PART block. */
120 next
= grub_be_to_cpu32 (rdsk
.partitionlst
);
126 return grub_error (GRUB_ERR_BAD_PART_TABLE
,
127 "Amiga partition map not found");
129 /* The end of the partition list is marked using "-1". */
132 struct grub_amiga_partition apart
;
134 /* Read the RDSK block which is a descriptor for the entire disk. */
135 if (grub_disk_read (disk
, next
, 0, sizeof (apart
), &apart
))
138 if (grub_memcmp (apart
.magic
, GRUB_AMIGA_PART_MAGIC
,
139 sizeof (apart
.magic
)) != 0
140 || amiga_partition_map_checksum (&apart
) != 0)
141 return grub_error (GRUB_ERR_BAD_PART_TABLE
,
142 "invalid Amiga partition map");
143 /* Calculate the first block and the size of the partition. */
144 part
.start
= (grub_be_to_cpu32 (apart
.lowcyl
)
145 * grub_be_to_cpu32 (apart
.heads
)
146 * grub_be_to_cpu32 (apart
.block_per_track
));
147 part
.len
= ((grub_be_to_cpu32 (apart
.highcyl
)
148 - grub_be_to_cpu32 (apart
.lowcyl
) + 1)
149 * grub_be_to_cpu32 (apart
.heads
)
150 * grub_be_to_cpu32 (apart
.block_per_track
));
153 part
.number
= partno
;
155 part
.partmap
= &grub_amiga_partition_map
;
157 if (hook (disk
, &part
, hook_data
))
160 next
= grub_be_to_cpu32 (apart
.next
);
168 /* Partition map type. */
169 static struct grub_partition_map grub_amiga_partition_map
=
172 .iterate
= amiga_partition_map_iterate
,
175 GRUB_MOD_INIT(part_amiga
)
177 grub_partition_map_register (&grub_amiga_partition_map
);
180 GRUB_MOD_FINI(part_amiga
)
182 grub_partition_map_unregister (&grub_amiga_partition_map
);