/* * Copyright (c) 2002 Juli Mallett. All rights reserved. * * This software was written by Juli Mallett for the * FreeBSD project. Redistribution and use in source and binary forms, with * or without modification, are permitted provided that the following * conditions are met: * * 1. Redistribution of source code must retain the above copyright notice, * this list of conditions and the following disclaimer. * 2. Redistribution in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. */ #include __FBSDID("$FreeBSD: head/lib/libufs/type.c 109506 2003-01-19 00:43:17Z jmallett $"); #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include /* Internally, track the 'name' value, it's ours. */ #define MINE_NAME 0x01 struct uufsd * ufs_disk_ctor(const char *name) { struct uufsd *new; new = NULL; ERROR(new, NULL); new = malloc(sizeof(*new)); if (new == NULL) { ERROR(new, "unable to allocate memory for disk"); return NULL; } if (ufs_disk_fillout(new, name) == -1) { ERROR(new, "could not fill out disk"); free(new); return NULL; } return new; } void ufs_disk_dtor(struct uufsd **diskp) { struct uufsd *disk; if (diskp != NULL) disk = *diskp; else return; ERROR(disk, NULL); ufs_disk_close(disk); free(disk); *diskp = NULL; } int ufs_disk_close(struct uufsd *disk) { ERROR(disk, NULL); close(disk->d_fd); if (disk->d_inoblock != NULL) { free(disk->d_inoblock); disk->d_inoblock = NULL; } if (disk->d_mine & MINE_NAME) { free((char *)(uintptr_t)disk->d_name); disk->d_name = NULL; } return 0; } int ufs_disk_fillout(struct uufsd *disk, const char *name) { struct stat st; struct fstab *fs; const char *oname; char dev[MAXPATHLEN]; int fd; ERROR(disk, NULL); oname = name; fs = getfsfile(name); if (fs != NULL) name = fs->fs_spec; again: if (stat(name, &st) < 0) { if (*name != '/') { if (*name == 'r') name++; snprintf(dev, sizeof(dev), "%s%s", _PATH_DEV, name); name = dev; goto again; } ERROR(disk, "could not find special device"); return -1; } fd = open(name, O_RDONLY); if (fd == -1) { ERROR(disk, "could not open special device"); return -1; } disk->d_bsize = 1; disk->d_fd = fd; disk->d_inoblock = NULL; disk->d_inomin = 0; disk->d_inomax = 0; disk->d_mine = 0; disk->d_ufs = 0; disk->d_error = NULL; if (oname != name) { name = strdup(name); if (name == NULL) { ERROR(disk, "could not allocate memory for disk name"); return -1; } disk->d_mine |= MINE_NAME; } disk->d_name = name; if (sbread(disk) == -1) { ERROR(disk, "could not read superblock to fill out disk"); return -1; } return 0; }