fix bugs in indirect-file code

clean up test program
This commit is contained in:
kaashoek 2006-08-24 17:28:01 +00:00
parent ea2909b6b5
commit 1be7668537
2 changed files with 143 additions and 36 deletions

17
fs.c
View file

@ -241,8 +241,7 @@ bmap(struct inode *ip, uint bn)
if (x == 0) if (x == 0)
panic("bmap 2"); panic("bmap 2");
} else { } else {
cprintf("indirect block read\n"); inbp = bread(ip->dev, ip->addrs[INDIRECT]);
inbp = bread(ip->dev, INDIRECT);
a = (uint *) inbp->data; a = (uint *) inbp->data;
x = a[bn - NDIRECT]; x = a[bn - NDIRECT];
brelse(inbp); brelse(inbp);
@ -256,20 +255,22 @@ void
iunlink(struct inode *ip) iunlink(struct inode *ip)
{ {
int i, j; int i, j;
struct buf *inbp;
// free inode, its blocks, and remove dir entry // free inode, its blocks, and remove dir entry
for (i = 0; i < NADDRS; i++) { for (i = 0; i < NADDRS; i++) {
if (ip->addrs[i] != 0) { if (ip->addrs[i] != 0) {
if (i == INDIRECT) { if (i == INDIRECT) {
inbp = bread(ip->dev, ip->addrs[INDIRECT]);
for (j = 0; j < NINDIRECT; j++) { for (j = 0; j < NINDIRECT; j++) {
uint *a = (uint *) (ip->addrs[i]); uint *a = (uint *) inbp->data;
if (a[j] != 0) { if (a[j] != 0) {
bfree(ip->dev, a[j]); bfree(ip->dev, a[j]);
a[j] = 0; a[j] = 0;
} }
} }
brelse(inbp);
} }
else
bfree(ip->dev, ip->addrs[i]); bfree(ip->dev, ip->addrs[i]);
ip->addrs[i] = 0; ip->addrs[i] = 0;
} }
@ -354,7 +355,7 @@ readi(struct inode *ip, char *dst, uint off, uint n)
return target - n; return target - n;
} }
int static int
newblock(struct inode *ip, uint lbn) newblock(struct inode *ip, uint lbn)
{ {
struct buf *inbp; struct buf *inbp;
@ -368,20 +369,18 @@ newblock(struct inode *ip, uint lbn)
ip->addrs[lbn] = b; ip->addrs[lbn] = b;
} }
} else { } else {
cprintf("newblock: use indirect block\n");
if (ip->addrs[INDIRECT] == 0) { if (ip->addrs[INDIRECT] == 0) {
cprintf("newblock: allocate indirect block\n");
b = balloc(ip->dev); b = balloc(ip->dev);
if (b <= 0) return -1; if (b <= 0) return -1;
ip->addrs[INDIRECT] = b; ip->addrs[INDIRECT] = b;
} }
inbp = bread(ip->dev, bmap(ip, INDIRECT)); inbp = bread(ip->dev, ip->addrs[INDIRECT]);
inaddrs = (uint *) inbp->data; inaddrs = (uint *) inbp->data;
if (inaddrs[lbn - NDIRECT] == 0) { if (inaddrs[lbn - NDIRECT] == 0) {
b = balloc(ip->dev); b = balloc(ip->dev);
if (b <= 0) return -1; if (b <= 0) return -1;
inaddrs[lbn - NDIRECT] = b; inaddrs[lbn - NDIRECT] = b;
bwrite(inbp, INDIRECT); bwrite(inbp, ip->addrs[INDIRECT]);
} }
brelse(inbp); brelse(inbp);
} }

160
userfs.c
View file

@ -4,25 +4,18 @@
#include "fs.h" #include "fs.h"
#include "fcntl.h" #include "fcntl.h"
// file system tests // simple file system tests
char buf[2000]; char buf[2000];
char name[3]; char name[3];
char *echo_args[] = { "echo", "hello", "goodbye", 0 }; char *echo_args[] = { "echo", "hello", "goodbye", 0 };
char *cat_args[] = { "cat", "readme", 0 }; char *cat_args[] = { "cat", "readme", 0 };
int
main(void)
{
int fd;
int i;
int stdout = 1; int stdout = 1;
printf(stdout, "userfs is running\n"); void
opentest(void)
if (sbrk(4096) < 0) { {
printf(stdout, "sbrk failed\n"); int fd;
}
fd = open("echo", 0); fd = open("echo", 0);
if(fd >= 0){ if(fd >= 0){
@ -30,47 +23,123 @@ main(void)
close(fd); close(fd);
} else { } else {
printf(stdout, "open echo failed!\n"); printf(stdout, "open echo failed!\n");
exit();
} }
fd = open("doesnotexist", 0); fd = open("doesnotexist", 0);
if(fd >= 0){ if(fd >= 0){
printf(stdout, "open doesnotexist succeeded!\n"); printf(stdout, "open doesnotexist succeeded!\n");
close(fd); exit();
} else { } else {
printf(stdout, "open doesnotexist failed\n"); printf(stdout, "open doesnotexist failed\n");
} }
fd = open("doesnotexist", O_CREATE|O_RDWR); }
void
writetest(void)
{
int fd;
int i;
fd = open("small", O_CREATE|O_RDWR);
if(fd >= 0){ if(fd >= 0){
printf(stdout, "creat doesnotexist succeeded\n"); printf(stdout, "creat small succeeded\n");
} else { } else {
printf(stdout, "error: creat doesnotexist failed!\n"); printf(stdout, "error: creat small failed!\n");
exit();
} }
for (i = 0; i < 100; i++) { for (i = 0; i < 100; i++) {
if (write (fd, "aaaaaaaaaa", 10) != 10) { if (write (fd, "aaaaaaaaaa", 10) != 10) {
printf(stdout, "error: write aa %d new file failed\n", i); printf(stdout, "error: write aa %d new file failed\n", i);
exit();
} }
if (write (fd, "bbbbbbbbbb", 10) != 10) { if (write (fd, "bbbbbbbbbb", 10) != 10) {
printf(stdout, "error: write bb %d new file failed\n", i); printf(stdout, "error: write bb %d new file failed\n", i);
exit();
} }
} }
printf(stdout, "writes done\n"); printf(stdout, "writes done\n");
close(fd); close(fd);
fd = open("doesnotexist", O_RDONLY); fd = open("small", O_RDONLY);
if(fd >= 0){ if(fd >= 0){
printf(stdout, "open doesnotexist succeeded\n"); printf(stdout, "open small succeeded\n");
} else { } else {
printf(stdout, "error: open doesnotexist failed!\n"); printf(stdout, "error: open small failed!\n");
exit();
} }
i = read(fd, buf, 2000); i = read(fd, buf, 2000);
if (i == 2000) { if (i == 2000) {
printf(stdout, "read succeeded\n"); printf(stdout, "read succeeded\n");
} else { } else {
printf(stdout, "read failed\n"); printf(stdout, "read failed\n");
exit();
} }
close(fd); close(fd);
printf(stdout, "unlink doesnotexist\n"); if (unlink("small") < 0) {
printf(stdout, "unlink small failed\n");
exit();
}
}
unlink("doesnotexist"); void
writetest1(void)
{
int i, fd, n;
printf(stdout, "big files\n");
fd = open("big", O_CREATE|O_RDWR);
if(fd < 0){
printf(stdout, "error: creat big failed!\n");
exit();
}
for (i = 0; i < MAXFILE; i++) {
((int *) buf)[0] = i;
if (write (fd, buf, 512) != 512) {
printf(stdout, "error: write big file failed\n", i);
exit();
}
}
close(fd);
fd = open("big", O_RDONLY);
if(fd < 0){
printf(stdout, "error: open big failed!\n");
exit();
}
n = 0;
while (1) {
i = read(fd, buf, 512);
if (i == 0) {
if (n == MAXFILE - 1) {
printf(stdout, "read only %d blocks from big", n);
exit();
}
break;
} else if (i != 512) {
printf(stdout, "read failed %d\n", i);
exit();
}
if (((int *)buf)[0] != n) {
printf(stdout, "read content of block %d is %d\n", n, ((int *)buf)[0]);
exit();
}
n++;
}
close(fd);
if (unlink("big") < 0) {
printf(stdout, "unlink big failed\n");
exit();
}
}
void
createtest(void)
{
int i, fd;
printf(stdout, "many creates, followed by unlink\n"); printf(stdout, "many creates, followed by unlink\n");
@ -87,16 +156,55 @@ main(void)
name[1] = '0' + i; name[1] = '0' + i;
unlink(name); unlink(name);
} }
}
void dirtest(void)
{
printf(stdout, "mkdir\n"); printf(stdout, "mkdir\n");
if (mkdir("dir0") < 0) if (mkdir("dir0") < 0) {
printf(stdout, "mkdir failed\n"); printf(stdout, "mkdir failed\n");
exit();
}
// unlink("dir0"); if (chdir("dir0") < 0) {
printf(stdout, "chdir dir0 failed\n");
exit();
}
//exec("echo", echo_args); if (chdir("..") < 0) {
printf(stdout, "about to do exec\n"); printf(stdout, "chdir .. failed\n");
exec("cat", cat_args); exit ();
}
if (unlink("dir0") < 0) {
printf(stdout, "unlink dir0 failed\n");
exit();
}
}
void
exectest(void)
{
if (exec("echo", echo_args) < 0) {
printf(stdout, "exec echo failed\n");
exit();
}
if (exec("cat", cat_args) < 0) {
printf(stdout, "exec cat failed\n");
exit();
}
}
int
main(void)
{
printf(stdout, "userfs is running\n");
opentest();
writetest();
writetest1();
createtest();
exectest();
return 0; return 0;
} }