blob: 4282605344ea0bac677595c48ef169cca41aeaa3 [file] [log] [blame]
bigbiff bigbiffe60683a2013-02-22 20:55:50 -05001/*
2 * mac partitions parsing code
3 *
4 * Copyright (C) 2009 Karel Zak <kzak@redhat.com>
5 *
6 * This file may be redistributed under the terms of the
7 * GNU Lesser General Public License.
8 *
9 */
10#include <stdio.h>
11#include <string.h>
12#include <stdlib.h>
13#include <stdint.h>
14
15#include "partitions.h"
16
17#define MAC_PARTITION_MAGIC 0x504d
18#define MAC_PARTITION_MAGIC_OLD 0x5453
19
20/*
21 * Mac partition entry
22 * http://developer.apple.com/legacy/mac/library/documentation/mac/Devices/Devices-126.html
23 */
24struct mac_partition {
25 uint16_t signature; /* expected to be MAC_PARTITION_MAGIC */
26 uint16_t reserved; /* reserved */
27 uint32_t map_count; /* # blocks in partition map */
28 uint32_t start_block; /* absolute starting block # of partition */
29 uint32_t block_count; /* number of blocks in partition */
30 char name[32]; /* partition name */
31 char type[32]; /* string type description */
32 uint32_t data_start; /* rel block # of first data block */
33 uint32_t data_count; /* number of data blocks */
34 uint32_t status; /* partition status bits */
35 uint32_t boot_start; /* first logical block of boot code */
36 uint32_t boot_size; /* size of boot code, in bytes */
37 uint32_t boot_load; /* boot code load address */
38 uint32_t boot_load2; /* reserved */
39 uint32_t boot_entry; /* boot code entry point */
40 uint32_t boot_entry2; /* reserved */
41 uint32_t boot_cksum; /* boot code checksum */
42 char processor[16]; /* identifies ISA of boot */
43
44 /* there is more stuff after this that we don't need */
45} __attribute__((packed));
46
47/*
48 * Driver descriptor structure, in block 0
49 * http://developer.apple.com/legacy/mac/library/documentation/mac/Devices/Devices-121.html
50 */
51struct mac_driver_desc {
52 uint16_t signature; /* expected to be MAC_DRIVER_MAGIC */
53 uint16_t block_size; /* block size of the device */
54 uint32_t block_count; /* number of blocks on the device */
55
56 /* there is more stuff after this that we don't need */
57} __attribute__((packed));
58
59static inline unsigned char *get_mac_block(
60 blkid_probe pr,
61 uint16_t block_size,
62 uint32_t num)
63{
64 return blkid_probe_get_buffer(pr,
65 (blkid_loff_t) num * block_size, block_size);
66}
67
68static inline int has_part_signature(struct mac_partition *p)
69{
70 return be16_to_cpu(p->signature) == MAC_PARTITION_MAGIC ||
71 be16_to_cpu(p->signature) == MAC_PARTITION_MAGIC_OLD;
72}
73
74static int probe_mac_pt(blkid_probe pr,
75 const struct blkid_idmag *mag __attribute__((__unused__)))
76{
77 struct mac_driver_desc *md;
78 struct mac_partition *p;
79 blkid_parttable tab = NULL;
80 blkid_partlist ls;
81 uint16_t block_size;
82 uint16_t ssf; /* sector size fragment */
83 uint32_t nblks, i;
84
85
86 /* The driver descriptor record is always located at physical block 0,
87 * the first block on the disk.
88 */
89 md = (struct mac_driver_desc *) blkid_probe_get_sector(pr, 0);
bigbiff7b4c7a62015-01-01 19:44:14 -050090 if (!md) {
91 if (errno)
92 return -errno;
bigbiff bigbiffe60683a2013-02-22 20:55:50 -050093 goto nothing;
bigbiff7b4c7a62015-01-01 19:44:14 -050094 }
bigbiff bigbiffe60683a2013-02-22 20:55:50 -050095
96 block_size = be16_to_cpu(md->block_size);
97
98 /* The partition map always begins at physical block 1,
99 * the second block on the disk.
100 */
101 p = (struct mac_partition *) get_mac_block(pr, block_size, 1);
bigbiff7b4c7a62015-01-01 19:44:14 -0500102 if (!p) {
103 if (errno)
104 return -errno;
bigbiff bigbiffe60683a2013-02-22 20:55:50 -0500105 goto nothing;
bigbiff7b4c7a62015-01-01 19:44:14 -0500106 }
bigbiff bigbiffe60683a2013-02-22 20:55:50 -0500107
108 /* check the first partition signature */
109 if (!has_part_signature(p))
110 goto nothing;
111
112 if (blkid_partitions_need_typeonly(pr))
113 /* caller does not ask for details about partitions */
114 return 0;
115
116 ls = blkid_probe_get_partlist(pr);
117 if (!ls)
bigbiff7b4c7a62015-01-01 19:44:14 -0500118 goto nothing;
bigbiff bigbiffe60683a2013-02-22 20:55:50 -0500119
120 tab = blkid_partlist_new_parttable(ls, "mac", 0);
121 if (!tab)
122 goto err;
123
124 ssf = block_size / 512;
125 nblks = be32_to_cpu(p->map_count);
126
127 for (i = 1; i <= nblks; ++i) {
128 blkid_partition par;
129 uint32_t start;
130 uint32_t size;
131
132 p = (struct mac_partition *) get_mac_block(pr, block_size, i);
bigbiff7b4c7a62015-01-01 19:44:14 -0500133 if (!p) {
134 if (errno)
135 return -errno;
bigbiff bigbiffe60683a2013-02-22 20:55:50 -0500136 goto nothing;
bigbiff7b4c7a62015-01-01 19:44:14 -0500137 }
bigbiff bigbiffe60683a2013-02-22 20:55:50 -0500138 if (!has_part_signature(p))
139 goto nothing;
140
141 if (be32_to_cpu(p->map_count) != nblks) {
bigbiff7b4c7a62015-01-01 19:44:14 -0500142 DBG(LOWPROBE, ul_debug(
bigbiff bigbiffe60683a2013-02-22 20:55:50 -0500143 "mac: inconsisten map_count in partition map, "
bigbiff7b4c7a62015-01-01 19:44:14 -0500144 "entry[0]: %d, entry[%d]: %d",
bigbiff bigbiffe60683a2013-02-22 20:55:50 -0500145 nblks, i - 1,
146 be32_to_cpu(p->map_count)));
147 }
148
149 /*
150 * note that libparted ignores some mac partitions according to
151 * the partition name (e.g. "Apple_Free" or "Apple_Void"). We
152 * follows Linux kernel and all partitions are visible
153 */
154
155 start = be32_to_cpu(p->start_block) * ssf;
156 size = be32_to_cpu(p->block_count) * ssf;
157
158 par = blkid_partlist_add_partition(ls, tab, start, size);
159 if (!par)
160 goto err;
161
162 blkid_partition_set_name(par, (unsigned char *) p->name,
163 sizeof(p->name));
164
165 blkid_partition_set_type_string(par, (unsigned char *) p->type,
166 sizeof(p->type));
167 }
168
bigbiff7b4c7a62015-01-01 19:44:14 -0500169 return BLKID_PROBE_OK;
bigbiff bigbiffe60683a2013-02-22 20:55:50 -0500170
171nothing:
bigbiff7b4c7a62015-01-01 19:44:14 -0500172 return BLKID_PROBE_NONE;
bigbiff bigbiffe60683a2013-02-22 20:55:50 -0500173err:
bigbiff7b4c7a62015-01-01 19:44:14 -0500174 return -ENOMEM;
bigbiff bigbiffe60683a2013-02-22 20:55:50 -0500175}
176
177/*
178 * Mac disk always begin with "Driver Descriptor Record"
179 * (struct mac_driver_desc) and magic 0x4552.
180 */
181const struct blkid_idinfo mac_pt_idinfo =
182{
183 .name = "mac",
184 .probefunc = probe_mac_pt,
185 .magics =
186 {
187 /* big-endian magic string */
188 { .magic = "\x45\x52", .len = 2 },
189 { NULL }
190 }
191};
192