1 /* acorn.c - Read Linux/ADFS partition tables. */
3 * GRUB -- GRand Unified Bootloader
4 * Copyright (C) 2005,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>
24 #include <grub/acorn_filecore.h>
26 GRUB_MOD_LICENSE ("GPLv3+");
28 #define LINUX_NATIVE_MAGIC grub_cpu_to_le32_compile_time (0xdeafa1de)
29 #define LINUX_SWAP_MAGIC grub_cpu_to_le32_compile_time (0xdeafab1e)
30 #define LINUX_MAP_ENTRIES (512 / 12)
32 #define NONADFS_PARTITION_TYPE_LINUX 9
33 #define NONADFS_PARTITION_TYPE_MASK 15
35 struct grub_acorn_boot_block
41 grub_uint8_t misc
[0x1C0];
42 struct grub_filecore_disc_record disc_record
;
44 grub_uint16_t start_cylinder
;
45 grub_uint8_t checksum
;
47 grub_uint8_t bin
[0x200];
58 static struct grub_partition_map grub_acorn_partition_map
;
61 acorn_partition_map_find (grub_disk_t disk
, struct linux_part
*m
,
62 grub_disk_addr_t
*sector
)
64 struct grub_acorn_boot_block boot
;
66 unsigned int checksum
= 0;
68 unsigned int sectors_per_cylinder
;
71 err
= grub_disk_read (disk
, 0xC00 / GRUB_DISK_SECTOR_SIZE
, 0,
72 sizeof (struct grub_acorn_boot_block
),
77 if ((boot
.flags
& NONADFS_PARTITION_TYPE_MASK
) != NONADFS_PARTITION_TYPE_LINUX
)
80 for (i
= 0; i
!= 0x1ff; ++i
)
81 checksum
= ((checksum
& 0xff) + (checksum
>> 8) + boot
.bin
[i
]);
83 if ((grub_uint8_t
) checksum
!= boot
.checksum
)
86 heads
= (boot
.disc_record
.heads
87 + ((boot
.disc_record
.lowsector
>> 6) & 1));
88 sectors_per_cylinder
= boot
.disc_record
.secspertrack
* heads
;
89 *sector
= grub_le_to_cpu16 (boot
.start_cylinder
) * sectors_per_cylinder
;
91 return grub_disk_read (disk
, *sector
, 0,
92 sizeof (struct linux_part
) * LINUX_MAP_ENTRIES
,
96 return grub_error (GRUB_ERR_BAD_PART_TABLE
,
97 "Linux/ADFS partition map not found");
103 acorn_partition_map_iterate (grub_disk_t disk
,
104 grub_partition_iterate_hook_t hook
,
107 struct grub_partition part
;
108 struct linux_part map
[LINUX_MAP_ENTRIES
];
110 grub_disk_addr_t sector
= 0;
113 err
= acorn_partition_map_find (disk
, map
, §or
);
117 part
.partmap
= &grub_acorn_partition_map
;
119 for (i
= 0; i
!= LINUX_MAP_ENTRIES
; ++i
)
121 if (map
[i
].magic
!= LINUX_NATIVE_MAGIC
122 && map
[i
].magic
!= LINUX_SWAP_MAGIC
)
123 return GRUB_ERR_NONE
;
125 part
.start
= sector
+ map
[i
].start
;
126 part
.len
= map
[i
].size
;
128 part
.number
= part
.index
= i
;
130 if (hook (disk
, &part
, hook_data
))
134 return GRUB_ERR_NONE
;
139 /* Partition map type. */
140 static struct grub_partition_map grub_acorn_partition_map
=
143 .iterate
= acorn_partition_map_iterate
,
146 GRUB_MOD_INIT(part_acorn
)
148 grub_partition_map_register (&grub_acorn_partition_map
);
151 GRUB_MOD_FINI(part_acorn
)
153 grub_partition_map_unregister (&grub_acorn_partition_map
);