bigbiff bigbiff | e60683a | 2013-02-22 20:55:50 -0500 | [diff] [blame] | 1 | /* |
| 2 | * Solaris x86 partition 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 | #include <stdio.h> |
| 10 | #include <string.h> |
| 11 | #include <stdlib.h> |
| 12 | #include <stdint.h> |
| 13 | |
| 14 | #include "partitions.h" |
| 15 | |
| 16 | /* |
| 17 | * Solaris-x86 is always within primary dos partition (nested PT table). The |
| 18 | * solaris-x86 vtoc allows to split the entire partition to "slices". The |
| 19 | * offset (start) of the slice is always relatively to the primary dos |
| 20 | * partition. |
| 21 | * |
| 22 | * Note that Solaris-SPARC uses entire disk with a different partitionning |
| 23 | * scheme. |
| 24 | */ |
| 25 | |
| 26 | /* some other implementation than Linux kernel assume 8 partitions only */ |
| 27 | #define SOLARIS_MAXPARTITIONS 16 |
| 28 | |
| 29 | /* disklabel (vtoc) location */ |
| 30 | #define SOLARIS_SECTOR 1 /* in 512-sectors */ |
| 31 | #define SOLARIS_OFFSET (SOLARIS_SECTOR << 9) /* in bytes */ |
| 32 | #define SOLARIS_MAGICOFFSET (SOLARIS_OFFSET + 12) /* v_sanity offset in bytes */ |
| 33 | |
| 34 | /* slice tags */ |
| 35 | #define SOLARIS_TAG_WHOLEDISK 5 |
| 36 | |
| 37 | struct solaris_slice { |
| 38 | uint16_t s_tag; /* ID tag of partition */ |
| 39 | uint16_t s_flag; /* permission flags */ |
| 40 | uint32_t s_start; /* start sector no of partition */ |
| 41 | uint32_t s_size; /* # of blocks in partition */ |
| 42 | } __attribute__((packed)); |
| 43 | |
| 44 | struct solaris_vtoc { |
| 45 | unsigned int v_bootinfo[3]; /* info needed by mboot (unsupported) */ |
| 46 | |
| 47 | uint32_t v_sanity; /* to verify vtoc sanity */ |
| 48 | uint32_t v_version; /* layout version */ |
| 49 | char v_volume[8]; /* volume name */ |
| 50 | uint16_t v_sectorsz; /* sector size in bytes */ |
| 51 | uint16_t v_nparts; /* number of partitions */ |
| 52 | unsigned int v_reserved[10]; /* free space */ |
| 53 | |
| 54 | struct solaris_slice v_slice[SOLARIS_MAXPARTITIONS]; /* slices */ |
| 55 | |
| 56 | unsigned int timestamp[SOLARIS_MAXPARTITIONS]; /* timestamp (unsupported) */ |
| 57 | char v_asciilabel[128]; /* for compatibility */ |
| 58 | } __attribute__((packed)); |
| 59 | |
| 60 | static int probe_solaris_pt(blkid_probe pr, |
| 61 | const struct blkid_idmag *mag __attribute__((__unused__))) |
| 62 | { |
| 63 | struct solaris_vtoc *l; /* disk label */ |
| 64 | struct solaris_slice *p; /* partitsion */ |
| 65 | blkid_parttable tab = NULL; |
| 66 | blkid_partition parent; |
| 67 | blkid_partlist ls; |
| 68 | int i; |
| 69 | uint16_t nparts; |
| 70 | |
| 71 | l = (struct solaris_vtoc *) blkid_probe_get_sector(pr, SOLARIS_SECTOR); |
| 72 | if (!l) |
| 73 | goto nothing; |
| 74 | |
| 75 | if (le32_to_cpu(l->v_version) != 1) { |
| 76 | DBG(DEBUG_LOWPROBE, printf( |
| 77 | "WARNING: unsupported solaris x86 version %d, ignore\n", |
| 78 | le32_to_cpu(l->v_version))); |
| 79 | goto nothing; |
| 80 | } |
| 81 | |
| 82 | if (blkid_partitions_need_typeonly(pr)) |
| 83 | /* caller does not ask for details about partitions */ |
| 84 | return 0; |
| 85 | |
| 86 | ls = blkid_probe_get_partlist(pr); |
| 87 | if (!ls) |
| 88 | goto err; |
| 89 | |
| 90 | parent = blkid_partlist_get_parent(ls); |
| 91 | |
| 92 | tab = blkid_partlist_new_parttable(ls, "solaris", SOLARIS_OFFSET); |
| 93 | if (!tab) |
| 94 | goto err; |
| 95 | |
| 96 | nparts = le16_to_cpu(l->v_nparts); |
| 97 | if (nparts > SOLARIS_MAXPARTITIONS) |
| 98 | nparts = SOLARIS_MAXPARTITIONS; |
| 99 | |
| 100 | for (i = 1, p = &l->v_slice[0]; i < nparts; i++, p++) { |
| 101 | |
| 102 | uint32_t start = le32_to_cpu(p->s_start); |
| 103 | uint32_t size = le32_to_cpu(p->s_size); |
| 104 | blkid_partition par; |
| 105 | |
| 106 | if (size == 0 || le16_to_cpu(p->s_tag) == SOLARIS_TAG_WHOLEDISK) |
| 107 | continue; |
| 108 | |
| 109 | if (parent) |
| 110 | /* Solaris slices are relative to the parent (primary |
| 111 | * DOS partition) */ |
| 112 | start += blkid_partition_get_start(parent); |
| 113 | |
| 114 | if (parent && !blkid_is_nested_dimension(parent, start, size)) { |
| 115 | DBG(DEBUG_LOWPROBE, printf( |
| 116 | "WARNING: solaris partition (%d) overflow " |
| 117 | "detected, ignore\n", i)); |
| 118 | continue; |
| 119 | } |
| 120 | |
| 121 | par = blkid_partlist_add_partition(ls, tab, start, size); |
| 122 | if (!par) |
| 123 | goto err; |
| 124 | |
| 125 | blkid_partition_set_type(par, le16_to_cpu(p->s_tag)); |
| 126 | blkid_partition_set_flags(par, le16_to_cpu(p->s_flag)); |
| 127 | } |
| 128 | |
| 129 | return 0; |
| 130 | |
| 131 | nothing: |
| 132 | return 1; |
| 133 | err: |
| 134 | return -1; |
| 135 | } |
| 136 | |
| 137 | const struct blkid_idinfo solaris_x86_pt_idinfo = |
| 138 | { |
| 139 | .name = "solaris", |
| 140 | .probefunc = probe_solaris_pt, |
| 141 | .magics = |
| 142 | { |
| 143 | { |
| 144 | .magic = "\xEE\xDE\x0D\x60", /* little-endian magic string */ |
| 145 | .len = 4, /* v_sanity size in bytes */ |
| 146 | .sboff = SOLARIS_MAGICOFFSET /* offset of v_sanity */ |
| 147 | }, |
| 148 | { NULL } |
| 149 | } |
| 150 | }; |
| 151 | |