commit fa5af687114e590d63af995951f6092f42ed0d58 from: rsc date: Thu Jun 15 05:52:24 2006 UTC better ext2 block read (sigh) commit - dbf91a04cc556bad9b03d8eb11f92468ee9da092 commit + fa5af687114e590d63af995951f6092f42ed0d58 blob - 87437f849749221813592e8de6b12c5548a8d6e2 blob + 25c21c82064478e3ea4b5d87ed77cdbfd6993376 --- src/libdiskfs/ext2.c +++ src/libdiskfs/ext2.c @@ -655,7 +655,7 @@ ext2readfile(Fsys *fsys, SunAuthUnix *au, Nfs3Handle * uchar *data; Block *b; Ext2 *fs; - int off, want, fragcount; + int skip1, tot, want, fragcount; Inode ino; Nfs3Status ok; @@ -674,34 +674,36 @@ ext2readfile(Fsys *fsys, SunAuthUnix *au, Nfs3Handle * } if(offset+count > ino.size) count = ino.size-offset; - if(offset/fs->blocksize != (offset+count-1)/fs->blocksize) - count = fs->blocksize - offset%fs->blocksize; data = malloc(count); if(data == nil) return Nfs3ErrNoMem; + memset(data, 0, count); - want = offset%fs->blocksize+count; - if(want%fs->blocksize) - want += fs->blocksize - want%fs->blocksize; + skip1 = offset%fs->blocksize; + offset -= skip1; + want = skip1+count; - b = ext2fileblock(fs, &ino, offset/fs->blocksize, want); - if(b == nil){ - /* BUG: distinguish sparse file from I/O error */ - memset(data, 0, count); - }else{ - off = offset%fs->blocksize; - fragcount = count; /* need signed variable */ - if(off+fragcount > b->len){ - fragcount = b->len - off; - if(fragcount < 0) - fragcount = 0; - } - if(fragcount > 0) - memmove(data, b->data+off, fragcount); - count = fragcount; + /* + * have to read multiple blocks if we get asked for a big read. + * Linux NFS client assumes that if you ask for 8k and only get 4k + * back, the remaining 4k is zeros. + */ + for(tot=0; totblocksize, fs->blocksize); + fragcount = fs->blocksize; + if(b == nil) + continue; + if(tot+fragcount > want) + fragcount = want - tot; + if(tot == 0) + memmove(data, b->data+skip1, fragcount-skip1); + else + memmove(data+tot-skip1, b->data, fragcount); blockput(b); } + count = tot - skip1; + *peof = (offset+count == ino.size); *pcount = count; *pdata = data;