245 lines
4.6 KiB
C
245 lines
4.6 KiB
C
|
/*
|
||
|
* JOS file system format
|
||
|
*/
|
||
|
|
||
|
// We don't actually want to define off_t!
|
||
|
#define off_t xxx_off_t
|
||
|
#define bool xxx_bool
|
||
|
#include <assert.h>
|
||
|
#include <errno.h>
|
||
|
#include <fcntl.h>
|
||
|
#include <inttypes.h>
|
||
|
#include <stdarg.h>
|
||
|
#include <stdio.h>
|
||
|
#include <stdlib.h>
|
||
|
#include <string.h>
|
||
|
#include <unistd.h>
|
||
|
#include <sys/mman.h>
|
||
|
#include <sys/stat.h>
|
||
|
#include <sys/types.h>
|
||
|
#undef off_t
|
||
|
#undef bool
|
||
|
|
||
|
// Prevent inc/types.h, included from inc/fs.h,
|
||
|
// from attempting to redefine types defined in the host's inttypes.h.
|
||
|
#define JOS_INC_TYPES_H
|
||
|
// Typedef the types that inc/mmu.h needs.
|
||
|
typedef uint32_t physaddr_t;
|
||
|
typedef uint32_t off_t;
|
||
|
typedef int bool;
|
||
|
|
||
|
#include <inc/mmu.h>
|
||
|
#include <inc/fs.h>
|
||
|
|
||
|
#define ROUNDUP(n, v) ((n) - 1 + (v) - ((n) - 1) % (v))
|
||
|
#define MAX_DIR_ENTS 128
|
||
|
|
||
|
struct Dir
|
||
|
{
|
||
|
struct File *f;
|
||
|
struct File *ents;
|
||
|
int n;
|
||
|
};
|
||
|
|
||
|
uint32_t nblocks;
|
||
|
char *diskmap, *diskpos;
|
||
|
struct Super *super;
|
||
|
uint32_t *bitmap;
|
||
|
|
||
|
void
|
||
|
panic(const char *fmt, ...)
|
||
|
{
|
||
|
va_list ap;
|
||
|
|
||
|
va_start(ap, fmt);
|
||
|
vfprintf(stderr, fmt, ap);
|
||
|
va_end(ap);
|
||
|
fputc('\n', stderr);
|
||
|
abort();
|
||
|
}
|
||
|
|
||
|
void
|
||
|
readn(int f, void *out, size_t n)
|
||
|
{
|
||
|
size_t p = 0;
|
||
|
while (p < n) {
|
||
|
ssize_t m = read(f, out + p, n - p);
|
||
|
if (m < 0)
|
||
|
panic("read: %s", strerror(errno));
|
||
|
if (m == 0)
|
||
|
panic("read: Unexpected EOF");
|
||
|
p += m;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
uint32_t
|
||
|
blockof(void *pos)
|
||
|
{
|
||
|
return ((char*)pos - diskmap) / BLKSIZE;
|
||
|
}
|
||
|
|
||
|
void *
|
||
|
alloc(uint32_t bytes)
|
||
|
{
|
||
|
void *start = diskpos;
|
||
|
diskpos += ROUNDUP(bytes, BLKSIZE);
|
||
|
if (blockof(diskpos) >= nblocks)
|
||
|
panic("out of disk blocks");
|
||
|
return start;
|
||
|
}
|
||
|
|
||
|
void
|
||
|
opendisk(const char *name)
|
||
|
{
|
||
|
int r, diskfd, nbitblocks;
|
||
|
|
||
|
if ((diskfd = open(name, O_RDWR | O_CREAT, 0666)) < 0)
|
||
|
panic("open %s: %s", name, strerror(errno));
|
||
|
|
||
|
if ((r = ftruncate(diskfd, 0)) < 0
|
||
|
|| (r = ftruncate(diskfd, nblocks * BLKSIZE)) < 0)
|
||
|
panic("truncate %s: %s", name, strerror(errno));
|
||
|
|
||
|
if ((diskmap = mmap(NULL, nblocks * BLKSIZE, PROT_READ|PROT_WRITE,
|
||
|
MAP_SHARED, diskfd, 0)) == MAP_FAILED)
|
||
|
panic("mmap %s: %s", name, strerror(errno));
|
||
|
|
||
|
close(diskfd);
|
||
|
|
||
|
diskpos = diskmap;
|
||
|
alloc(BLKSIZE);
|
||
|
super = alloc(BLKSIZE);
|
||
|
super->s_magic = FS_MAGIC;
|
||
|
super->s_nblocks = nblocks;
|
||
|
super->s_root.f_type = FTYPE_DIR;
|
||
|
strcpy(super->s_root.f_name, "/");
|
||
|
|
||
|
nbitblocks = (nblocks + BLKBITSIZE - 1) / BLKBITSIZE;
|
||
|
bitmap = alloc(nbitblocks * BLKSIZE);
|
||
|
memset(bitmap, 0xFF, nbitblocks * BLKSIZE);
|
||
|
}
|
||
|
|
||
|
void
|
||
|
finishdisk(void)
|
||
|
{
|
||
|
int r, i;
|
||
|
|
||
|
for (i = 0; i < blockof(diskpos); ++i)
|
||
|
bitmap[i/32] &= ~(1<<(i%32));
|
||
|
|
||
|
if ((r = msync(diskmap, nblocks * BLKSIZE, MS_SYNC)) < 0)
|
||
|
panic("msync: %s", strerror(errno));
|
||
|
}
|
||
|
|
||
|
void
|
||
|
finishfile(struct File *f, uint32_t start, uint32_t len)
|
||
|
{
|
||
|
int i;
|
||
|
f->f_size = len;
|
||
|
len = ROUNDUP(len, BLKSIZE);
|
||
|
for (i = 0; i < len / BLKSIZE && i < NDIRECT; ++i)
|
||
|
f->f_direct[i] = start + i;
|
||
|
if (i == NDIRECT) {
|
||
|
uint32_t *ind = alloc(BLKSIZE);
|
||
|
f->f_indirect = blockof(ind);
|
||
|
for (; i < len / BLKSIZE; ++i)
|
||
|
ind[i - NDIRECT] = start + i;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void
|
||
|
startdir(struct File *f, struct Dir *dout)
|
||
|
{
|
||
|
dout->f = f;
|
||
|
dout->ents = malloc(MAX_DIR_ENTS * sizeof *dout->ents);
|
||
|
dout->n = 0;
|
||
|
}
|
||
|
|
||
|
struct File *
|
||
|
diradd(struct Dir *d, uint32_t type, const char *name)
|
||
|
{
|
||
|
struct File *out = &d->ents[d->n++];
|
||
|
if (d->n > MAX_DIR_ENTS)
|
||
|
panic("too many directory entries");
|
||
|
strcpy(out->f_name, name);
|
||
|
out->f_type = type;
|
||
|
return out;
|
||
|
}
|
||
|
|
||
|
void
|
||
|
finishdir(struct Dir *d)
|
||
|
{
|
||
|
int size = d->n * sizeof(struct File);
|
||
|
struct File *start = alloc(size);
|
||
|
memmove(start, d->ents, size);
|
||
|
finishfile(d->f, blockof(start), ROUNDUP(size, BLKSIZE));
|
||
|
free(d->ents);
|
||
|
d->ents = NULL;
|
||
|
}
|
||
|
|
||
|
void
|
||
|
writefile(struct Dir *dir, const char *name)
|
||
|
{
|
||
|
int r, fd;
|
||
|
struct File *f;
|
||
|
struct stat st;
|
||
|
const char *last;
|
||
|
char *start;
|
||
|
|
||
|
if ((fd = open(name, O_RDONLY)) < 0)
|
||
|
panic("open %s: %s", name, strerror(errno));
|
||
|
if ((r = fstat(fd, &st)) < 0)
|
||
|
panic("stat %s: %s", name, strerror(errno));
|
||
|
if (!S_ISREG(st.st_mode))
|
||
|
panic("%s is not a regular file", name);
|
||
|
if (st.st_size >= MAXFILESIZE)
|
||
|
panic("%s too large", name);
|
||
|
|
||
|
last = strrchr(name, '/');
|
||
|
if (last)
|
||
|
last++;
|
||
|
else
|
||
|
last = name;
|
||
|
|
||
|
f = diradd(dir, FTYPE_REG, last);
|
||
|
start = alloc(st.st_size);
|
||
|
readn(fd, start, st.st_size);
|
||
|
finishfile(f, blockof(start), st.st_size);
|
||
|
close(fd);
|
||
|
}
|
||
|
|
||
|
void
|
||
|
usage(void)
|
||
|
{
|
||
|
fprintf(stderr, "Usage: fsformat fs.img NBLOCKS files...\n");
|
||
|
exit(2);
|
||
|
}
|
||
|
|
||
|
int
|
||
|
main(int argc, char **argv)
|
||
|
{
|
||
|
int i;
|
||
|
char *s;
|
||
|
struct Dir root;
|
||
|
|
||
|
assert(BLKSIZE % sizeof(struct File) == 0);
|
||
|
|
||
|
if (argc < 3)
|
||
|
usage();
|
||
|
|
||
|
nblocks = strtol(argv[2], &s, 0);
|
||
|
if (*s || s == argv[2] || nblocks < 2 || nblocks > 1024)
|
||
|
usage();
|
||
|
|
||
|
opendisk(argv[1]);
|
||
|
|
||
|
startdir(&super->s_root, &root);
|
||
|
for (i = 3; i < argc; i++)
|
||
|
writefile(&root, argv[i]);
|
||
|
finishdir(&root);
|
||
|
|
||
|
finishdisk();
|
||
|
return 0;
|
||
|
}
|
||
|
|