1 /* 2 * CDDL HEADER START 3 * 4 * The contents of this file are subject to the terms of the 5 * Common Development and Distribution License, Version 1.0 only 6 * (the "License"). You may not use this file except in compliance 7 * with the License. 8 * 9 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 10 * or http://www.opensolaris.org/os/licensing. 11 * See the License for the specific language governing permissions 12 * and limitations under the License. 13 * 14 * When distributing Covered Code, include this CDDL HEADER in each 15 * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 16 * If applicable, add the following below this CDDL HEADER, with the 17 * fields enclosed by brackets "[]" replaced with your own identifying 18 * information: Portions Copyright [yyyy] [name of copyright owner] 19 * 20 * CDDL HEADER END 21 */ 22 /* 23 * Copyright 2005 Sun Microsystems, Inc. All rights reserved. 24 * Use is subject to license terms. 25 */ 26 27 #pragma ident "%Z%%M% %I% %E% SMI" 28 29 #include <sys/param.h> 30 #include <sys/systm.h> 31 #include <sys/kmem.h> 32 #include <sys/user.h> 33 #include <sys/proc.h> 34 #include <sys/cred.h> 35 #include <sys/disp.h> 36 #include <sys/buf.h> 37 #include <sys/vfs.h> 38 #include <sys/vnode.h> 39 #include <sys/fdio.h> 40 #include <sys/file.h> 41 #include <sys/uio.h> 42 #include <sys/conf.h> 43 #undef NFSCLIENT 44 #include <sys/statvfs.h> 45 #include <sys/mount.h> 46 #include <sys/pathname.h> 47 #include <sys/cmn_err.h> 48 #include <sys/debug.h> 49 #include <sys/sysmacros.h> 50 #include <sys/conf.h> 51 #include <sys/mkdev.h> 52 #include <sys/swap.h> 53 #include <sys/sunddi.h> 54 #include <sys/sunldi.h> 55 #include <sys/dktp/fdisk.h> 56 #include <sys/fs/pc_label.h> 57 #include <sys/fs/pc_fs.h> 58 #include <sys/fs/pc_dir.h> 59 #include <sys/fs/pc_node.h> 60 #include <fs/fs_subr.h> 61 #include <sys/modctl.h> 62 #include <sys/vol.h> 63 #include <sys/dkio.h> 64 #include <sys/open.h> 65 #include <sys/mntent.h> 66 #include <sys/policy.h> 67 68 /* 69 * The majority of PC media use a 512 sector size, but 70 * occasionally you will run across a 1k sector size. 71 * For media with a 1k sector size, fd_strategy() requires 72 * the I/O size to be a 1k multiple; so when the sector size 73 * is not yet known, always read 1k. 74 */ 75 #define PC_SAFESECSIZE (PC_SECSIZE * 2) 76 77 static int pcfs_pseudo_floppy(dev_t); 78 79 static int pcfsinit(int, char *); 80 static int pcfs_mount(struct vfs *, struct vnode *, struct mounta *, 81 struct cred *); 82 static int pcfs_unmount(struct vfs *, int, struct cred *); 83 static int pcfs_root(struct vfs *, struct vnode **); 84 static int pcfs_statvfs(struct vfs *, struct statvfs64 *); 85 static int pc_syncfsnodes(struct pcfs *); 86 static int pcfs_sync(struct vfs *, short, struct cred *); 87 static int pcfs_vget(struct vfs *vfsp, struct vnode **vpp, struct fid *fidp); 88 89 static int pc_getfattype(struct vnode *, int, daddr_t *, int *); 90 static int pc_readfat(struct pcfs *fsp, uchar_t *fatp, daddr_t start, 91 size_t fatsize); 92 static int pc_writefat(struct pcfs *fsp, daddr_t start); 93 94 /* 95 * pcfs mount options table 96 */ 97 98 static char *nohidden_cancel[] = {MNTOPT_PCFS_HIDDEN, NULL}; 99 static char *hidden_cancel[] = {MNTOPT_PCFS_NOHIDDEN, NULL}; 100 static char *nofoldcase_cancel[] = {MNTOPT_PCFS_FOLDCASE, NULL}; 101 static char *foldcase_cancel[] = {MNTOPT_PCFS_NOFOLDCASE, NULL}; 102 103 static mntopt_t mntopts[] = { 104 /* 105 * option name cancel option default arg flags 106 * opt data 107 */ 108 { MNTOPT_PCFS_NOHIDDEN, nohidden_cancel, NULL, MO_DEFAULT, 109 NULL }, 110 { MNTOPT_PCFS_HIDDEN, hidden_cancel, NULL, 0, 111 NULL }, 112 { MNTOPT_PCFS_NOFOLDCASE, nofoldcase_cancel, NULL, MO_DEFAULT, 113 NULL }, 114 { MNTOPT_PCFS_FOLDCASE, foldcase_cancel, NULL, 0, 115 NULL } 116 }; 117 118 static mntopts_t pcfs_mntopts = { 119 sizeof (mntopts) / sizeof (mntopt_t), 120 mntopts 121 }; 122 123 int pcfsdebuglevel = 0; 124 125 /* 126 * pcfslock: protects the list of mounted pc filesystems "pc_mounttab. 127 * pcfs_lock: (inside per filesystem structure "pcfs") 128 * per filesystem lock. Most of the vfsops and vnodeops are 129 * protected by this lock. 130 * pcnodes_lock: protects the pcnode hash table "pcdhead", "pcfhead". 131 * 132 * Lock hierarchy: pcfslock > pcfs_lock > pcnodes_lock 133 */ 134 kmutex_t pcfslock; 135 krwlock_t pcnodes_lock; /* protect the pcnode hash table "pcdhead", "pcfhead" */ 136 137 static int pcfstype; 138 139 static vfsdef_t vfw = { 140 VFSDEF_VERSION, 141 "pcfs", 142 pcfsinit, 143 VSW_HASPROTO|VSW_CANREMOUNT, 144 &pcfs_mntopts 145 }; 146 147 extern struct mod_ops mod_fsops; 148 149 static struct modlfs modlfs = { 150 &mod_fsops, 151 "PC filesystem v%I%", 152 &vfw 153 }; 154 155 static struct modlinkage modlinkage = { 156 MODREV_1, 157 &modlfs, 158 NULL 159 }; 160 161 int 162 _init(void) 163 { 164 int error; 165 166 #if !defined(lint) 167 /* make sure the on-disk structures are sane */ 168 ASSERT(sizeof (struct pcdir) == 32); 169 ASSERT(sizeof (struct pcdir_lfn) == 32); 170 #endif 171 mutex_init(&pcfslock, NULL, MUTEX_DEFAULT, NULL); 172 rw_init(&pcnodes_lock, NULL, RW_DEFAULT, NULL); 173 error = mod_install(&modlinkage); 174 if (error) { 175 mutex_destroy(&pcfslock); 176 rw_destroy(&pcnodes_lock); 177 } 178 return (error); 179 } 180 181 int 182 _fini(void) 183 { 184 int error; 185 186 error = mod_remove(&modlinkage); 187 if (error) 188 return (error); 189 mutex_destroy(&pcfslock); 190 rw_destroy(&pcnodes_lock); 191 /* 192 * Tear down the operations vectors 193 */ 194 (void) vfs_freevfsops_by_type(pcfstype); 195 vn_freevnodeops(pcfs_fvnodeops); 196 vn_freevnodeops(pcfs_dvnodeops); 197 return (0); 198 } 199 200 int 201 _info(struct modinfo *modinfop) 202 { 203 return (mod_info(&modlinkage, modinfop)); 204 } 205 206 /* ARGSUSED1 */ 207 static int 208 pcfsinit(int fstype, char *name) 209 { 210 static const fs_operation_def_t pcfs_vfsops_template[] = { 211 VFSNAME_MOUNT, pcfs_mount, 212 VFSNAME_UNMOUNT, pcfs_unmount, 213 VFSNAME_ROOT, pcfs_root, 214 VFSNAME_STATVFS, pcfs_statvfs, 215 VFSNAME_SYNC, (fs_generic_func_p) pcfs_sync, 216 VFSNAME_VGET, pcfs_vget, 217 NULL, NULL 218 }; 219 int error; 220 221 error = vfs_setfsops(fstype, pcfs_vfsops_template, NULL); 222 if (error != 0) { 223 cmn_err(CE_WARN, "pcfsinit: bad vfs ops template"); 224 return (error); 225 } 226 227 error = vn_make_ops("pcfs", pcfs_fvnodeops_template, &pcfs_fvnodeops); 228 if (error != 0) { 229 (void) vfs_freevfsops_by_type(fstype); 230 cmn_err(CE_WARN, "pcfsinit: bad file vnode ops template"); 231 return (error); 232 } 233 234 error = vn_make_ops("pcfsd", pcfs_dvnodeops_template, &pcfs_dvnodeops); 235 if (error != 0) { 236 (void) vfs_freevfsops_by_type(fstype); 237 vn_freevnodeops(pcfs_fvnodeops); 238 cmn_err(CE_WARN, "pcfsinit: bad dir vnode ops template"); 239 return (error); 240 } 241 242 pcfstype = fstype; 243 (void) pc_init(); 244 return (0); 245 } 246 247 static struct pcfs *pc_mounttab = NULL; 248 249 extern struct pcfs_args pc_tz; 250 251 /* 252 * Define some special logical drives we use internal to this file. 253 */ 254 #define BOOT_PARTITION_DRIVE 99 255 #define PRIMARY_DOS_DRIVE 1 256 257 /* 258 * pc_mount system call 259 */ 260 static int 261 pcfs_mount( 262 struct vfs *vfsp, 263 struct vnode *mvp, 264 struct mounta *uap, 265 struct cred *cr) 266 { 267 struct pcfs *fsp; 268 struct vnode *bvp; 269 struct vnode *devvp; 270 struct pathname special; 271 daddr_t dosstart; 272 dev_t pseudodev; 273 dev_t xdev; 274 char *spnp; 275 char *data = uap->dataptr; 276 int datalen = uap->datalen; 277 int dos_ldrive = 0; 278 int error; 279 int fattype; 280 int spnlen; 281 int wantbootpart = 0; 282 struct vioc_info info; 283 int rval; /* set but not used */ 284 minor_t minor; 285 int oflag, aflag; 286 287 if ((error = secpolicy_fs_mount(cr, mvp, vfsp)) != 0) 288 return (error); 289 290 PC_DPRINTF0(4, "pcfs_mount\n"); 291 if (mvp->v_type != VDIR) { 292 return (ENOTDIR); 293 } 294 mutex_enter(&mvp->v_lock); 295 if ((uap->flags & MS_REMOUNT) == 0 && 296 (uap->flags & MS_OVERLAY) == 0 && 297 (mvp->v_count != 1 || (mvp->v_flag & VROOT))) { 298 mutex_exit(&mvp->v_lock); 299 return (EBUSY); 300 } 301 mutex_exit(&mvp->v_lock); 302 303 /* 304 * The caller is responsible for making sure to always 305 * pass in sizeof(struct pcfs_args) (or the old one). 306 * Doing this is the only way to know an EINVAL return 307 * from mount(2) is due to the "not a DOS filesystem" 308 * EINVAL that pc_verify/pc_getfattype could return. 309 */ 310 if ((datalen != sizeof (struct pcfs_args)) && 311 (datalen != sizeof (struct old_pcfs_args))) { 312 return (EINVAL); 313 } else { 314 struct pcfs_args tmp_tz; 315 int hidden = 0; 316 int foldcase = 0; 317 318 tmp_tz.flags = 0; 319 if (copyin(data, &tmp_tz, datalen)) { 320 return (EFAULT); 321 } 322 if (datalen == sizeof (struct pcfs_args)) { 323 hidden = tmp_tz.flags & PCFS_MNT_HIDDEN; 324 foldcase = tmp_tz.flags & PCFS_MNT_FOLDCASE; 325 } 326 327 if (hidden) 328 vfs_setmntopt(vfsp, MNTOPT_PCFS_HIDDEN, NULL, 0); 329 if (foldcase) 330 vfs_setmntopt(vfsp, MNTOPT_PCFS_FOLDCASE, NULL, 0); 331 /* 332 * more than one pc filesystem can be mounted on x86 333 * so the pc_tz structure is now a critical region 334 */ 335 mutex_enter(&pcfslock); 336 if (pc_mounttab == NULL) 337 bcopy(&tmp_tz, &pc_tz, sizeof (struct pcfs_args)); 338 mutex_exit(&pcfslock); 339 } 340 /* 341 * Resolve path name of special file being mounted. 342 */ 343 if (error = pn_get(uap->spec, UIO_USERSPACE, &special)) { 344 return (error); 345 } 346 if (error = 347 lookupname(special.pn_path, UIO_SYSSPACE, FOLLOW, NULLVPP, &bvp)) { 348 /* 349 * look for suffix to special 350 * which indicates a request to mount the solaris boot 351 * partition, or a DOS logical drive on the hard disk 352 */ 353 spnlen = special.pn_pathlen; 354 355 if (spnlen > 5) { 356 spnp = special.pn_path + spnlen - 5; 357 if (*spnp++ == ':' && *spnp++ == 'b' && 358 *spnp++ == 'o' && *spnp++ == 'o' && 359 *spnp++ == 't') { 360 /* 361 * Looks as if they want to mount 362 * the Solaris boot partition 363 */ 364 wantbootpart = 1; 365 dos_ldrive = BOOT_PARTITION_DRIVE; 366 spnp = special.pn_path + spnlen - 5; 367 *spnp = '\0'; 368 error = lookupname(special.pn_path, 369 UIO_SYSSPACE, FOLLOW, NULLVPP, &bvp); 370 } 371 } 372 373 if (!wantbootpart) { 374 spnp = special.pn_path + spnlen - 1; 375 if (spnlen > 2 && *spnp >= 'c' && *spnp <= 'z') { 376 spnlen--; 377 dos_ldrive = *spnp-- - 'c' + 1; 378 } else if (spnlen > 2 && *spnp >= '0' && *spnp <= '9') { 379 spnlen--; 380 dos_ldrive = *spnp-- - '0'; 381 if (spnlen > 2 && *spnp >= '0' && 382 *spnp <= '9') { 383 spnlen--; 384 dos_ldrive += 10 * (*spnp-- - '0'); 385 } 386 } 387 if (spnlen > 1 && dos_ldrive && dos_ldrive <= 24 && 388 *spnp == ':') { 389 /* 390 * remove suffix so that we have a real 391 * device name 392 */ 393 *spnp = '\0'; 394 error = lookupname(special.pn_path, 395 UIO_SYSSPACE, FOLLOW, NULLVPP, &bvp); 396 } 397 } 398 if (error) { 399 pn_free(&special); 400 return (error); 401 } 402 } 403 pn_free(&special); 404 if (bvp->v_type != VBLK) { 405 VN_RELE(bvp); 406 return (ENOTBLK); 407 } 408 xdev = bvp->v_rdev; 409 /* 410 * Verify caller's permission to open the device special file. 411 */ 412 if ((vfsp->vfs_flag & VFS_RDONLY) != 0 || 413 ((uap->flags & MS_RDONLY) != 0)) { 414 oflag = FREAD; 415 aflag = VREAD; 416 } else { 417 oflag = FREAD | FWRITE; 418 aflag = VREAD | VWRITE; 419 } 420 if ((error = VOP_ACCESS(bvp, aflag, 0, cr)) != 0 || 421 (error = secpolicy_spec_open(cr, bvp, oflag)) != 0) { 422 VN_RELE(bvp); 423 return (error); 424 } 425 426 VN_RELE(bvp); 427 if (getmajor(xdev) >= devcnt) { 428 return (ENXIO); 429 } 430 /* 431 * Ensure that this device (or logical drive) isn't already mounted, 432 * unless this is a REMOUNT request 433 */ 434 if (dos_ldrive) { 435 mutex_enter(&pcfslock); 436 for (fsp = pc_mounttab; fsp; fsp = fsp->pcfs_nxt) 437 if (fsp->pcfs_xdev == xdev && 438 fsp->pcfs_ldrv == dos_ldrive) { 439 mutex_exit(&pcfslock); 440 if (uap->flags & MS_REMOUNT) { 441 return (0); 442 } else { 443 return (EBUSY); 444 } 445 } 446 /* 447 * Assign a unique device number for the vfs 448 * The old way (getudev() + a constantly incrementing 449 * major number) was wrong because it changes vfs_dev 450 * across mounts and reboots, which breaks nfs file handles. 451 * UFS just uses the real dev_t. We can't do that because 452 * of the way pcfs opens fdisk partitons (the :c and :d 453 * partitions are on the same dev_t). Though that _might_ 454 * actually be ok, since the file handle contains an 455 * absolute block number, it's probably better to make them 456 * different. So I think we should retain the original 457 * dev_t, but come up with a different minor number based 458 * on the logical drive that will _always_ come up the same. 459 * For now, we steal the upper 6 bits. 460 */ 461 #ifdef notdef 462 /* what should we do here? */ 463 if (((getminor(xdev) >> 12) & 0x3F) != 0) 464 printf("whoops - upper bits used!\n"); 465 #endif 466 minor = ((dos_ldrive << 12) | getminor(xdev)) & MAXMIN32; 467 pseudodev = makedevice(getmajor(xdev), minor); 468 if (vfs_devmounting(pseudodev, vfsp)) { 469 mutex_exit(&pcfslock); 470 return (EBUSY); 471 } 472 if (vfs_devismounted(pseudodev)) { 473 mutex_exit(&pcfslock); 474 if (uap->flags & MS_REMOUNT) { 475 return (0); 476 } else { 477 return (EBUSY); 478 } 479 } 480 mutex_exit(&pcfslock); 481 } else { 482 if (vfs_devmounting(xdev, vfsp)) { 483 return (EBUSY); 484 } 485 if (vfs_devismounted(xdev)) 486 if (uap->flags & MS_REMOUNT) { 487 return (0); 488 } else { 489 return (EBUSY); 490 } 491 pseudodev = xdev; 492 } 493 494 if (uap->flags & MS_RDONLY) { 495 vfsp->vfs_flag |= VFS_RDONLY; 496 vfs_setmntopt(vfsp, MNTOPT_RO, NULL, 0); 497 } 498 499 /* 500 * Mount the filesystem 501 */ 502 devvp = makespecvp(xdev, VBLK); 503 if (IS_SWAPVP(devvp)) { 504 VN_RELE(devvp); 505 return (EBUSY); 506 } 507 508 /* 509 * special handling for PCMCIA memory card 510 * with pseudo floppies organization 511 */ 512 if (dos_ldrive == 0 && pcfs_pseudo_floppy(xdev)) { 513 dosstart = (daddr_t)0; 514 fattype = PCFS_PCMCIA_NO_CIS; 515 } else { 516 if (error = pc_getfattype(devvp, dos_ldrive, &dosstart, 517 &fattype)) { 518 VN_RELE(devvp); 519 return (error); 520 } 521 } 522 523 (void) VOP_PUTPAGE(devvp, (offset_t)0, (uint_t)0, B_INVAL, cr); 524 fsp = kmem_zalloc((uint_t)sizeof (struct pcfs), KM_SLEEP); 525 fsp->pcfs_vfs = vfsp; 526 fsp->pcfs_flags = fattype; 527 fsp->pcfs_devvp = devvp; 528 fsp->pcfs_xdev = xdev; 529 fsp->pcfs_ldrv = dos_ldrive; 530 fsp->pcfs_dosstart = dosstart; 531 mutex_init(&fsp->pcfs_lock, NULL, MUTEX_DEFAULT, NULL); 532 533 /* set the "nocheck" flag if volmgt is managing this volume */ 534 info.vii_pathlen = 0; 535 info.vii_devpath = 0; 536 error = cdev_ioctl(fsp->pcfs_xdev, VOLIOCINFO, (intptr_t)&info, 537 FKIOCTL|FREAD, kcred, &rval); 538 if (error == 0) { 539 fsp->pcfs_flags |= PCFS_NOCHK; 540 } 541 542 if (vfs_optionisset(vfsp, MNTOPT_PCFS_HIDDEN, NULL)) 543 fsp->pcfs_flags |= PCFS_HIDDEN; 544 if (vfs_optionisset(vfsp, MNTOPT_PCFS_FOLDCASE, NULL)) 545 fsp->pcfs_flags |= PCFS_FOLDCASE; 546 vfsp->vfs_dev = pseudodev; 547 vfsp->vfs_fstype = pcfstype; 548 vfs_make_fsid(&vfsp->vfs_fsid, pseudodev, pcfstype); 549 vfsp->vfs_data = (caddr_t)fsp; 550 vfsp->vfs_bcount = 0; 551 552 error = pc_verify(fsp); 553 if (error) { 554 VN_RELE(devvp); 555 mutex_destroy(&fsp->pcfs_lock); 556 kmem_free(fsp, (uint_t)sizeof (struct pcfs)); 557 return (error); 558 } 559 vfsp->vfs_bsize = fsp->pcfs_clsize; 560 561 mutex_enter(&pcfslock); 562 fsp->pcfs_nxt = pc_mounttab; 563 pc_mounttab = fsp; 564 mutex_exit(&pcfslock); 565 return (0); 566 } 567 568 /* 569 * vfs operations 570 */ 571 572 /* ARGSUSED */ 573 static int 574 pcfs_unmount( 575 struct vfs *vfsp, 576 int flag, 577 struct cred *cr) 578 { 579 struct pcfs *fsp, *fsp1; 580 581 if (secpolicy_fs_unmount(cr, vfsp) != 0) 582 return (EPERM); 583 584 /* 585 * forced unmount is not supported by this file system 586 * and thus, ENOTSUP, is being returned. 587 */ 588 if (flag & MS_FORCE) 589 return (ENOTSUP); 590 591 PC_DPRINTF0(4, "pcfs_unmount\n"); 592 fsp = VFSTOPCFS(vfsp); 593 /* 594 * We don't have to lock fsp because the VVFSLOCK in vfs layer will 595 * prevent lookuppn from crossing the mount point. 596 */ 597 if (fsp->pcfs_nrefs) { 598 return (EBUSY); 599 } 600 601 /* 602 * Allow an unmount (regardless of state) if the fs instance has 603 * been marked as beyond recovery. 604 */ 605 if (fsp->pcfs_flags & PCFS_IRRECOV) { 606 mutex_enter(&pcfslock); 607 rw_enter(&pcnodes_lock, RW_WRITER); 608 pc_diskchanged(fsp); 609 rw_exit(&pcnodes_lock); 610 mutex_exit(&pcfslock); 611 } 612 613 /* now there should be no pcp node on pcfhead or pcdhead. */ 614 615 mutex_enter(&pcfslock); 616 if (fsp == pc_mounttab) { 617 pc_mounttab = fsp->pcfs_nxt; 618 } else { 619 for (fsp1 = pc_mounttab; fsp1 != NULL; fsp1 = fsp1->pcfs_nxt) 620 if (fsp1->pcfs_nxt == fsp) 621 fsp1->pcfs_nxt = fsp->pcfs_nxt; 622 } 623 624 if (fsp->pcfs_fatp != (uchar_t *)0) { 625 pc_invalfat(fsp); 626 } 627 mutex_exit(&pcfslock); 628 629 VN_RELE(fsp->pcfs_devvp); 630 mutex_destroy(&fsp->pcfs_lock); 631 kmem_free(fsp, (uint_t)sizeof (struct pcfs)); 632 return (0); 633 } 634 635 /* 636 * find root of pcfs 637 */ 638 static int 639 pcfs_root( 640 struct vfs *vfsp, 641 struct vnode **vpp) 642 { 643 struct pcfs *fsp; 644 struct pcnode *pcp; 645 int error; 646 647 fsp = VFSTOPCFS(vfsp); 648 if (error = pc_lockfs(fsp, 0, 0)) 649 return (error); 650 pcp = pc_getnode(fsp, (daddr_t)0, 0, (struct pcdir *)0); 651 PC_DPRINTF2(9, "pcfs_root(0x%p) pcp= 0x%p\n", 652 (void *)vfsp, (void *)pcp); 653 pc_unlockfs(fsp); 654 *vpp = PCTOV(pcp); 655 pcp->pc_flags |= PC_EXTERNAL; 656 return (0); 657 } 658 659 /* 660 * Get file system statistics. 661 */ 662 static int 663 pcfs_statvfs( 664 struct vfs *vfsp, 665 struct statvfs64 *sp) 666 { 667 struct pcfs *fsp; 668 int error; 669 dev32_t d32; 670 671 fsp = VFSTOPCFS(vfsp); 672 error = pc_getfat(fsp); 673 if (error) 674 return (error); 675 bzero(sp, sizeof (*sp)); 676 sp->f_bsize = sp->f_frsize = fsp->pcfs_clsize; 677 sp->f_blocks = (fsblkcnt64_t)fsp->pcfs_ncluster; 678 sp->f_bavail = sp->f_bfree = (fsblkcnt64_t)pc_freeclusters(fsp); 679 sp->f_files = (fsfilcnt64_t)-1; 680 sp->f_ffree = (fsfilcnt64_t)-1; 681 sp->f_favail = (fsfilcnt64_t)-1; 682 #ifdef notdef 683 (void) cmpldev(&d32, fsp->pcfs_devvp->v_rdev); 684 #endif /* notdef */ 685 (void) cmpldev(&d32, vfsp->vfs_dev); 686 sp->f_fsid = d32; 687 (void) strcpy(sp->f_basetype, vfssw[vfsp->vfs_fstype].vsw_name); 688 sp->f_flag = vf_to_stf(vfsp->vfs_flag); 689 sp->f_namemax = PCFNAMESIZE; 690 return (0); 691 } 692 693 static int 694 pc_syncfsnodes(struct pcfs *fsp) 695 { 696 struct pchead *hp; 697 struct pcnode *pcp; 698 int error; 699 700 PC_DPRINTF0(7, "pcfs_syncfsnodes\n"); 701 if (error = pc_lockfs(fsp, 0, 0)) 702 return (error); 703 704 if (!(error = pc_syncfat(fsp))) { 705 hp = pcfhead; 706 while (hp < & pcfhead [ NPCHASH ]) { 707 rw_enter(&pcnodes_lock, RW_READER); 708 pcp = hp->pch_forw; 709 while (pcp != (struct pcnode *)hp) { 710 if (VFSTOPCFS(PCTOV(pcp) -> v_vfsp) == fsp) 711 if (error = pc_nodesync(pcp)) 712 break; 713 pcp = pcp -> pc_forw; 714 } 715 rw_exit(&pcnodes_lock); 716 if (error) 717 break; 718 hp++; 719 } 720 } 721 pc_unlockfs(fsp); 722 return (error); 723 } 724 725 /* 726 * Flush any pending I/O. 727 */ 728 /*ARGSUSED*/ 729 static int 730 pcfs_sync( 731 struct vfs *vfsp, 732 short flag, 733 struct cred *cr) 734 { 735 struct pcfs *fsp; 736 int error = 0; 737 738 /* this prevents the filesystem from being umounted. */ 739 mutex_enter(&pcfslock); 740 if (vfsp != NULL) { 741 fsp = VFSTOPCFS(vfsp); 742 if (!(fsp->pcfs_flags & PCFS_IRRECOV)) { 743 error = pc_syncfsnodes(fsp); 744 } else { 745 rw_enter(&pcnodes_lock, RW_WRITER); 746 pc_diskchanged(fsp); 747 rw_exit(&pcnodes_lock); 748 error = EIO; 749 } 750 } else { 751 fsp = pc_mounttab; 752 while (fsp != NULL) { 753 if (fsp->pcfs_flags & PCFS_IRRECOV) { 754 rw_enter(&pcnodes_lock, RW_WRITER); 755 pc_diskchanged(fsp); 756 rw_exit(&pcnodes_lock); 757 error = EIO; 758 break; 759 } 760 error = pc_syncfsnodes(fsp); 761 if (error) break; 762 fsp = fsp->pcfs_nxt; 763 } 764 } 765 mutex_exit(&pcfslock); 766 return (error); 767 } 768 769 int 770 pc_lockfs(struct pcfs *fsp, int diskchanged, int releasing) 771 { 772 if ((fsp->pcfs_flags & PCFS_IRRECOV) && !releasing) 773 return (EIO); 774 775 if ((fsp->pcfs_flags & PCFS_LOCKED) && (fsp->pcfs_owner == curthread)) { 776 fsp->pcfs_count++; 777 } else { 778 mutex_enter(&fsp->pcfs_lock); 779 if (fsp->pcfs_flags & PCFS_LOCKED) 780 panic("pc_lockfs"); 781 /* 782 * We check the IRRECOV bit again just in case somebody 783 * snuck past the initial check but then got held up before 784 * they could grab the lock. (And in the meantime someone 785 * had grabbed the lock and set the bit) 786 */ 787 if (!diskchanged && !(fsp->pcfs_flags & PCFS_IRRECOV)) { 788 int err; 789 if ((err = pc_getfat(fsp))) 790 return (err); 791 } 792 fsp->pcfs_flags |= PCFS_LOCKED; 793 fsp->pcfs_owner = curthread; 794 fsp->pcfs_count++; 795 } 796 return (0); 797 } 798 799 void 800 pc_unlockfs(struct pcfs *fsp) 801 { 802 803 if ((fsp->pcfs_flags & PCFS_LOCKED) == 0) 804 panic("pc_unlockfs"); 805 if (--fsp->pcfs_count < 0) 806 panic("pc_unlockfs: count"); 807 if (fsp->pcfs_count == 0) { 808 fsp->pcfs_flags &= ~PCFS_LOCKED; 809 fsp->pcfs_owner = 0; 810 mutex_exit(&fsp->pcfs_lock); 811 } 812 } 813 814 /* 815 * isDosDrive() 816 * Boolean function. Give it the systid field for an fdisk partition 817 * and it decides if that's a systid that describes a DOS drive. We 818 * use systid values defined in sys/dktp/fdisk.h. 819 */ 820 static int 821 isDosDrive(uchar_t checkMe) 822 { 823 return ((checkMe == DOSOS12) || (checkMe == DOSOS16) || 824 (checkMe == DOSHUGE) || (checkMe == FDISK_WINDOWS) || 825 (checkMe == FDISK_EXT_WIN) || (checkMe == FDISK_FAT95) || 826 (checkMe == DIAGPART)); 827 } 828 829 /* 830 * isDosExtended() 831 * Boolean function. Give it the systid field for an fdisk partition 832 * and it decides if that's a systid that describes an extended DOS 833 * partition. 834 */ 835 static int 836 isDosExtended(uchar_t checkMe) 837 { 838 return ((checkMe == EXTDOS) || (checkMe == FDISK_EXTLBA)); 839 } 840 841 /* 842 * isBootPart() 843 * Boolean function. Give it the systid field for an fdisk partition 844 * and it decides if that's a systid that describes a Solaris boot 845 * partition. 846 */ 847 static int 848 isBootPart(uchar_t checkMe) 849 { 850 return (checkMe == X86BOOT); 851 } 852 853 /* 854 * noLogicalDrive() 855 * Display error message about not being able to find a logical 856 * drive. 857 */ 858 static void 859 noLogicalDrive(int requested) 860 { 861 if (requested == BOOT_PARTITION_DRIVE) { 862 cmn_err(CE_NOTE, "!pcfs: no boot partition"); 863 } else { 864 cmn_err(CE_NOTE, "!pcfs: no such logical drive"); 865 } 866 } 867 868 /* 869 * findTheDrive() 870 * Discover offset of the requested logical drive, and return 871 * that offset (startSector), the systid of that drive (sysid), 872 * and a buffer pointer (bp), with the buffer contents being 873 * the first sector of the logical drive (i.e., the sector that 874 * contains the BPB for that drive). 875 */ 876 static int 877 findTheDrive(dev_t dev, int askedFor, int *error, buf_t **bp, 878 daddr_t *startSector, uchar_t *sysid) 879 { 880 struct ipart dosp[FD_NUMPART]; /* incore fdisk partition structure */ 881 struct mboot *dosp_ptr; /* boot structure pointer */ 882 daddr_t lastseek = 0; /* Disk block we sought previously */ 883 daddr_t diskblk = 0; /* Disk block to get */ 884 daddr_t xstartsect; /* base of Extended DOS partition */ 885 int logicalDriveCount = 0; /* Count of logical drives seen */ 886 int extendedPart = -1; /* index of extended dos partition */ 887 int primaryPart = -1; /* index of primary dos partition */ 888 int bootPart = -1; /* index of a Solaris boot partition */ 889 int xnumsect = -1; /* length of extended DOS partition */ 890 int driveIndex; /* computed FDISK table index */ 891 int i; 892 /* 893 * Count of drives in the current extended partition's 894 * FDISK table, and indexes of the drives themselves. 895 */ 896 int extndDrives[FD_NUMPART]; 897 int numDrives = 0; 898 899 /* 900 * Count of drives (beyond primary) in master boot record's 901 * FDISK table, and indexes of the drives themselves. 902 */ 903 int extraDrives[FD_NUMPART]; 904 int numExtraDrives = 0; 905 906 /* 907 * Copy from disk block into memory aligned structure for fdisk usage. 908 */ 909 dosp_ptr = (struct mboot *)(*bp)->b_un.b_addr; 910 bcopy(dosp_ptr->parts, dosp, sizeof (struct ipart) * FD_NUMPART); 911 912 /* 913 * Get a summary of what is in the Master FDISK table. 914 * Normally we expect to find one partition marked as a DOS drive. 915 * This partition is the one Windows calls the primary dos partition. 916 * If the machine has any logical drives then we also expect 917 * to find a partition marked as an extended DOS partition. 918 * 919 * Sometimes we'll find multiple partitions marked as DOS drives. 920 * The Solaris fdisk program allows these partitions 921 * to be created, but Windows fdisk no longer does. We still need 922 * to support these, though, since Windows does. We also need to fix 923 * our fdisk to behave like the Windows version. 924 * 925 * It turns out that some off-the-shelf media have *only* an 926 * Extended partition, so we need to deal with that case as well. 927 * 928 * Only a single (the first) Extended or Boot Partition will 929 * be recognized. Any others will be ignored. 930 */ 931 for (i = 0; i < FD_NUMPART; i++) { 932 PC_DPRINTF1(2, "findTheDrive: found partition type %02x", 933 dosp[i].systid); 934 935 if (isDosDrive(dosp[i].systid)) { 936 if (primaryPart < 0) { 937 logicalDriveCount++; 938 primaryPart = i; 939 } else { 940 extraDrives[numExtraDrives++] = i; 941 } 942 continue; 943 } 944 if ((extendedPart < 0) && isDosExtended(dosp[i].systid)) { 945 extendedPart = i; 946 continue; 947 } 948 if ((bootPart < 0) && isBootPart(dosp[i].systid)) { 949 bootPart = i; 950 continue; 951 } 952 } 953 954 if (askedFor == BOOT_PARTITION_DRIVE) { 955 if (bootPart < 0) { 956 noLogicalDrive(askedFor); 957 *error = EINVAL; 958 return (0); 959 } 960 *sysid = dosp[bootPart].systid; 961 *startSector = ltohi(dosp[bootPart].relsect); 962 return (1); 963 } 964 965 if (askedFor == PRIMARY_DOS_DRIVE && primaryPart >= 0) { 966 *sysid = dosp[primaryPart].systid; 967 *startSector = ltohi(dosp[primaryPart].relsect); 968 return (1); 969 } 970 971 /* 972 * We are not looking for the C: drive (or the primary drive 973 * was not found), so we had better have an extended partition 974 * or extra drives in the Master FDISK table. 975 */ 976 if ((extendedPart < 0) && (numExtraDrives == 0)) { 977 cmn_err(CE_NOTE, "!pcfs: no extended dos partition"); 978 noLogicalDrive(askedFor); 979 *error = EINVAL; 980 return (0); 981 } 982 983 if (extendedPart >= 0) { 984 diskblk = xstartsect = ltohi(dosp[extendedPart].relsect); 985 xnumsect = ltohi(dosp[extendedPart].numsect); 986 do { 987 /* 988 * If the seek would not cause us to change 989 * position on the drive, then we're out of 990 * extended partitions to examine. 991 */ 992 if (diskblk == lastseek) 993 break; 994 logicalDriveCount += numDrives; 995 /* 996 * Seek the next extended partition, and find 997 * logical drives within it. 998 */ 999 brelse(*bp); 1000 *bp = bread(dev, diskblk, PC_SAFESECSIZE); 1001 if ((*bp)->b_flags & B_ERROR) { 1002 PC_DPRINTF0(1, "pc_getfattype: read error\n"); 1003 *error = EIO; 1004 return (0); 1005 } 1006 lastseek = diskblk; 1007 dosp_ptr = (struct mboot *)(*bp)->b_un.b_addr; 1008 if (ltohs(dosp_ptr->signature) != MBB_MAGIC) { 1009 cmn_err(CE_NOTE, "!pcfs: " 1010 "extended partition signature err"); 1011 *error = EINVAL; 1012 return (0); 1013 } 1014 bcopy(dosp_ptr->parts, dosp, 1015 sizeof (struct ipart) * FD_NUMPART); 1016 /* 1017 * Count up drives, and track where the next 1018 * extended partition is in case we need it. We 1019 * are expecting only one extended partition. If 1020 * there is more than one we'll only go to the 1021 * first one we see, but warn about ignoring. 1022 */ 1023 numDrives = 0; 1024 for (i = 0; i < FD_NUMPART; i++) { 1025 if (isDosDrive(dosp[i].systid)) { 1026 extndDrives[numDrives++] = i; 1027 continue; 1028 } else if (isDosExtended(dosp[i].systid)) { 1029 if (diskblk != lastseek) { 1030 /* 1031 * Already found an extended 1032 * partition in this table. 1033 */ 1034 cmn_err(CE_NOTE, 1035 "!pcfs: ignoring unexpected" 1036 " additional extended" 1037 " partition"); 1038 continue; 1039 } 1040 diskblk = xstartsect + 1041 ltohi(dosp[i].relsect); 1042 continue; 1043 } 1044 } 1045 } while (askedFor > logicalDriveCount + numDrives); 1046 1047 if (askedFor <= logicalDriveCount + numDrives) { 1048 /* 1049 * The number of logical drives we've found thus 1050 * far is enough to get us to the one we were 1051 * searching for. 1052 */ 1053 driveIndex = logicalDriveCount + numDrives - askedFor; 1054 *sysid = dosp[extndDrives[driveIndex]].systid; 1055 *startSector = 1056 ltohi(dosp[extndDrives[driveIndex]].relsect) + 1057 lastseek; 1058 if (*startSector > (xstartsect + xnumsect)) { 1059 cmn_err(CE_NOTE, "!pcfs: extended partition " 1060 "values bad"); 1061 *error = EINVAL; 1062 return (0); 1063 } 1064 return (1); 1065 } else { 1066 /* 1067 * We ran out of extended dos partition 1068 * drives. The only hope now is to go 1069 * back to extra drives defined in the master 1070 * fdisk table. But we overwrote that table 1071 * already, so we must load it in again. 1072 */ 1073 logicalDriveCount += numDrives; 1074 brelse(*bp); 1075 *bp = bread(dev, (daddr_t)0, PC_SAFESECSIZE); 1076 if ((*bp)->b_flags & B_ERROR) { 1077 PC_DPRINTF0(1, "pc_getfattype: read error\n"); 1078 *error = EIO; 1079 return (0); 1080 } 1081 dosp_ptr = (struct mboot *)(*bp)->b_un.b_addr; 1082 bcopy(dosp_ptr->parts, dosp, 1083 sizeof (struct ipart) * FD_NUMPART); 1084 } 1085 } 1086 /* 1087 * Still haven't found the drive, is it an extra 1088 * drive defined in the main FDISK table? 1089 */ 1090 if (askedFor <= logicalDriveCount + numExtraDrives) { 1091 driveIndex = logicalDriveCount + numExtraDrives - askedFor; 1092 *sysid = dosp[extraDrives[driveIndex]].systid; 1093 *startSector = ltohi(dosp[extraDrives[driveIndex]].relsect); 1094 return (1); 1095 } 1096 /* 1097 * Still haven't found the drive, and there is 1098 * nowhere else to look. 1099 */ 1100 noLogicalDrive(askedFor); 1101 *error = EINVAL; 1102 return (0); 1103 } 1104 1105 /* 1106 * FAT12/FAT16 specific consistency checks. 1107 */ 1108 static int 1109 check_bpb_fat16(struct bootsec *bpb) 1110 { 1111 if (pcfsdebuglevel >= 5) { 1112 PC_DPRINTF1(5, "check_bpb_fat: RootEntCount = %d", 1113 ltohs(bpb->rdirents[0])); 1114 PC_DPRINTF1(5, "check_bpb_fat16: TotSec16 = %d", 1115 ltohs(bpb->numsect[0])); 1116 PC_DPRINTF1(5, "check_bpb_fat16: FATSz16 = %d", 1117 ltohs(bpb->fatsec)); 1118 PC_DPRINTF1(5, "check_bpb_fat16: TotSec32 = %d", 1119 ltohi(bpb->totalsec)); 1120 } 1121 return (ltohs(bpb->rdirents[0]) > 0 && /* RootEntCnt > 0 */ 1122 ((ltohs(bpb->numsect[0]) == 0 && /* TotSec16 == 0 */ 1123 ltohi(bpb->totalsec) > 0) || /* TotSec32 > 0 */ 1124 ltohs(bpb->numsect[0]) > 0) && /* TotSec16 > 0 */ 1125 ltohs(bpb->fatsec) > 0); /* FatSz16 > 0 */ 1126 } 1127 1128 /* 1129 * FAT32 specific consistency checks. 1130 */ 1131 static int 1132 check_bpb_fat32(struct fat32_bootsec *bpb) 1133 { 1134 if (pcfsdebuglevel >= 5) { 1135 PC_DPRINTF1(5, "check_bpb_fat32: RootEntCount = %d", 1136 ltohs(bpb->f_bs.rdirents[0])); 1137 PC_DPRINTF1(5, "check_bpb_fat32: TotSec16 = %d", 1138 ltohs(bpb->f_bs.numsect[0])); 1139 PC_DPRINTF1(5, "check_bpb_fat32: FATSz16 = %d", 1140 ltohs(bpb->f_bs.fatsec)); 1141 PC_DPRINTF1(5, "check_bpb_fat32: TotSec32 = %d", 1142 ltohi(bpb->f_bs.totalsec)); 1143 PC_DPRINTF1(5, "check_bpb_fat32: FATSz32 = %d", 1144 ltohi(bpb->f_fatlength)); 1145 } 1146 return (ltohs(bpb->f_bs.rdirents[0]) == 0 && 1147 ltohs(bpb->f_bs.numsect[0]) == 0 && 1148 ltohs(bpb->f_bs.fatsec) == 0 && 1149 ltohi(bpb->f_bs.totalsec) > 0 && 1150 ltohi(bpb->f_fatlength) > 0); 1151 } 1152 1153 /* 1154 * Calculate the number of clusters in order to determine 1155 * the type of FAT we are looking at. This is the only 1156 * recommended way of determining FAT type, though there 1157 * are other hints in the data, this is the best way. 1158 */ 1159 static ulong_t 1160 bpb_to_numclusters(uchar_t *cp) 1161 { 1162 struct fat32_bootsec *bpb; 1163 1164 ulong_t rootdirsectors; 1165 ulong_t FATsz; 1166 ulong_t TotSec; 1167 ulong_t DataSec; 1168 ulong_t CountOfClusters; 1169 char FileSysType[9]; 1170 1171 /* 1172 * Cast it to FAT32 bpb. If it turns out to be FAT12/16, its 1173 * OK, we won't try accessing the data beyond the FAT16 header 1174 * boundary. 1175 */ 1176 bpb = (struct fat32_bootsec *)cp; 1177 1178 if (pcfsdebuglevel >= 5) { 1179 if (ltohs(bpb->f_bs.rdirents[0]) != 0) { 1180 (void) memcpy(FileSysType, &cp[54], 8); 1181 FileSysType[8] = 0; 1182 PC_DPRINTF1(5, "debug_bpb: FAT12/FAT16 FileSysType = " 1183 "%s", FileSysType); 1184 } 1185 } 1186 1187 rootdirsectors = ((ltohs(bpb->f_bs.rdirents[0]) * 32) + 1188 (ltohs(bpb->f_bs.bps[0]) - 1)) / ltohs(bpb->f_bs.bps[0]); 1189 1190 if (ltohs(bpb->f_bs.fatsec) != 0) 1191 FATsz = ltohs(bpb->f_bs.fatsec); 1192 else 1193 FATsz = ltohi(bpb->f_fatlength); 1194 1195 if (ltohs(bpb->f_bs.numsect[0]) != 0) 1196 TotSec = ltohs(bpb->f_bs.numsect[0]); 1197 else 1198 TotSec = ltohi(bpb->f_bs.totalsec); 1199 1200 DataSec = TotSec - (ltohs(bpb->f_bs.res_sec[0]) + 1201 (bpb->f_bs.nfat * FATsz) + rootdirsectors); 1202 1203 CountOfClusters = DataSec / bpb->f_bs.spcl; 1204 1205 PC_DPRINTF1(5, "debug_bpb: CountOfClusters = %ld", CountOfClusters); 1206 1207 return (CountOfClusters); 1208 1209 } 1210 1211 static int 1212 fattype(ulong_t CountOfClusters) 1213 { 1214 /* 1215 * From Microsoft: 1216 * In the following example, when it says <, it does not mean <=. 1217 * Note also that the numbers are correct. The first number for 1218 * FAT12 is 4085; the second number for FAT16 is 65525. These numbers 1219 * and the '<' signs are not wrong. 1220 */ 1221 1222 /* Watch for edge cases */ 1223 if ((CountOfClusters >= 4085 && CountOfClusters <= 4095) || 1224 (CountOfClusters >= 65525 && CountOfClusters <= 65535)) { 1225 PC_DPRINTF1(5, "debug_bpb: Cannot determine FAT yet - %ld", 1226 CountOfClusters); 1227 return (-1); /* Cannot be determined yet */ 1228 } else if (CountOfClusters < 4085) { 1229 /* Volume is FAT12 */ 1230 PC_DPRINTF0(5, "debug_bpb: This must be FAT12"); 1231 return (0); 1232 } else if (CountOfClusters < 65525) { 1233 /* Volume is FAT16 */ 1234 PC_DPRINTF0(5, "debug_bpb: This must be FAT16"); 1235 return (PCFS_FAT16); 1236 } else { 1237 /* Volume is FAT32 */ 1238 PC_DPRINTF0(5, "debug_bpb: This must be FAT32"); 1239 return (PCFS_FAT32); 1240 } 1241 } 1242 1243 #define VALID_SECSIZE(s) (s == 512 || s == 1024 || s == 2048 || s == 4096) 1244 1245 #define VALID_SPCL(s) (s == 1 || s == 2 || s == 4 || s == 8 || s == 16 ||\ 1246 s == 32 || s == 64 || s == 128) 1247 1248 static int 1249 secondaryBPBChecks(uchar_t *cp) 1250 { 1251 struct bootsec *bpb = (struct bootsec *)cp; 1252 struct fat32_bootsec *f32bpb = (struct fat32_bootsec *)cp; 1253 1254 /* 1255 * Perform secondary checks to try and determine what sort 1256 * of FAT partition we have based on other, less reliable, 1257 * data in the BPB header. 1258 */ 1259 if (ltohs(bpb->fatsec) != 0) { 1260 /* 1261 * Must be FAT12 or FAT16, check the 1262 * FilSysType string (not 100% reliable). 1263 */ 1264 if (!memcmp((cp + PCFS_TYPESTRING_OFFSET16), "FAT12", 5)) { 1265 PC_DPRINTF0(5, "secondaryBPBCheck says: FAT12"); 1266 return (0); /* FAT12 */ 1267 } else if (!memcmp((cp + PCFS_TYPESTRING_OFFSET16), "FAT16", 1268 5)) { 1269 PC_DPRINTF0(5, "secondaryBPBCheck says: FAT16"); 1270 return (PCFS_FAT16); 1271 } else { 1272 /* 1273 * Try to use the BPB_Media byte 1274 * 1275 * If the media byte indicates a floppy we'll 1276 * assume FAT12, otherwise we'll assume FAT16. 1277 */ 1278 switch (bpb->mediadesriptor) { 1279 case SS8SPT: 1280 case DS8SPT: 1281 case SS9SPT: 1282 case DS9SPT: 1283 case DS18SPT: 1284 case DS9_15SPT: 1285 PC_DPRINTF0(5, 1286 "secondaryBPBCheck says: FAT12"); 1287 return (0); /* FAT12 */ 1288 case MD_FIXED: 1289 PC_DPRINTF0(5, 1290 "secondaryBPBCheck says: FAT16"); 1291 return (PCFS_FAT16); 1292 default: 1293 cmn_err(CE_NOTE, 1294 "!pcfs: unknown FAT type"); 1295 return (-1); 1296 } 1297 } 1298 } else if (ltohi(f32bpb->f_fatlength) > 0) { 1299 PC_DPRINTF0(5, "secondaryBPBCheck says: FAT32"); 1300 return (PCFS_FAT32); 1301 } else { 1302 /* We don't know */ 1303 PC_DPRINTF0(5, "secondaryBPBCheck says: unknown!!"); 1304 return (-1); 1305 } 1306 } 1307 1308 /* 1309 * Check to see if the BPB we found is correct. 1310 * 1311 * First, look for obvious, tell-tale signs of trouble: 1312 * The NumFATs value should always be 2. Sometimes it can be a '1' 1313 * on FLASH memory cards and other non-disk-based media, so we 1314 * will allow that as well. 1315 * 1316 * We also look at the Media byte, the valid range is 0xF0, or 1317 * 0xF8 thru 0xFF, anything else means this is probably not a good 1318 * BPB. 1319 * 1320 * Finally, check the BPB Magic number at the end of the 512 byte 1321 * block, it must be 0xAA55. 1322 * 1323 * If that all is good, calculate the number of clusters and 1324 * do some final verification steps. 1325 * 1326 * If all is well, return success (1) and set the fattypep 1327 * value to the correct FAT value. 1328 */ 1329 static int 1330 isBPB(uchar_t *cp, int *fattypep) 1331 { 1332 struct bootsec *bpb = (struct bootsec *)cp; 1333 1334 uint_t numclusters; /* number of clusters in file area */ 1335 ushort_t secsize = (int)ltohs(bpb->bps[0]); 1336 1337 if (pcfsdebuglevel >= 3) { 1338 if (!VALID_SECSIZE(secsize)) 1339 PC_DPRINTF1(3, "check_bpb: invalid bps value %d", 1340 secsize); 1341 1342 if (!VALID_SPCL(bpb->spcl)) 1343 PC_DPRINTF1(3, "check_bpb: invalid spcl value %d", 1344 bpb->spcl); 1345 1346 if ((secsize * bpb->spcl) >= (32 * 1024)) 1347 PC_DPRINTF3(3, "check_bpb: BPC > 32K %d x %d = %d", 1348 secsize, 1349 bpb->spcl, 1350 secsize * bpb->spcl); 1351 1352 if (bpb->nfat == 0) 1353 PC_DPRINTF1(3, "check_bpb: bad NumFATs value %d", 1354 bpb->nfat); 1355 1356 if (ltohs(bpb->res_sec[0]) == 0) 1357 PC_DPRINTF1(3, "check_bpb: bad RsvdSecCnt value %d", 1358 ltohs(bpb->res_sec[0])); 1359 1360 PC_DPRINTF1(5, "check_bpb: Media byte = %02x", 1361 bpb->mediadesriptor); 1362 1363 } 1364 if ((bpb->nfat == 0) || 1365 (bpb->mediadesriptor != 0xF0 && bpb->mediadesriptor < 0xF8) || 1366 (ltohs(cp[510]) != MBB_MAGIC) || 1367 !VALID_SECSIZE(secsize) || 1368 !VALID_SPCL(bpb->spcl) || 1369 (secsize * bpb->spcl >= (64 * 1024)) || 1370 !(ltohs(bpb->res_sec[0]))) 1371 return (0); 1372 1373 /* 1374 * Basic sanity checks passed so far, now try to determine which 1375 * FAT format to use. 1376 */ 1377 numclusters = bpb_to_numclusters(cp); 1378 1379 *fattypep = fattype(numclusters); 1380 1381 /* Do some final sanity checks for each specific type of FAT */ 1382 switch (*fattypep) { 1383 case 0: /* FAT12 */ 1384 case PCFS_FAT16: 1385 if (!check_bpb_fat16((struct bootsec *)cp)) 1386 return (0); 1387 break; 1388 case PCFS_FAT32: 1389 if (!check_bpb_fat32((struct fat32_bootsec *)cp)) 1390 return (0); 1391 break; 1392 default: /* not sure yet */ 1393 *fattypep = secondaryBPBChecks(cp); 1394 if (*fattypep == -1) { 1395 /* Still nothing, give it up. */ 1396 return (0); 1397 } 1398 break; 1399 } 1400 PC_DPRINTF0(5, "isBPB: BPB passes verification tests"); 1401 return (1); 1402 } 1403 1404 1405 /* 1406 * Get the FAT type for the DOS medium. 1407 * 1408 * ------------------------- 1409 * According to Microsoft: 1410 * The FAT type one of FAT12, FAT16, or FAT32 is determined by the 1411 * count of clusters on the volume and nothing else. 1412 * ------------------------- 1413 * 1414 */ 1415 static int 1416 pc_getfattype( 1417 struct vnode *devvp, 1418 int ldrive, 1419 daddr_t *strtsectp, 1420 int *fattypep) 1421 { 1422 uchar_t *cp; /* for searching out FAT string */ 1423 buf_t *bp = NULL; /* Disk buffer pointer */ 1424 int rval = 0; 1425 uchar_t sysid = 0; /* System ID character */ 1426 dev_t dev = devvp->v_rdev; 1427 1428 *strtsectp = (daddr_t)0; 1429 1430 /* 1431 * Open the device so we can check out the BPB or FDISK table, 1432 * then read in the sector. 1433 */ 1434 PC_DPRINTF2(5, "pc_getfattype: dev=%x ldrive=%x ", (int)dev, ldrive); 1435 if (rval = VOP_OPEN(&devvp, FREAD, CRED())) { 1436 PC_DPRINTF1(1, "pc_getfattype: open error=%d\n", rval); 1437 return (rval); 1438 } 1439 1440 /* 1441 * Read block 0 from device 1442 */ 1443 bp = bread(dev, (daddr_t)0, PC_SAFESECSIZE); 1444 if (bp->b_flags & B_ERROR) { 1445 PC_DPRINTF0(1, "pc_getfattype: read error\n"); 1446 rval = EIO; 1447 goto out; 1448 } 1449 1450 cp = (uchar_t *)bp->b_un.b_addr; 1451 1452 /* 1453 * If the first block is not a valid BPB, look for the 1454 * through the FDISK table. 1455 */ 1456 if (!isBPB(cp, fattypep)) { 1457 /* find the partition table and get 512 bytes from it. */ 1458 PC_DPRINTF0(5, "pc_getfattype: using FDISK table to find BPB"); 1459 1460 if (findTheDrive(dev, ldrive, &rval, &bp, 1461 strtsectp, &sysid) == 0) 1462 goto out; 1463 1464 brelse(bp); 1465 bp = bread(dev, *strtsectp, PC_SAFESECSIZE); 1466 if (bp->b_flags & B_ERROR) { 1467 PC_DPRINTF0(1, "pc_getfattype: read error\n"); 1468 rval = EIO; 1469 goto out; 1470 } 1471 cp = (uchar_t *)bp->b_un.b_addr; 1472 1473 /* If this one is still no good, give it up. */ 1474 if (!isBPB(cp, fattypep)) { 1475 rval = EIO; 1476 goto out; 1477 } 1478 } 1479 1480 out: 1481 /* 1482 * Release the buffer used 1483 */ 1484 if (bp != NULL) 1485 brelse(bp); 1486 (void) VOP_CLOSE(devvp, FREAD, 1, (offset_t)0, CRED()); 1487 return (rval); 1488 } 1489 1490 1491 /* 1492 * Get the boot parameter block and file allocation table. 1493 * If there is an old FAT, invalidate it. 1494 */ 1495 int 1496 pc_getfat(struct pcfs *fsp) 1497 { 1498 struct vfs *vfsp = PCFSTOVFS(fsp); 1499 struct buf *tp = 0; 1500 struct buf *bp = 0; 1501 uchar_t *fatp = NULL; 1502 uchar_t *fat_changemap = NULL; 1503 struct bootsec *bootp; 1504 struct fat32_bootsec *f32b; 1505 struct vnode *devvp; 1506 int error; 1507 int fatsize; 1508 int fat_changemapsize; 1509 int flags = 0; 1510 int nfat; 1511 int secsize; 1512 int fatsec; 1513 1514 PC_DPRINTF0(5, "pc_getfat\n"); 1515 devvp = fsp->pcfs_devvp; 1516 if (fsp->pcfs_fatp) { 1517 /* 1518 * There is a FAT in core. 1519 * If there are open file pcnodes or we have modified it or 1520 * it hasn't timed out yet use the in core FAT. 1521 * Otherwise invalidate it and get a new one 1522 */ 1523 #ifdef notdef 1524 if (fsp->pcfs_frefs || 1525 (fsp->pcfs_flags & PCFS_FATMOD) || 1526 (gethrestime_sec() < fsp->pcfs_fattime)) { 1527 return (0); 1528 } else { 1529 mutex_enter(&pcfslock); 1530 pc_invalfat(fsp); 1531 mutex_exit(&pcfslock); 1532 } 1533 #endif /* notdef */ 1534 return (0); 1535 } 1536 /* 1537 * Open block device mounted on. 1538 */ 1539 error = VOP_OPEN(&devvp, 1540 (vfsp->vfs_flag & VFS_RDONLY) ? FREAD : FREAD|FWRITE, 1541 CRED()); 1542 if (error) { 1543 PC_DPRINTF1(1, "pc_getfat: open error=%d\n", error); 1544 return (error); 1545 } 1546 /* 1547 * Get boot parameter block and check it for validity 1548 */ 1549 tp = bread(fsp->pcfs_xdev, fsp->pcfs_dosstart, PC_SAFESECSIZE); 1550 if (tp->b_flags & (B_ERROR | B_STALE)) { 1551 PC_DPRINTF0(1, "pc_getfat: boot block error\n"); 1552 flags = tp->b_flags & B_ERROR; 1553 error = EIO; 1554 goto out; 1555 } 1556 tp->b_flags |= B_STALE | B_AGE; 1557 bootp = (struct bootsec *)tp->b_un.b_addr; 1558 1559 1560 /* get the sector size - may be more than 512 bytes */ 1561 secsize = (int)ltohs(bootp->bps[0]); 1562 /* check for bogus sector size - fat should be at least 1 sector */ 1563 if (IS_FAT32(fsp)) { 1564 f32b = (struct fat32_bootsec *)bootp; 1565 fatsec = ltohi(f32b->f_fatlength); 1566 } else { 1567 fatsec = ltohs(bootp->fatsec); 1568 } 1569 if (secsize < 512 || fatsec < 1 || bootp->nfat < 1) { 1570 cmn_err(CE_NOTE, "!pcfs: FAT size error"); 1571 error = EINVAL; 1572 goto out; 1573 } 1574 1575 switch (bootp->mediadesriptor) { 1576 default: 1577 cmn_err(CE_NOTE, "!pcfs: media-descriptor error, 0x%x", 1578 bootp->mediadesriptor); 1579 error = EINVAL; 1580 goto out; 1581 1582 case MD_FIXED: 1583 /* 1584 * PCMCIA pseudo floppy is type MD_FIXED, 1585 * but is accessed like a floppy 1586 */ 1587 if (!(fsp->pcfs_flags & PCFS_PCMCIA_NO_CIS)) { 1588 fsp->pcfs_flags |= PCFS_NOCHK; 1589 } 1590 /* FALLTHRU */ 1591 case SS8SPT: 1592 case DS8SPT: 1593 case SS9SPT: 1594 case DS9SPT: 1595 case DS18SPT: 1596 case DS9_15SPT: 1597 fsp->pcfs_secsize = secsize; 1598 fsp->pcfs_sdshift = secsize / DEV_BSIZE - 1; 1599 fsp->pcfs_entps = secsize / sizeof (struct pcdir); 1600 fsp->pcfs_spcl = (int)bootp->spcl; 1601 fsp->pcfs_fatsec = fatsec; 1602 fsp->pcfs_spt = (int)ltohs(bootp->spt); 1603 fsp->pcfs_rdirsec = (int)ltohs(bootp->rdirents[0]) 1604 * sizeof (struct pcdir) / secsize; 1605 fsp->pcfs_clsize = fsp->pcfs_spcl * secsize; 1606 fsp->pcfs_fatstart = fsp->pcfs_dosstart + 1607 (daddr_t)ltohs(bootp->res_sec[0]); 1608 fsp->pcfs_rdirstart = fsp->pcfs_fatstart + 1609 (bootp->nfat * fsp->pcfs_fatsec); 1610 fsp->pcfs_datastart = fsp->pcfs_rdirstart + fsp->pcfs_rdirsec; 1611 if (IS_FAT32(fsp)) 1612 fsp->pcfs_rdirstart = ltohi(f32b->f_rootcluster); 1613 fsp->pcfs_ncluster = (((int)(ltohs(bootp->numsect[0]) ? 1614 ltohs(bootp->numsect[0]) : ltohi(bootp->totalsec))) - 1615 fsp->pcfs_datastart + fsp->pcfs_dosstart) / fsp->pcfs_spcl; 1616 fsp->pcfs_numfat = (int)bootp->nfat; 1617 fsp->pcfs_nxfrecls = PCF_FIRSTCLUSTER; 1618 break; 1619 } 1620 1621 /* 1622 * Get FAT and check it for validity 1623 */ 1624 fatsize = fsp->pcfs_fatsec * fsp->pcfs_secsize; 1625 fatp = kmem_alloc(fatsize, KM_SLEEP); 1626 error = pc_readfat(fsp, fatp, fsp->pcfs_fatstart, fatsize); 1627 if (error) { 1628 flags = B_ERROR; 1629 goto out; 1630 } 1631 fat_changemapsize = (fatsize / fsp->pcfs_clsize) + 1; 1632 fat_changemap = kmem_zalloc(fat_changemapsize, KM_SLEEP); 1633 1634 /* 1635 * The only definite signature check is that the 1636 * media descriptor byte should match the first byte 1637 * of the FAT block. 1638 */ 1639 if (fatp[0] != bootp->mediadesriptor) { 1640 cmn_err(CE_NOTE, "!pcfs: FAT signature error"); 1641 error = EINVAL; 1642 goto out; 1643 } 1644 /* 1645 * Checking for fatsec and number of supported clusters, should 1646 * actually determine a FAT12/FAT media. 1647 */ 1648 if (fsp->pcfs_flags & PCFS_FAT16) { 1649 if ((fsp->pcfs_fatsec <= 12) && 1650 ((fatsize * 2 / 3) >= fsp->pcfs_ncluster)) { 1651 /* 1652 * We have a 12-bit FAT, rather than a 16-bit FAT. 1653 * Ignore what the fdisk table says. 1654 */ 1655 PC_DPRINTF0(2, "pc_getfattype: forcing 12-bit FAT\n"); 1656 fsp->pcfs_flags &= ~PCFS_FAT16; 1657 } 1658 } 1659 /* 1660 * Sanity check our FAT is large enough for the 1661 * clusters we think we have. 1662 */ 1663 if ((fsp->pcfs_flags & PCFS_FAT16) && 1664 ((fatsize / 2) < fsp->pcfs_ncluster)) { 1665 cmn_err(CE_NOTE, "!pcfs: FAT too small for number of clusters"); 1666 error = EINVAL; 1667 goto out; 1668 } 1669 1670 /* 1671 * Get alternate FATs and check for consistency 1672 * This is an inlined version of pc_readfat(). 1673 * Since we're only comparing FAT and alternate FAT, 1674 * there's no reason to let pc_readfat() copy data out 1675 * of the buf. Instead, compare in-situ, one cluster 1676 * at a time. 1677 */ 1678 for (nfat = 1; nfat < fsp->pcfs_numfat; nfat++) { 1679 size_t startsec; 1680 size_t off; 1681 1682 startsec = fsp->pcfs_fatstart + nfat * fsp->pcfs_fatsec; 1683 1684 for (off = 0; off < fatsize; off += fsp->pcfs_clsize) { 1685 bp = bread(fsp->pcfs_xdev, pc_dbdaddr(fsp, 1686 startsec + 1687 pc_cltodb(fsp, pc_lblkno(fsp, off))), 1688 MIN(fsp->pcfs_clsize, fatsize - off)); 1689 if (bp->b_flags & (B_ERROR | B_STALE)) { 1690 cmn_err(CE_NOTE, 1691 "!pcfs: alternate FAT #%d read error" 1692 " at byte %ld", nfat, off); 1693 flags = B_ERROR; 1694 error = EIO; 1695 goto out; 1696 } 1697 bp->b_flags |= B_STALE | B_AGE; 1698 if (bcmp(bp->b_un.b_addr, 1699 fatp + off, 1700 MIN(fsp->pcfs_clsize, fatsize - off))) { 1701 cmn_err(CE_NOTE, 1702 "!pcfs: alternate FAT #%d corrupted" 1703 " at byte %ld", nfat, off); 1704 flags = B_ERROR; 1705 } 1706 brelse(bp); 1707 bp = NULL; /* prevent double release */ 1708 } 1709 } 1710 1711 fsp->pcfs_fatsize = fatsize; 1712 fsp->pcfs_fatp = fatp; 1713 fsp->pcfs_fat_changemapsize = fat_changemapsize; 1714 fsp->pcfs_fat_changemap = fat_changemap; 1715 fsp->pcfs_fattime = gethrestime_sec() + PCFS_DISKTIMEOUT; 1716 fsp->pcfs_fatjustread = 1; 1717 1718 brelse(tp); 1719 tp = NULL; 1720 if (IS_FAT32(fsp)) { 1721 /* get fsinfo */ 1722 struct fat32_boot_fsinfo fsinfo_disk; 1723 1724 fsp->f32fsinfo_sector = ltohs(f32b->f_infosector); 1725 tp = bread(fsp->pcfs_xdev, 1726 fsp->pcfs_dosstart + pc_dbdaddr(fsp, fsp->f32fsinfo_sector), 1727 PC_SAFESECSIZE); 1728 if (tp->b_flags & (B_ERROR | B_STALE)) { 1729 cmn_err(CE_NOTE, "!pcfs: error reading fat32 fsinfo"); 1730 flags = tp->b_flags & B_ERROR; 1731 brelse(tp); 1732 tp = NULL; 1733 error = EIO; 1734 goto out; 1735 } 1736 tp->b_flags |= B_STALE | B_AGE; 1737 bcopy((void *)(tp->b_un.b_addr + FAT32_BOOT_FSINFO_OFF), 1738 &fsinfo_disk, sizeof (struct fat32_boot_fsinfo)); 1739 brelse(tp); 1740 tp = NULL; 1741 1742 /* translated fields */ 1743 fsp->fsinfo_native.fs_signature = 1744 ltohi(fsinfo_disk.fs_signature); 1745 fsp->fsinfo_native.fs_free_clusters = 1746 ltohi(fsinfo_disk.fs_free_clusters); 1747 if (fsp->fsinfo_native.fs_signature != FAT32_FS_SIGN) { 1748 cmn_err(CE_NOTE, 1749 "!pcfs: fat32 fsinfo signature mismatch."); 1750 error = EINVAL; 1751 goto out; 1752 } 1753 } 1754 1755 return (0); 1756 1757 out: 1758 cmn_err(CE_NOTE, "!pcfs: illegal disk format"); 1759 if (tp) 1760 brelse(tp); 1761 if (bp) 1762 brelse(bp); 1763 if (fatp) 1764 kmem_free(fatp, fatsize); 1765 if (fat_changemap) 1766 kmem_free(fat_changemap, fat_changemapsize); 1767 1768 if (flags) { 1769 pc_mark_irrecov(fsp); 1770 } 1771 (void) VOP_CLOSE(devvp, (vfsp->vfs_flag & VFS_RDONLY) ? 1772 FREAD : FREAD|FWRITE, 1, (offset_t)0, CRED()); 1773 return (error); 1774 } 1775 1776 int 1777 pc_syncfat(struct pcfs *fsp) 1778 { 1779 struct buf *bp; 1780 int nfat; 1781 int error; 1782 struct fat32_boot_fsinfo fsinfo_disk; 1783 1784 PC_DPRINTF0(7, "pcfs_syncfat\n"); 1785 if ((fsp->pcfs_fatp == (uchar_t *)0) || 1786 !(fsp->pcfs_flags & PCFS_FATMOD)) 1787 return (0); 1788 /* 1789 * write out all copies of FATs 1790 */ 1791 fsp->pcfs_flags &= ~PCFS_FATMOD; 1792 fsp->pcfs_fattime = gethrestime_sec() + PCFS_DISKTIMEOUT; 1793 for (nfat = 0; nfat < fsp->pcfs_numfat; nfat++) { 1794 error = pc_writefat(fsp, 1795 fsp->pcfs_fatstart + nfat*fsp->pcfs_fatsec); 1796 if (error) { 1797 pc_mark_irrecov(fsp); 1798 return (EIO); 1799 } 1800 } 1801 pc_clear_fatchanges(fsp); 1802 PC_DPRINTF0(6, "pcfs_syncfat: wrote out FAT\n"); 1803 /* write out fsinfo */ 1804 if (IS_FAT32(fsp)) { 1805 bp = bread(fsp->pcfs_xdev, 1806 fsp->pcfs_dosstart + pc_dbdaddr(fsp, fsp->f32fsinfo_sector), 1807 PC_SAFESECSIZE); 1808 if (bp->b_flags & (B_ERROR | B_STALE)) { 1809 brelse(bp); 1810 return (EIO); 1811 } 1812 bcopy((void *)(bp->b_un.b_addr + FAT32_BOOT_FSINFO_OFF), 1813 &fsinfo_disk, sizeof (struct fat32_boot_fsinfo)); 1814 /* translate fields */ 1815 fsinfo_disk.fs_free_clusters = 1816 htoli(fsp->fsinfo_native.fs_free_clusters); 1817 fsinfo_disk.fs_next_cluster = (uint32_t)FSINFO_UNKNOWN; 1818 bcopy(&fsinfo_disk, 1819 (void *)(bp->b_un.b_addr + FAT32_BOOT_FSINFO_OFF), 1820 sizeof (struct fat32_boot_fsinfo)); 1821 bwrite2(bp); 1822 error = geterror(bp); 1823 brelse(bp); 1824 if (error) { 1825 pc_mark_irrecov(fsp); 1826 return (EIO); 1827 } 1828 } 1829 return (0); 1830 } 1831 1832 void 1833 pc_invalfat(struct pcfs *fsp) 1834 { 1835 struct pcfs *xfsp; 1836 int mount_cnt = 0; 1837 1838 PC_DPRINTF0(7, "pc_invalfat\n"); 1839 if (fsp->pcfs_fatp == (uchar_t *)0) 1840 panic("pc_invalfat"); 1841 /* 1842 * Release FAT 1843 */ 1844 kmem_free(fsp->pcfs_fatp, fsp->pcfs_fatsize); 1845 fsp->pcfs_fatp = NULL; 1846 kmem_free(fsp->pcfs_fat_changemap, fsp->pcfs_fat_changemapsize); 1847 fsp->pcfs_fat_changemap = NULL; 1848 /* 1849 * Invalidate all the blocks associated with the device. 1850 * Not needed if stateless. 1851 */ 1852 for (xfsp = pc_mounttab; xfsp; xfsp = xfsp->pcfs_nxt) 1853 if (xfsp != fsp && xfsp->pcfs_xdev == fsp->pcfs_xdev) 1854 mount_cnt++; 1855 1856 if (!mount_cnt) 1857 binval(fsp->pcfs_xdev); 1858 /* 1859 * close mounted device 1860 */ 1861 (void) VOP_CLOSE(fsp->pcfs_devvp, 1862 (PCFSTOVFS(fsp)->vfs_flag & VFS_RDONLY) ? FREAD : FREAD|FWRITE, 1863 1, (offset_t)0, CRED()); 1864 } 1865 1866 void 1867 pc_badfs(struct pcfs *fsp) 1868 { 1869 cmn_err(CE_WARN, "corrupted PC file system on dev %x.%x\n", 1870 getmajor(fsp->pcfs_devvp->v_rdev), 1871 getminor(fsp->pcfs_devvp->v_rdev)); 1872 } 1873 1874 /* 1875 * The problem with supporting NFS on the PCFS filesystem is that there 1876 * is no good place to keep the generation number. The only possible 1877 * place is inside a directory entry. There are a few words that we 1878 * don't use - they store NT & OS/2 attributes, and the creation/last access 1879 * time of the file - but it seems wrong to use them. In addition, directory 1880 * entries come and go. If a directory is removed completely, its directory 1881 * blocks are freed and the generation numbers are lost. Whereas in ufs, 1882 * inode blocks are dedicated for inodes, so the generation numbers are 1883 * permanently kept on the disk. 1884 */ 1885 static int 1886 pcfs_vget(struct vfs *vfsp, struct vnode **vpp, struct fid *fidp) 1887 { 1888 struct pcnode *pcp; 1889 struct pc_fid *pcfid; 1890 struct pcfs *fsp; 1891 struct pcdir *ep; 1892 daddr_t eblkno; 1893 int eoffset; 1894 struct buf *bp; 1895 int error; 1896 pc_cluster32_t cn; 1897 1898 pcfid = (struct pc_fid *)fidp; 1899 fsp = VFSTOPCFS(vfsp); 1900 1901 error = pc_lockfs(fsp, 0, 0); 1902 if (error) { 1903 *vpp = NULL; 1904 return (error); 1905 } 1906 1907 if (pcfid->pcfid_block == 0) { 1908 pcp = pc_getnode(fsp, (daddr_t)0, 0, (struct pcdir *)0); 1909 pcp->pc_flags |= PC_EXTERNAL; 1910 *vpp = PCTOV(pcp); 1911 pc_unlockfs(fsp); 1912 return (0); 1913 } 1914 eblkno = pcfid->pcfid_block; 1915 eoffset = pcfid->pcfid_offset; 1916 if ((pc_dbtocl(fsp, 1917 eblkno - fsp->pcfs_dosstart) >= fsp->pcfs_ncluster) || 1918 (eoffset > fsp->pcfs_clsize)) { 1919 pc_unlockfs(fsp); 1920 *vpp = NULL; 1921 return (EINVAL); 1922 } 1923 1924 if (eblkno >= fsp->pcfs_datastart || (eblkno-fsp->pcfs_rdirstart) 1925 < (fsp->pcfs_rdirsec & ~(fsp->pcfs_spcl - 1))) { 1926 bp = bread(fsp->pcfs_xdev, eblkno, fsp->pcfs_clsize); 1927 } else { 1928 bp = bread(fsp->pcfs_xdev, eblkno, 1929 (int)(fsp->pcfs_datastart - eblkno) * fsp->pcfs_secsize); 1930 } 1931 if (bp->b_flags & (B_ERROR | B_STALE)) { 1932 error = geterror(bp); 1933 brelse(bp); 1934 if (error) 1935 pc_mark_irrecov(fsp); 1936 *vpp = NULL; 1937 pc_unlockfs(fsp); 1938 return (error); 1939 } 1940 ep = (struct pcdir *)(bp->b_un.b_addr + eoffset); 1941 /* 1942 * Ok, if this is a valid file handle that we gave out, 1943 * then simply ensuring that the creation time matches, 1944 * the entry has not been deleted, and it has a valid first 1945 * character should be enough. 1946 * 1947 * Unfortunately, verifying that the <blkno, offset> _still_ 1948 * refers to a directory entry is not easy, since we'd have 1949 * to search _all_ directories starting from root to find it. 1950 * That's a high price to pay just in case somebody is forging 1951 * file handles. So instead we verify that as much of the 1952 * entry is valid as we can: 1953 * 1954 * 1. The starting cluster is 0 (unallocated) or valid 1955 * 2. It is not an LFN entry 1956 * 3. It is not hidden (unless mounted as such) 1957 * 4. It is not the label 1958 */ 1959 cn = pc_getstartcluster(fsp, ep); 1960 /* 1961 * if the starting cluster is valid, but not valid according 1962 * to pc_validcl(), force it to be to simplify the following if. 1963 */ 1964 if (cn == 0) 1965 cn = PCF_FIRSTCLUSTER; 1966 if (IS_FAT32(fsp)) { 1967 if (cn >= PCF_LASTCLUSTER32) 1968 cn = PCF_FIRSTCLUSTER; 1969 } else { 1970 if (cn >= PCF_LASTCLUSTER) 1971 cn = PCF_FIRSTCLUSTER; 1972 } 1973 if ((!pc_validcl(fsp, cn)) || 1974 (PCDL_IS_LFN(ep)) || 1975 (PCA_IS_HIDDEN(fsp, ep->pcd_attr)) || 1976 ((ep->pcd_attr & PCA_LABEL) == PCA_LABEL)) { 1977 bp->b_flags |= B_STALE | B_AGE; 1978 brelse(bp); 1979 pc_unlockfs(fsp); 1980 return (EINVAL); 1981 } 1982 if ((ep->pcd_crtime.pct_time == pcfid->pcfid_ctime) && 1983 (ep->pcd_filename[0] != PCD_ERASED) && 1984 (pc_validchar(ep->pcd_filename[0]) || 1985 (ep->pcd_filename[0] == '.' && ep->pcd_filename[1] == '.'))) { 1986 pcp = pc_getnode(fsp, eblkno, eoffset, ep); 1987 pcp->pc_flags |= PC_EXTERNAL; 1988 *vpp = PCTOV(pcp); 1989 } else { 1990 *vpp = NULL; 1991 } 1992 bp->b_flags |= B_STALE | B_AGE; 1993 brelse(bp); 1994 pc_unlockfs(fsp); 1995 return (0); 1996 } 1997 1998 /* 1999 * if device is a PCMCIA pseudo floppy, return 1 2000 * otherwise, return 0 2001 */ 2002 static int 2003 pcfs_pseudo_floppy(dev_t rdev) 2004 { 2005 int error, err; 2006 struct dk_cinfo info; 2007 ldi_handle_t lh; 2008 ldi_ident_t li; 2009 2010 err = ldi_ident_from_mod(&modlinkage, &li); 2011 if (err) { 2012 PC_DPRINTF1(1, 2013 "pcfs_pseudo_floppy: ldi_ident_from_mod err=%d\n", err); 2014 return (0); 2015 } 2016 2017 err = ldi_open_by_dev(&rdev, OTYP_CHR, FREAD, CRED(), &lh, li); 2018 ldi_ident_release(li); 2019 if (err) { 2020 PC_DPRINTF1(1, 2021 "pcfs_pseudo_floppy: ldi_open err=%d\n", err); 2022 return (0); 2023 } 2024 2025 /* return value stored in err is purposfully ignored */ 2026 error = ldi_ioctl(lh, DKIOCINFO, (intptr_t)&info, FKIOCTL, 2027 CRED(), &err); 2028 2029 err = ldi_close(lh, FREAD, CRED()); 2030 if (err != 0) { 2031 PC_DPRINTF1(1, 2032 "pcfs_pseudo_floppy: ldi_close err=%d\n", err); 2033 return (0); 2034 } 2035 2036 if ((error == 0) && (info.dki_ctype == DKC_PCMCIA_MEM) && 2037 (info.dki_flags & DKI_PCMCIA_PFD)) 2038 return (1); 2039 else 2040 return (0); 2041 } 2042 2043 /* 2044 * Unfortunately, FAT32 fat's can be pretty big (On a 1 gig jaz drive, about 2045 * a meg), so we can't bread() it all in at once. This routine reads a 2046 * fat a chunk at a time. 2047 */ 2048 static int 2049 pc_readfat(struct pcfs *fsp, uchar_t *fatp, daddr_t start, size_t fatsize) 2050 { 2051 struct buf *bp; 2052 size_t off; 2053 size_t readsize; 2054 2055 readsize = fsp->pcfs_clsize; 2056 for (off = 0; off < fatsize; off += readsize, fatp += readsize) { 2057 if (readsize > (fatsize - off)) 2058 readsize = fatsize - off; 2059 bp = bread(fsp->pcfs_xdev, 2060 pc_dbdaddr(fsp, start + 2061 pc_cltodb(fsp, pc_lblkno(fsp, off))), 2062 readsize); 2063 if (bp->b_flags & (B_ERROR | B_STALE)) { 2064 brelse(bp); 2065 return (EIO); 2066 } 2067 bp->b_flags |= B_STALE | B_AGE; 2068 bcopy(bp->b_un.b_addr, fatp, readsize); 2069 brelse(bp); 2070 } 2071 return (0); 2072 } 2073 2074 /* 2075 * We write the FAT out a _lot_, in order to make sure that it 2076 * is up-to-date. But on a FAT32 system (large drive, small clusters) 2077 * the FAT might be a couple of megabytes, and writing it all out just 2078 * because we created or deleted a small file is painful (especially 2079 * since we do it for each alternate FAT too). So instead, for FAT16 and 2080 * FAT32 we only write out the bit that has changed. We don't clear 2081 * the 'updated' fields here because the caller might be writing out 2082 * several FATs, so the caller must use pc_clear_fatchanges() after 2083 * all FATs have been updated. 2084 */ 2085 static int 2086 pc_writefat(struct pcfs *fsp, daddr_t start) 2087 { 2088 struct buf *bp; 2089 size_t off; 2090 size_t writesize; 2091 int error; 2092 uchar_t *fatp = fsp->pcfs_fatp; 2093 size_t fatsize = fsp->pcfs_fatsize; 2094 2095 writesize = fsp->pcfs_clsize; 2096 for (off = 0; off < fatsize; off += writesize, fatp += writesize) { 2097 if (writesize > (fatsize - off)) 2098 writesize = fatsize - off; 2099 if (!pc_fat_is_changed(fsp, pc_lblkno(fsp, off))) { 2100 continue; 2101 } 2102 bp = ngeteblk(writesize); 2103 bp->b_edev = fsp->pcfs_xdev; 2104 bp->b_dev = cmpdev(bp->b_edev); 2105 bp->b_blkno = pc_dbdaddr(fsp, start + 2106 pc_cltodb(fsp, pc_lblkno(fsp, off))); 2107 bcopy(fatp, bp->b_un.b_addr, writesize); 2108 bwrite2(bp); 2109 error = geterror(bp); 2110 brelse(bp); 2111 if (error) { 2112 return (error); 2113 } 2114 } 2115 return (0); 2116 } 2117 2118 /* 2119 * Mark the FAT cluster that 'cn' is stored in as modified. 2120 */ 2121 void 2122 pc_mark_fat_updated(struct pcfs *fsp, pc_cluster32_t cn) 2123 { 2124 pc_cluster32_t bn; 2125 size_t size; 2126 2127 /* which fat block is the cluster number stored in? */ 2128 if (IS_FAT32(fsp)) { 2129 size = sizeof (pc_cluster32_t); 2130 bn = pc_lblkno(fsp, cn * size); 2131 fsp->pcfs_fat_changemap[bn] = 1; 2132 } else if (IS_FAT16(fsp)) { 2133 size = sizeof (pc_cluster16_t); 2134 bn = pc_lblkno(fsp, cn * size); 2135 fsp->pcfs_fat_changemap[bn] = 1; 2136 } else { 2137 offset_t off; 2138 pc_cluster32_t nbn; 2139 2140 ASSERT(IS_FAT12(fsp)); 2141 off = cn + (cn >> 1); 2142 bn = pc_lblkno(fsp, off); 2143 fsp->pcfs_fat_changemap[bn] = 1; 2144 /* does this field wrap into the next fat cluster? */ 2145 nbn = pc_lblkno(fsp, off + 1); 2146 if (nbn != bn) { 2147 fsp->pcfs_fat_changemap[nbn] = 1; 2148 } 2149 } 2150 } 2151 2152 /* 2153 * return whether the FAT cluster 'bn' is updated and needs to 2154 * be written out. 2155 */ 2156 int 2157 pc_fat_is_changed(struct pcfs *fsp, pc_cluster32_t bn) 2158 { 2159 return (fsp->pcfs_fat_changemap[bn] == 1); 2160 } 2161