/* $OpenBSD: disk.c,v 1.47 2015/03/30 17:11:49 krw Exp $ */ /* * Copyright (c) 1997 Tobias Weingartner * * Permission to use, copy, modify, and distribute this software for any * purpose with or without fee is hereby granted, provided that the above * copyright notice and this permission notice appear in all copies. * * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include "disk.h" #include "misc.h" struct disk disk; struct disklabel dl; int DISK_open(char *disk, int mode) { struct stat st; int fd; fd = opendev(disk, mode, OPENDEV_PART, NULL); if (fd == -1) err(1, "%s", disk); if (fstat(fd, &st) == -1) err(1, "%s", disk); if (!S_ISCHR(st.st_mode) && !S_ISREG(st.st_mode)) errx(1, "%s is not a character device or a regular file", disk); return (fd); } void DISK_getlabelgeometry(void) { u_int64_t sz, spc; int fd; /* Get label geometry. */ if ((fd = DISK_open(disk.name, O_RDONLY)) != -1) { if (ioctl(fd, DIOCGPDINFO, &dl) == -1) { warn("DIOCGPDINFO"); } else { disk.cylinders = dl.d_ncylinders; disk.heads = dl.d_ntracks; disk.sectors = dl.d_nsectors; /* MBR handles only first UINT32_MAX sectors. */ spc = (u_int64_t)disk.heads * disk.sectors; sz = DL_GETDSIZE(&dl); if (sz > UINT32_MAX) { disk.cylinders = UINT32_MAX / spc; disk.size = disk.cylinders * spc; warnx("disk too large (%llu sectors)." " size truncated.", sz); } else disk.size = sz; unit_types[SECTORS].conversion = dl.d_secsize; } close(fd); } } /* * Print the disk geometry information. Take an optional modifier * to indicate the units that should be used for display. */ int DISK_printgeometry(char *units) { const int secsize = unit_types[SECTORS].conversion; double size; int i; i = unit_lookup(units); size = ((double)disk.size * secsize) / unit_types[i].conversion; printf("Disk: %s\t", disk.name); if (disk.size) { printf("geometry: %d/%d/%d [%.0f ", disk.cylinders, disk.heads, disk.sectors, size); if (i == SECTORS && secsize != sizeof(struct dos_mbr)) printf("%d-byte ", secsize); printf("%s]\n", unit_types[i].lname); } else printf("geometry: \n"); return (0); } /* * Read the sector at 'where' from the file descriptor 'fd' into newly * calloc'd memory. Return a pointer to the memory if it contains the * requested data, or NULL if it does not. * * The caller must free() the memory it gets. */ char * DISK_readsector(int fd, off_t where) { int secsize; char *secbuf; ssize_t len; off_t off; secsize = dl.d_secsize; where *= secsize; off = lseek(fd, where, SEEK_SET); if (off != where) return (NULL); secbuf = calloc(1, secsize); if (secbuf == NULL) return (NULL); len = read(fd, secbuf, secsize); if (len == -1 || len != secsize) { free(secbuf); return (NULL); } return (secbuf); } /* * Write the sector-sized 'secbuf' to the sector 'where' on the file * descriptor 'fd'. Return 0 if the write works. Return -1 and set * errno if the write fails. */ int DISK_writesector(int fd, char *secbuf, off_t where) { int secsize; ssize_t len; off_t off; len = -1; secsize = dl.d_secsize; where *= secsize; off = lseek(fd, where, SEEK_SET); if (off == where) len = write(fd, secbuf, secsize); if (len == -1 || len != secsize) { /* short read or write */ errno = EIO; return (-1); } return (0); }