2010-08-02 13:44:45 +02:00
|
|
|
/* Created (MFS based):
|
|
|
|
* February 2010 (Evgeniy Ivanov)
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "fs.h"
|
|
|
|
#include <assert.h>
|
|
|
|
#include <minix/vfsif.h>
|
2011-11-09 14:29:12 +01:00
|
|
|
#include <minix/bdev.h>
|
2010-08-02 13:44:45 +02:00
|
|
|
#include "inode.h"
|
|
|
|
#include "super.h"
|
|
|
|
|
|
|
|
/*===========================================================================*
|
|
|
|
* fs_sync *
|
|
|
|
*===========================================================================*/
|
2012-03-25 20:25:53 +02:00
|
|
|
int fs_sync()
|
2010-08-02 13:44:45 +02:00
|
|
|
{
|
|
|
|
/* Perform the sync() system call. Flush all the tables.
|
|
|
|
* The order in which the various tables are flushed is critical. The
|
|
|
|
* blocks must be flushed last, since rw_inode() leaves its results in
|
|
|
|
* the block cache.
|
|
|
|
*/
|
|
|
|
struct inode *rip;
|
|
|
|
struct buf *bp;
|
|
|
|
|
|
|
|
assert(nr_bufs > 0);
|
|
|
|
assert(buf);
|
|
|
|
|
|
|
|
if (superblock->s_rd_only)
|
|
|
|
return(OK); /* nothing to sync */
|
|
|
|
|
|
|
|
/* Write all the dirty inodes to the disk. */
|
|
|
|
for(rip = &inode[0]; rip < &inode[NR_INODES]; rip++)
|
|
|
|
if(rip->i_count > 0 && rip->i_dirt == DIRTY) rw_inode(rip, WRITING);
|
|
|
|
|
|
|
|
/* Write all the dirty blocks to the disk, one drive at a time. */
|
|
|
|
for(bp = &buf[0]; bp < &buf[nr_bufs]; bp++)
|
|
|
|
if(bp->b_dev != NO_DEV && bp->b_dirt == DIRTY)
|
|
|
|
flushall(bp->b_dev);
|
|
|
|
|
|
|
|
if (superblock->s_dev != NO_DEV) {
|
|
|
|
superblock->s_wtime = clock_time();
|
|
|
|
write_super(superblock);
|
|
|
|
}
|
|
|
|
|
|
|
|
return(OK); /* sync() can't fail */
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*===========================================================================*
|
|
|
|
* fs_flush *
|
|
|
|
*===========================================================================*/
|
2012-03-25 20:25:53 +02:00
|
|
|
int fs_flush()
|
2010-08-02 13:44:45 +02:00
|
|
|
{
|
|
|
|
/* Flush the blocks of a device from the cache after writing any dirty blocks
|
|
|
|
* to disk.
|
|
|
|
*/
|
|
|
|
dev_t dev = (dev_t) fs_m_in.REQ_DEV;
|
|
|
|
|
|
|
|
if(dev == fs_dev) return(EBUSY);
|
|
|
|
|
|
|
|
flushall(dev);
|
|
|
|
invalidate(dev);
|
|
|
|
|
|
|
|
return(OK);
|
|
|
|
}
|
2011-11-09 14:29:12 +01:00
|
|
|
|
|
|
|
/*===========================================================================*
|
|
|
|
* fs_new_driver *
|
|
|
|
*===========================================================================*/
|
2012-03-25 20:25:53 +02:00
|
|
|
int fs_new_driver(void)
|
2011-11-09 14:29:12 +01:00
|
|
|
{
|
|
|
|
/* Set a new driver endpoint for this device. */
|
|
|
|
dev_t dev;
|
2011-11-30 19:05:26 +01:00
|
|
|
cp_grant_id_t label_gid;
|
|
|
|
size_t label_len;
|
|
|
|
char label[sizeof(fs_dev_label)];
|
|
|
|
int r;
|
2011-11-09 14:29:12 +01:00
|
|
|
|
|
|
|
dev = (dev_t) fs_m_in.REQ_DEV;
|
2011-11-30 19:05:26 +01:00
|
|
|
label_gid = (cp_grant_id_t) fs_m_in.REQ_GRANT;
|
|
|
|
label_len = (size_t) fs_m_in.REQ_PATH_LEN;
|
2011-11-09 14:29:12 +01:00
|
|
|
|
2011-11-30 19:05:26 +01:00
|
|
|
if (label_len > sizeof(label))
|
|
|
|
return(EINVAL);
|
|
|
|
|
|
|
|
r = sys_safecopyfrom(fs_m_in.m_source, label_gid, (vir_bytes) 0,
|
2012-06-16 03:46:15 +02:00
|
|
|
(vir_bytes) label, label_len);
|
2011-11-30 19:05:26 +01:00
|
|
|
|
|
|
|
if (r != OK) {
|
|
|
|
printf("ext2: fs_new_driver safecopyfrom failed (%d)\n", r);
|
|
|
|
return(EINVAL);
|
|
|
|
}
|
|
|
|
|
|
|
|
bdev_driver(dev, label);
|
2011-11-09 14:29:12 +01:00
|
|
|
|
|
|
|
return(OK);
|
|
|
|
}
|