1 /* $NetBSD: disksubr.c,v 1.37 2024/01/07 07:58:35 isaki Exp $ */ 2 3 /* 4 * Copyright (c) 1982, 1986, 1988 Regents of the University of California. 5 * All rights reserved. 6 * 7 * Redistribution and use in source and binary forms, with or without 8 * modification, are permitted provided that the following conditions 9 * are met: 10 * 1. Redistributions of source code must retain the above copyright 11 * notice, this list of conditions and the following disclaimer. 12 * 2. Redistributions in binary form must reproduce the above copyright 13 * notice, this list of conditions and the following disclaimer in the 14 * documentation and/or other materials provided with the distribution. 15 * 3. Neither the name of the University nor the names of its contributors 16 * may be used to endorse or promote products derived from this software 17 * without specific prior written permission. 18 * 19 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 20 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 21 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 22 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 23 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 24 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 25 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 26 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 27 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 28 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 29 * SUCH DAMAGE. 30 * 31 * @(#)ufs_disksubr.c 7.16 (Berkeley) 5/4/91 32 */ 33 34 #include <sys/cdefs.h> 35 __KERNEL_RCSID(0, "$NetBSD: disksubr.c,v 1.37 2024/01/07 07:58:35 isaki Exp $"); 36 37 #include "opt_compat_netbsd.h" 38 39 #include <sys/param.h> 40 #include <sys/systm.h> 41 #include <sys/buf.h> 42 #include <sys/disklabel.h> 43 #include <sys/syslog.h> 44 #include <sys/disk.h> 45 46 /* get rid of DEV_BSIZE dependency */ 47 #define DEF_BSIZE DEV_BSIZE /* default sector size = 512 */ 48 49 static void parttbl_consistency_check(struct disklabel *, 50 struct dos_partition *); 51 52 53 /* 54 * Attempt to read a disk label from a device 55 * using the indicated strategy routine. 56 * The label must be partly set up before this: 57 * secpercyl, secsize and anything required for a block i/o read 58 * operation in the driver's strategy/start routines 59 * must be filled in before calling us. 60 * 61 * Returns null on success and an error string on failure. 62 */ 63 const char * 64 readdisklabel(dev_t dev, void (*strat)(struct buf *), 65 struct disklabel *lp, struct cpu_disklabel *osdep) 66 { 67 struct dos_partition *dp = 0; 68 struct dkbad *bdp = &osdep->bad; 69 struct buf *bp; 70 struct disklabel *dlp; 71 const char *msg = NULL; 72 int i, bsdlabelsz, humanlabelsz; 73 74 if (osdep) 75 dp = osdep->dosparts; 76 /* minimal requirements for archtypal disk label */ 77 if (lp->d_secsize == 0) 78 lp->d_secsize = DEF_BSIZE; 79 if (lp->d_secperunit == 0) 80 lp->d_secperunit = 0x1fffffff; 81 lp->d_npartitions = RAW_PART + 1; 82 for (i = 0; i < RAW_PART; i++) { 83 lp->d_partitions[i].p_size = 0; 84 lp->d_partitions[i].p_offset = 0; 85 } 86 if (lp->d_partitions[RAW_PART].p_size == 0) 87 lp->d_partitions[RAW_PART].p_size = lp->d_secperunit; 88 lp->d_partitions[RAW_PART].p_offset = 0; 89 90 lp->d_partitions[0].p_size = lp->d_partitions[RAW_PART].p_size; 91 92 /* get a buffer and initialize it */ 93 bsdlabelsz = 94 howmany(LABELOFFSET + sizeof(struct disklabel), lp->d_secsize) 95 * lp->d_secsize; 96 humanlabelsz = 97 howmany(sizeof(struct cpu_disklabel), lp->d_secsize) 98 * lp->d_secsize; 99 bp = geteblk(MAX(bsdlabelsz, humanlabelsz)); 100 bp->b_dev = dev; 101 102 /* read BSD disklabel first */ 103 bp->b_blkno = LABELSECTOR; 104 bp->b_cylinder = LABELSECTOR/lp->d_secpercyl; 105 bp->b_bcount = bsdlabelsz; /* to support < 512B/sector disks */ 106 bp->b_flags |= B_READ; 107 (*strat)(bp); 108 109 /* if successful, locate disk label within block and validate */ 110 if (biowait(bp)) { 111 msg = "disk label I/O error"; 112 goto dodospart; 113 } 114 for (dlp = (struct disklabel *)bp->b_data; 115 dlp <= (struct disklabel *) 116 ((char *)bp->b_data + bsdlabelsz - sizeof(*dlp)); 117 dlp = (struct disklabel *)((uint8_t *)dlp + sizeof(long))) { 118 if (dlp->d_magic != DISKMAGIC || dlp->d_magic2 != DISKMAGIC) { 119 if (msg == NULL) 120 msg = "no disk label"; 121 } else if (dlp->d_npartitions > MAXPARTITIONS || 122 dkcksum(dlp) != 0) 123 msg = "disk label corrupted"; 124 else { 125 *lp = *dlp; 126 msg = NULL; 127 break; 128 } 129 } 130 131 dodospart: 132 /* next do the Human68k-style partition table */ 133 /* Human68k does not support > 2048B/sector devices (?) */ 134 if (lp->d_secsize >= 2048) { 135 if (msg) 136 goto done; 137 goto dobadsect; 138 } 139 bp->b_blkno = DOSPARTOFF * DEF_BSIZE / lp->d_secsize; 140 /* DOSPARTOFF in DEV_BSIZE unit */ 141 bp->b_cylinder = DOSBBSECTOR / lp->d_secpercyl; 142 bp->b_bcount = humanlabelsz; /* to support < 512B/sector disks */ 143 bp->b_oflags &= ~(BO_DONE); 144 (*strat)(bp); 145 146 /* if successful, wander through Human68k partition table */ 147 if (biowait(bp)) 148 goto done; 149 if (strncmp(bp->b_data, "X68K", 4) != 0) { 150 /* Human68k-style partition table does not exist */ 151 if (msg) 152 goto done; 153 goto dobadsect; 154 } 155 156 /* XXX how do we check veracity/bounds of this? */ 157 if (dp) 158 memcpy(dp, (char *)bp->b_data + sizeof(*dp) /*DOSPARTOFF*/, 159 NDOSPART * sizeof(*dp)); 160 else 161 dp = (void *)((char *)bp->b_data + sizeof(*dp) /*DOSPARTOFF*/); 162 163 /* if BSD disklabel does not exist, fall back to Human68k partition */ 164 if (msg != NULL) { 165 msg = NULL; 166 lp->d_bbsize = 8192; 167 lp->d_sbsize = 2048; 168 for (i = 0; i < NDOSPART; i++, dp++) 169 /* is this ours? */ 170 if (dp->dp_size) { 171 u_char fstype; 172 int part = i + (i < RAW_PART ? 0 : 1); 173 int start = dp->dp_start * 2; 174 int size = dp->dp_size * 2; 175 176 /* update disklabel with details */ 177 lp->d_partitions[part].p_size = size; 178 lp->d_partitions[part].p_offset = start; 179 /* get partition type */ 180 #ifndef COMPAT_10 181 if (dp->dp_flag == 1) 182 fstype = FS_UNUSED; 183 else 184 #endif 185 if (!memcmp(dp->dp_typname, "Human68k", 8)) 186 fstype = FS_MSDOS; 187 else if (!memcmp(dp->dp_typname, 188 "BSD ffs ", 8)) 189 fstype = FS_BSDFFS; 190 else if (!memcmp(dp->dp_typname, 191 "BSD lfs ", 8)) 192 fstype = FS_BSDLFS; 193 else if (!memcmp(dp->dp_typname, 194 "BSD swap", 8)) 195 fstype = FS_SWAP; 196 #ifndef COMPAT_14 197 else if (part == 1) 198 fstype = FS_SWAP; 199 #endif 200 else 201 fstype = FS_BSDFFS; /* XXX */ 202 lp->d_partitions[part].p_fstype = fstype; /* XXX */ 203 if (lp->d_npartitions <= part) 204 lp->d_npartitions = part + 1; 205 } 206 } else { 207 parttbl_consistency_check(lp, dp); 208 } 209 210 dobadsect: 211 /* obtain bad sector table if requested and present */ 212 if (bdp && (lp->d_flags & D_BADSECT)) { 213 struct dkbad *db; 214 215 i = 0; 216 do { 217 /* read a bad sector table */ 218 bp->b_oflags &= ~(BO_DONE); 219 bp->b_flags |= B_READ; 220 bp->b_blkno = lp->d_secperunit - lp->d_nsectors + i; 221 if (lp->d_secsize > DEF_BSIZE) 222 bp->b_blkno *= lp->d_secsize / DEF_BSIZE; 223 else 224 bp->b_blkno /= DEF_BSIZE / lp->d_secsize; 225 bp->b_bcount = lp->d_secsize; 226 bp->b_cylinder = lp->d_ncylinders - 1; 227 (*strat)(bp); 228 229 /* if successful, validate, otherwise try another */ 230 if (biowait(bp)) { 231 msg = "bad sector table I/O error"; 232 } else { 233 db = (struct dkbad *)(bp->b_data); 234 #define DKBAD_MAGIC 0x4321 235 if (db->bt_mbz == 0 236 && db->bt_flag == DKBAD_MAGIC) { 237 msg = NULL; 238 *bdp = *db; 239 break; 240 } else 241 msg = "bad sector table corrupted"; 242 } 243 } while (bp->b_error != 0 && (i += 2) < 10 && 244 i < lp->d_nsectors); 245 } 246 247 done: 248 brelse(bp, 0); 249 return (msg); 250 } 251 252 /* 253 * Write disk label back to device after modification. 254 */ 255 int 256 writedisklabel(dev_t dev, void (*strat)(struct buf *), 257 struct disklabel *lp, struct cpu_disklabel *osdep) 258 { 259 struct dos_partition *dp = 0; 260 struct buf *bp; 261 struct disklabel *dlp; 262 int error, bsdlabelsz, humanlabelsz, i; 263 const char *np; 264 265 if (osdep) 266 dp = osdep->dosparts; 267 /* sanity clause */ 268 if (lp->d_secpercyl == 0 || lp->d_secsize == 0 269 /*|| (lp->d_secsize % DEF_BSIZE) != 0*/) 270 return(EINVAL); 271 if (dp) 272 parttbl_consistency_check(lp, dp); 273 274 /* get a buffer and initialize it */ 275 bsdlabelsz = 276 howmany(LABELOFFSET + sizeof(struct disklabel), lp->d_secsize) 277 * lp->d_secsize; 278 humanlabelsz = 279 howmany(sizeof(struct cpu_disklabel), lp->d_secsize) 280 * lp->d_secsize; 281 bp = geteblk(MAX(bsdlabelsz, humanlabelsz)); 282 bp->b_dev = dev; 283 284 /* attempt to write BSD disklabel first */ 285 bp->b_blkno = LABELSECTOR; 286 bp->b_cylinder = LABELSECTOR / lp->d_secpercyl; 287 bp->b_bcount = bsdlabelsz; /* to support < 512B/sector disks */ 288 bp->b_flags |= B_READ; 289 (*strat)(bp); 290 291 /* if successful, locate disk label within block and validate */ 292 if (biowait(bp)) 293 goto dodospart; 294 error = ESRCH; 295 for (dlp = (struct disklabel *)bp->b_data; 296 dlp <= (struct disklabel *) 297 ((char *)bp->b_data + bsdlabelsz - sizeof(*dlp)); 298 dlp = (struct disklabel *)((char *)dlp + sizeof(long))) { 299 if (dlp->d_magic == DISKMAGIC && dlp->d_magic2 == DISKMAGIC && 300 dkcksum(dlp) == 0) { 301 *dlp = *lp; 302 bp->b_oflags &= ~(BO_DONE); 303 bp->b_flags &= ~(B_READ); 304 bp->b_flags |= B_WRITE; 305 (*strat)(bp); 306 error = biowait(bp); 307 break; 308 } 309 } 310 311 /* do dos partitions in the process of getting disklabel? */ 312 if (error) { 313 dodospart: 314 if (lp->d_secsize >= 2048) { 315 error = ESRCH; 316 goto done; 317 } 318 #if 0 /* there is no mark on floppies */ 319 /* read the x68k disk magic */ 320 bp->b_blkno = DOSBBSECTOR; 321 bp->b_bcount = lp->d_secsize; 322 bp->b_oflags &= ~(BO_DONE); 323 bp->b_flags &= ~(B_WRITE); 324 bp->b_flags |= B_READ; 325 bp->b_cylinder = DOSBBSECTOR / lp->d_secpercyl; 326 (*strat)(bp); 327 if ((error = biowait(bp)) || memcmp(bp->b_data, "X68SCSI1", 8)) 328 printf("warning: disk not marked for x68k"); 329 #endif 330 331 /* read the partition table */ 332 bp->b_blkno = DOSPARTOFF; 333 bp->b_bcount = humanlabelsz; 334 bp->b_oflags &= ~(BO_DONE); 335 bp->b_flags &= ~(B_WRITE); 336 bp->b_flags |= B_READ; 337 bp->b_cylinder = DOSPARTOFF / lp->d_secpercyl; 338 (*strat)(bp); 339 340 if ((error = biowait(bp)) == 0) { 341 /* XXX how do we check veracity/bounds of this? */ 342 dp = (struct dos_partition *)bp->b_data + 1; 343 for (i = 0; i < NDOSPART; i++, dp++) { 344 int part = i + (i < RAW_PART ? 0 : 1); 345 int start, size; 346 347 start = lp->d_partitions[part].p_offset >> 1; 348 size = lp->d_partitions[part].p_size >> 1; 349 350 switch (lp->d_partitions[part].p_fstype) { 351 case FS_MSDOS: 352 np = "Human68k"; 353 dp->dp_flag = 0; /* autoboot */ 354 break; 355 356 case FS_SWAP: 357 np = "BSD swap"; 358 dp->dp_flag = 2; /* in use */ 359 break; 360 361 case FS_BSDFFS: 362 np = "BSD ffs "; 363 if (part == 0) 364 dp->dp_flag = 0; /* autoboot */ 365 else 366 dp->dp_flag = 2; /* in use */ 367 break; 368 369 case FS_BSDLFS: 370 np = "BSD lfs "; 371 if (part == 0) 372 dp->dp_flag = 0; /* autoboot */ 373 else 374 dp->dp_flag = 2; /* in use */ 375 break; 376 377 case FS_UNUSED: 378 np = "\0\0\0\0\0\0\0\0"; 379 start = size = 0; 380 if (part < lp->d_npartitions) { 381 dp->dp_flag = 1; 382 } else { 383 dp->dp_flag = 0; 384 } 385 break; 386 387 default: 388 /* XXX OS-9, MINIX etc. */ 389 continue; 390 } 391 memcpy(dp->dp_typname, np, 8); 392 dp->dp_start = start; 393 dp->dp_size = size; 394 } 395 bp->b_oflags &= ~(BO_DONE); 396 bp->b_flags &= ~(B_READ); 397 bp->b_flags |= B_WRITE; 398 (*strat)(bp); 399 error = biowait(bp); 400 } 401 } 402 403 #ifdef maybe 404 /* disklabel in appropriate location? */ 405 if (lp->d_partitions[0].p_offset != 0 406 && lp->d_partitions[0].p_offset != dospartoff) { 407 error = EXDEV; 408 goto done; 409 } 410 #endif 411 412 done: 413 brelse(bp, 0); 414 return (error); 415 } 416 417 static void 418 parttbl_consistency_check(struct disklabel *lp, struct dos_partition *dp) 419 { 420 int i, j; 421 int f = (lp->d_secsize >= 1024) ? lp->d_secsize/1024 : 1; 422 int g = (lp->d_secsize >= 1024) ? 1 : 1024/lp->d_secsize; 423 424 /* 1. overlapping check on partition table */ 425 for (i = 0; i < NDOSPART; i++) { 426 if (dp[i].dp_size == 0) 427 continue; 428 for (j = i+1; j < NDOSPART; j++) { 429 if (dp[j].dp_size == 0) 430 continue; 431 if (((dp[i].dp_start <= dp[j].dp_start) && 432 (dp[i].dp_start + dp[i].dp_size > dp[j].dp_start))|| 433 ((dp[j].dp_start <= dp[i].dp_start) && 434 (dp[j].dp_start + dp[j].dp_size > dp[i].dp_start))) { 435 printf("warning: Human68k partition %d and %d" 436 " are overlapping\n", i+1, j+1); 437 return; 438 } 439 } 440 } 441 442 /* 2. scan disklabel partitions */ 443 #define bp lp->d_partitions 444 for (i = 0; i < lp->d_npartitions; i++) { 445 int c = 0; 446 447 if (lp->d_partitions[i].p_fstype == FS_UNUSED || 448 lp->d_partitions[i].p_size == 0) 449 continue; 450 for (j = 0; j < NDOSPART; j++) { 451 if (dp[j].dp_size == 0) 452 continue; 453 if ((bp[i].p_offset * f < (dp[j].dp_start + dp[j].dp_size) * g) && 454 ((bp[i].p_offset + bp[i].p_size) * f >= (dp[j].dp_start + dp[j].dp_size) * g)) 455 c++; 456 if ((bp[i].p_offset * f > dp[j].dp_start * g) && 457 ((bp[i].p_offset + bp[i].p_size) * f < (dp[j].dp_start + dp[j].dp_size) * g)) 458 c++; 459 if ((bp[i].p_offset * f >= dp[j].dp_start * g) && 460 ((bp[i].p_offset + bp[i].p_size) * f < dp[j].dp_start * g)) 461 c++; 462 } 463 if (c > 1) 464 printf ("warning: partition %c spans for 2 or more" 465 " partitions in Human68k partition table.\n", 466 i+'a'); 467 } 468 #undef bp 469 470 /* more checks? */ 471 } 472