#ifdef PLAN9PORT /* SORRY! */ # include # include # ifdef __linux__ /* REALLY SORRY! */ # define CANBLOCKSIZE 1 # include # elif defined(__FreeBSD__) # define CANBLOCKSIZE 1 # include # include # include # endif #endif #include "stdinc.h" #include #include "dat.h" #include "fns.h" u32int maxblocksize; int readonly; static int strtoullsuf(char *p, char **pp, int rad, u64int *u) { u64int v; if(!isdigit((uchar)*p)) return -1; v = strtoull(p, &p, rad); switch(*p){ case 'k': case 'K': v *= 1024; p++; break; case 'm': case 'M': v *= 1024*1024; p++; break; case 'g': case 'G': v *= 1024*1024*1024; p++; break; case 't': case 'T': v *= 1024*1024; v *= 1024*1024; p++; break; } *pp = p; *u = v; return 0; } static int parsepart(char *name, char **file, u64int *lo, u64int *hi) { char *p; *file = estrdup(name); if((p = strrchr(*file, ':')) == nil){ *lo = 0; *hi = 0; return 0; } *p++ = 0; if(*p == '-') *lo = 0; else{ if(strtoullsuf(p, &p, 0, lo) < 0){ free(*file); return -1; } } if(*p == '-') p++; if(*p == 0){ *hi = 0; return 0; } if(strtoullsuf(p, &p, 0, hi) < 0 || *p != 0){ free(*file); return -1; } return 0; } Part* initpart(char *name, int mode) { Part *part; Dir *dir; char *file; u64int lo, hi; if(parsepart(name, &file, &lo, &hi) < 0) return nil; trace(TraceDisk, "initpart %s file %s lo 0x%llx hi 0x%llx", name, file, lo, hi); part = MKZ(Part); part->name = estrdup(name); part->filename = estrdup(file); if(readonly){ mode &= (OREAD|OWRITE|ORDWR); mode |= OREAD; } #ifdef __linux__ /* sorry, but linus made O_DIRECT unusable! */ mode &= ~ODIRECT; #endif part->fd = open(file, mode); if(part->fd < 0){ if((mode&(OREAD|OWRITE|ORDWR)) == ORDWR) part->fd = open(file, (mode&~ORDWR)|OREAD); if(part->fd < 0){ freepart(part); fprint(2, "can't open partition='%s': %r\n", file); seterr(EOk, "can't open partition='%s': %r", file); fprint(2, "%r\n"); free(file); return nil; } fprint(2, "warning: %s opened for reading only\n", name); } #ifdef __linux__ /* sorry again! still linus's fault! */ posix_fadvise(part->fd, 0, 0, POSIX_FADV_RANDOM); /* disable readahead */ #endif part->offset = lo; dir = dirfstat(part->fd); if(dir == nil){ freepart(part); seterr(EOk, "can't stat partition='%s': %r", file); free(file); return nil; } if(dir->length == 0){ free(dir); freepart(part); seterr(EOk, "can't determine size of partition %s", file); free(file); return nil; } if(dir->length < hi || dir->length < lo){ freepart(part); seterr(EOk, "partition '%s': bounds out of range (max %lld)", name, dir->length); free(dir); free(file); return nil; } if(hi == 0) hi = dir->length; part->size = hi - part->offset; #ifdef CANBLOCKSIZE { struct statfs sfs; if(fstatfs(part->fd, &sfs) >= 0) part->fsblocksize = sfs.f_bsize; } #endif free(dir); return part; } int flushpart(Part *part) { USED(part); #ifdef __linux__ /* grrr! */ if(fsync(part->fd) < 0){ logerr(EAdmin, "flushpart %s: %r", part->name); return -1; } posix_fadvise(part->fd, 0, 0, POSIX_FADV_DONTNEED); #endif return 0; } void freepart(Part *part) { if(part == nil) return; if(part->fd >= 0) close(part->fd); free(part->name); free(part); } void partblocksize(Part *part, u32int blocksize) { if(part->blocksize) sysfatal("resetting partition=%s's block size", part->name); part->blocksize = blocksize; if(blocksize > maxblocksize) maxblocksize = blocksize; } /* * Read/write some amount of data between a block device or file and a memory buffer. * * Most Unix systems require that when accessing a block device directly, * the buffer, offset, and count are all multiples of the device block size, * making this a lot more complicated than it otherwise would be. * * Most of our callers will make things easy on us, but for some callers it's best * if we just do the work here, with only one place to get it right (hopefully). * * If everything is aligned properly, prwb will try to do big transfers in the main * body of the loop: up to MaxIo bytes at a time. If everything isn't aligned properly, * we work one block at a time. */ #undef min #define min(a, b) ((a) < (b) ? (a) : (b)) int prwb(char *name, int fd, int isread, u64int offset, void *vbuf, u32int count, u32int blocksize) { char *op; u8int *buf, *freetmp, *dst; u32int icount, opsize; int r; #ifndef PLAN9PORT USED(blocksize); icount = count; buf = vbuf; op = isread ? "read" : "write"; dst = buf; freetmp = nil; while(count > 0){ opsize = min(count, 131072 /* blocksize */); if(isread) r = pread(fd, dst, opsize, offset); else r = pwrite(fd, dst, opsize, offset); if(r <= 0) goto Error; offset += r; count -= r; dst += r; if(r != opsize) goto Error; } return icount; #else u32int c, delta; u8int *tmp; icount = count; buf = vbuf; tmp = nil; freetmp = nil; opsize = blocksize; if(count == 0){ logerr(EStrange, "pwrb %s called to %s 0 bytes", name, isread ? "read" : "write"); return 0; } assert(blocksize > 0); /* allocate blocksize-aligned temp buffer if needed */ if((ulong)offset%blocksize || (ulong)buf%blocksize || count%blocksize){ if((freetmp = malloc(blocksize*2)) == nil) return -1; tmp = freetmp; tmp += blocksize - (ulong)tmp%blocksize; } /* handle beginning fringe */ if((delta = (ulong)offset%blocksize) != 0){ assert(tmp != nil); if((r=pread(fd, tmp, blocksize, offset-delta)) != blocksize){ dst = tmp; offset = offset-delta; op = "read"; goto Error; } c = min(count, blocksize-delta); assert(c > 0 && c < blocksize); if(isread) memmove(buf, tmp+delta, c); else{ memmove(tmp+delta, buf, c); if((r=pwrite(fd, tmp, blocksize, offset-delta)) != blocksize){ dst = tmp; offset = offset-delta; op = "read"; goto Error; } } assert(c > 0); offset += c; buf += c; count -= c; } /* handle full blocks */ while(count >= blocksize){ assert((ulong)offset%blocksize == 0); if((ulong)buf%blocksize){ assert(tmp != nil); dst = tmp; opsize = blocksize; }else{ dst = buf; opsize = count - count%blocksize; if(opsize > MaxIo) opsize = MaxIo; } if(isread){ if((r=pread(fd, dst, opsize, offset))<=0 || r%blocksize){ op = "read"; goto Error; } if(dst == tmp){ assert(r == blocksize); memmove(buf, tmp, blocksize); } }else{ if(dst == tmp){ assert(opsize == blocksize); memmove(dst, buf, blocksize); } if((r=pwrite(fd, dst, opsize, offset))<=0 || r%blocksize){ op = "write"; goto Error; } if(dst == tmp) assert(r == blocksize); } assert(r > 0); offset += r; buf += r; count -= r; } /* handle ending fringe */ if(count > 0){ assert((ulong)offset%blocksize == 0); assert(tmp != nil); /* * Complicated condition: if we're reading it's okay to get less than * a block as long as it's enough to satisfy the read - maybe this is * a normal file. (We never write to normal files, or else things would * be even more complicated.) */ r = pread(fd, tmp, blocksize, offset); if((isread && r < count) || (!isread && r != blocksize)){ print("FAILED isread=%d r=%d count=%d blocksize=%d\n", isread, r, count, blocksize); dst = tmp; op = "read"; goto Error; } if(isread) memmove(buf, tmp, count); else{ memmove(tmp, buf, count); if(pwrite(fd, tmp, opsize, offset) != blocksize){ dst = tmp; op = "write"; goto Error; } } } if(freetmp) free(freetmp); return icount; #endif Error: seterr(EAdmin, "%s %s offset 0x%llux count %ud buf %p returned %d: %r", op, name, offset, opsize, dst, r); if(freetmp) free(freetmp); return -1; } #ifndef PLAN9PORT static int sdreset(Part*); static int reopen(Part*); static int threadspawnl(int[3], char*, char*, ...); #endif int rwpart(Part *part, int isread, u64int offset, u8int *buf, u32int count) { int n, try; u32int blocksize; trace(TraceDisk, "%s %s %ud at 0x%llx", isread ? "read" : "write", part->name, count, offset); if(offset >= part->size || offset+count > part->size){ seterr(EStrange, "out of bounds %s offset 0x%llux count %ud to partition %s size 0x%llux", isread ? "read" : "write", offset, count, part->name, part->size); return -1; } blocksize = part->fsblocksize; if(blocksize == 0) blocksize = part->blocksize; if(blocksize == 0) blocksize = 4096; for(try=0;; try++){ n = prwb(part->filename, part->fd, isread, part->offset+offset, buf, count, blocksize); if(n >= 0 || try > 10) break; #ifndef PLAN9PORT { char err[ERRMAX]; /* * This happens with the sdmv disks frustratingly often. * Try to fix things up and continue. */ rerrstr(err, sizeof err); if(strstr(err, "i/o timeout") || strstr(err, "i/o error") || strstr(err, "partition has changed")){ reopen(part); continue; } } #endif break; } #ifdef __linux__ /* sigh */ posix_fadvise(part->fd, part->offset+offset, n, POSIX_FADV_DONTNEED); #endif return n; } int readpart(Part *part, u64int offset, u8int *buf, u32int count) { return rwpart(part, 1, offset, buf, count); } int writepart(Part *part, u64int offset, u8int *buf, u32int count) { return rwpart(part, 0, offset, buf, count); } ZBlock* readfile(char *name) { Part *p; ZBlock *b; p = initpart(name, OREAD); if(p == nil) return nil; b = alloczblock(p->size, 0, p->blocksize); if(b == nil){ seterr(EOk, "can't alloc %s: %r", name); freepart(p); return nil; } if(readpart(p, 0, b->data, p->size) < 0){ seterr(EOk, "can't read %s: %r", name); freepart(p); freezblock(b); return nil; } freepart(p); return b; } #ifndef PLAN9PORT static int sdreset(Part *part) { char *name, *p; int i, fd, xfd[3], rv; static QLock resetlk; Dir *d, *dd; fprint(2, "sdreset %s\n", part->name); name = emalloc(strlen(part->filename)+20); strcpy(name, part->filename); p = strrchr(name, '/'); if(p) p++; else p = name; strcpy(p, "ctl"); d = dirstat(name); if(d == nil){ free(name); return -1; } /* * We don't need multiple people resetting the disk. */ qlock(&resetlk); if((fd = open(name, OWRITE)) < 0) goto error; dd = dirfstat(fd); if(d && dd && d->qid.vers != dd->qid.vers){ fprint(2, "sdreset %s: got scooped\n", part->name); /* Someone else got here first. */ if(access(part->filename, AEXIST) >= 0) goto ok; goto error; } /* * Write "reset" to the ctl file to cause the chipset * to reinitialize itself (specific to sdmv driver). * Ignore error in case using other disk. */ fprint(2, "sdreset %s: reset ctl\n", part->name); write(fd, "reset", 5); if(access(part->filename, AEXIST) >= 0) goto ok; /* * Re-run fdisk and prep. Don't use threadwaitchan * to avoid coordinating for it. Reopen ctl because * we reset the disk. */ strcpy(p, "ctl"); close(fd); if((fd = open(name, OWRITE)) < 0) goto error; strcpy(p, "data"); xfd[0] = open("/dev/null", OREAD); xfd[1] = dup(fd, -1); xfd[2] = dup(2, -1); fprint(2, "sdreset %s: run fdisk %s\n", part->name, name); if(threadspawnl(xfd, "/bin/disk/fdisk", "disk/fdisk", "-p", name, nil) < 0){ close(xfd[0]); close(xfd[1]); close(xfd[2]); goto error; } strcpy(p, "plan9"); for(i=0; i<=20; i++){ sleep(i*100); if(access(part->filename, AEXIST) >= 0) goto ok; if(access(name, AEXIST) >= 0) goto prep; } goto error; prep: strcpy(p, "ctl"); close(fd); if((fd = open(name, OWRITE)) < 0) goto error; strcpy(p, "plan9"); xfd[0] = open("/dev/null", OREAD); xfd[1] = dup(fd, -1); xfd[2] = dup(2, -1); fprint(2, "sdreset %s: run prep\n", part->name); if(threadspawnl(xfd, "/bin/disk/prep", "disk/prep", "-p", name, nil) < 0){ close(xfd[0]); close(xfd[1]); close(xfd[2]); goto error; } for(i=0; i<=20; i++){ sleep(i*100); if(access(part->filename, AEXIST) >= 0) goto ok; } error: fprint(2, "sdreset %s: error: %r\n", part->name); rv = -1; if(fd >= 0) close(fd); goto out; ok: fprint(2, "sdreset %s: all okay\n", part->name); rv = 0; goto out; out: free(name); qunlock(&resetlk); return rv; } static int reopen(Part *part) { int fd; fprint(2, "reopen %s\n", part->filename); if((fd = open(part->filename, ORDWR)) < 0){ if(access(part->filename, AEXIST) < 0){ sdreset(part); fd = open(part->filename, ORDWR); } if(fd < 0){ fprint(2, "reopen %s: %r\n", part->filename); return -1; } } if(fd != part->fd){ dup(fd, part->fd); close(fd); } return 0; } typedef struct Spawn Spawn; struct Spawn { Channel *c; int fd[3]; char *file; char **argv; }; static void spawnproc(void *v) { int i, *fd; Spawn *s; rfork(RFFDG); s = v; fd = s->fd; for(i=0; i<3; i++) dup(fd[i], i); if(fd[0] > 2) close(fd[0]); if(fd[1] > 2 && fd[1] != fd[0]) close(fd[1]); if(fd[2] > 2 && fd[2] != fd[1] && fd[2] != fd[0]) close(fd[2]); procexec(s->c, s->file, s->argv); } static int threadspawnl(int fd[3], char *file, char *argv0, ...) { int pid; Spawn s; s.c = chancreate(sizeof(void*), 0); memmove(s.fd, fd, sizeof(s.fd)); s.file = file; s.argv = &argv0; vtproc(spawnproc, &s); pid = recvul(s.c); if(pid < 0) return -1; close(fd[0]); if(fd[1] != fd[0]) close(fd[1]); if(fd[2] != fd[1] && fd[2] != fd[0]) close(fd[2]); return pid; } #endif