InodeAllocator now takes care of calling new_vnode() and the new publish_vnode().
bfs_inode::InitCheck() tested INODE_NOT_READY before making sure the block is an inode at all, resulting in spurious "inode is not becoming unbusy" errors. Since there is now publish_vnode() we don't even have to return B_BUSY at all anymore, though - INODE_NOT_READY is no longer used at all. bfs_inode::etc is no longer used either (and for the same reason). Inode::Create() no longer locks the whole volume, but only the write lock of the parent directory (if any). git-svn-id: file:///srv/svn/repos/haiku/trunk/current@11723 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
parent
fa4244bffc
commit
b7366070fa
@ -44,12 +44,15 @@ InodeAllocator::InodeAllocator(Transaction &transaction)
|
|||||||
InodeAllocator::~InodeAllocator()
|
InodeAllocator::~InodeAllocator()
|
||||||
{
|
{
|
||||||
if (fTransaction != NULL) {
|
if (fTransaction != NULL) {
|
||||||
|
Volume *volume = fTransaction->GetVolume();
|
||||||
|
|
||||||
if (fInode != NULL) {
|
if (fInode != NULL) {
|
||||||
fInode->Node().flags &= ~HOST_ENDIAN_TO_BFS_INT32(INODE_IN_USE | INODE_NOT_READY);
|
fInode->Node().flags &= ~HOST_ENDIAN_TO_BFS_INT32(INODE_IN_USE);
|
||||||
// this unblocks any pending bfs_read_vnode() calls
|
// this unblocks any pending bfs_read_vnode() calls
|
||||||
fInode->Free(*fTransaction);
|
fInode->Free(*fTransaction);
|
||||||
|
remove_vnode(volume->ID(), fInode->ID());
|
||||||
} else
|
} else
|
||||||
fTransaction->GetVolume()->Free(*fTransaction, fRun);
|
volume->Free(*fTransaction, fRun);
|
||||||
}
|
}
|
||||||
|
|
||||||
delete fInode;
|
delete fInode;
|
||||||
@ -74,6 +77,13 @@ InodeAllocator::New(block_run *parentRun, mode_t mode, block_run &run, Inode **_
|
|||||||
if (fInode == NULL)
|
if (fInode == NULL)
|
||||||
RETURN_ERROR(B_NO_MEMORY);
|
RETURN_ERROR(B_NO_MEMORY);
|
||||||
|
|
||||||
|
status = new_vnode(volume->ID(), fInode->ID(), fInode);
|
||||||
|
if (status < B_OK) {
|
||||||
|
delete fInode;
|
||||||
|
fInode = NULL;
|
||||||
|
RETURN_ERROR(status);
|
||||||
|
}
|
||||||
|
|
||||||
*_inode = fInode;
|
*_inode = fInode;
|
||||||
return B_OK;
|
return B_OK;
|
||||||
}
|
}
|
||||||
@ -105,9 +115,15 @@ status_t
|
|||||||
InodeAllocator::Keep()
|
InodeAllocator::Keep()
|
||||||
{
|
{
|
||||||
ASSERT(fInode != NULL && fTransaction != NULL);
|
ASSERT(fInode != NULL && fTransaction != NULL);
|
||||||
|
Volume *volume = fTransaction->GetVolume();
|
||||||
|
|
||||||
fInode->Node().flags &= ~HOST_ENDIAN_TO_BFS_INT32(INODE_NOT_READY);
|
|
||||||
status_t status = fInode->WriteBack(*fTransaction);
|
status_t status = fInode->WriteBack(*fTransaction);
|
||||||
|
if (status < B_OK) {
|
||||||
|
FATAL(("writing new inode %Ld failed!\n", fInode->ID()));
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
|
||||||
|
status = publish_vnode(volume->ID(), fInode->ID(), fInode);
|
||||||
|
|
||||||
fTransaction = NULL;
|
fTransaction = NULL;
|
||||||
fInode = NULL;
|
fInode = NULL;
|
||||||
@ -122,13 +138,6 @@ InodeAllocator::Keep()
|
|||||||
status_t
|
status_t
|
||||||
bfs_inode::InitCheck(Volume *volume)
|
bfs_inode::InitCheck(Volume *volume)
|
||||||
{
|
{
|
||||||
if (Flags() & INODE_NOT_READY) {
|
|
||||||
// the other fields may not yet contain valid values
|
|
||||||
return B_BUSY;
|
|
||||||
}
|
|
||||||
if (Flags() & INODE_DELETED)
|
|
||||||
return B_NOT_ALLOWED;
|
|
||||||
|
|
||||||
if (Magic1() != INODE_MAGIC1
|
if (Magic1() != INODE_MAGIC1
|
||||||
|| !(Flags() & INODE_IN_USE)
|
|| !(Flags() & INODE_IN_USE)
|
||||||
|| inode_num.Length() != 1
|
|| inode_num.Length() != 1
|
||||||
@ -145,6 +154,9 @@ bfs_inode::InitCheck(Volume *volume)
|
|||||||
|| attributes.Start() > (1L << volume->AllocationGroupShift()))
|
|| attributes.Start() > (1L << volume->AllocationGroupShift()))
|
||||||
RETURN_ERROR(B_BAD_DATA);
|
RETURN_ERROR(B_BAD_DATA);
|
||||||
|
|
||||||
|
if (Flags() & INODE_DELETED)
|
||||||
|
return B_NOT_ALLOWED;
|
||||||
|
|
||||||
// ToDo: Add some tests to check the integrity of the other stuff here,
|
// ToDo: Add some tests to check the integrity of the other stuff here,
|
||||||
// especially for the data_stream!
|
// especially for the data_stream!
|
||||||
|
|
||||||
@ -205,12 +217,7 @@ Inode::Inode(Volume *volume, Transaction &transaction, vnode_id id, mode_t mode,
|
|||||||
Node().magic1 = HOST_ENDIAN_TO_BFS_INT32(INODE_MAGIC1);
|
Node().magic1 = HOST_ENDIAN_TO_BFS_INT32(INODE_MAGIC1);
|
||||||
Node().inode_num = run;
|
Node().inode_num = run;
|
||||||
Node().mode = HOST_ENDIAN_TO_BFS_INT32(mode);
|
Node().mode = HOST_ENDIAN_TO_BFS_INT32(mode);
|
||||||
Node().flags = HOST_ENDIAN_TO_BFS_INT32(INODE_IN_USE | INODE_NOT_READY);
|
Node().flags = HOST_ENDIAN_TO_BFS_INT32(INODE_IN_USE);
|
||||||
// INODE_NOT_READY prevents the inode from being opened - it is
|
|
||||||
// cleared in InodeAllocator::Keep()
|
|
||||||
Node().etc = (uint32)this;
|
|
||||||
// this is temporarily set along INODE_NOT_READY and lets bfs_read_vnode()
|
|
||||||
// find the associated Inode object
|
|
||||||
|
|
||||||
Node().create_time = HOST_ENDIAN_TO_BFS_INT64((bigtime_t)time(NULL) << INODE_TIME_SHIFT);
|
Node().create_time = HOST_ENDIAN_TO_BFS_INT64((bigtime_t)time(NULL) << INODE_TIME_SHIFT);
|
||||||
Node().last_modified_time = HOST_ENDIAN_TO_BFS_INT64(Node().create_time
|
Node().last_modified_time = HOST_ENDIAN_TO_BFS_INT64(Node().create_time
|
||||||
@ -2087,15 +2094,16 @@ Inode::Create(Transaction &transaction, Inode *parent, const char *name, int32 m
|
|||||||
Volume *volume = transaction.GetVolume();
|
Volume *volume = transaction.GetVolume();
|
||||||
BPlusTree *tree = NULL;
|
BPlusTree *tree = NULL;
|
||||||
|
|
||||||
RecursiveLocker locker(volume->Lock());
|
|
||||||
// ToDo: it would be nicer to only lock the parent directory, if possible
|
|
||||||
// (but that lock will already be held during any B+tree action)
|
|
||||||
|
|
||||||
if (parent && (mode & S_ATTR_DIR) == 0 && parent->IsContainer()) {
|
if (parent && (mode & S_ATTR_DIR) == 0 && parent->IsContainer()) {
|
||||||
// check if the file already exists in the directory
|
// check if the file already exists in the directory
|
||||||
if (parent->GetTree(&tree) != B_OK)
|
if (parent->GetTree(&tree) != B_OK)
|
||||||
RETURN_ERROR(B_BAD_VALUE);
|
RETURN_ERROR(B_BAD_VALUE);
|
||||||
|
}
|
||||||
|
|
||||||
|
WriteLocked locker(tree != NULL ? &parent->Lock() : NULL);
|
||||||
|
// the parent directory is locked during the whole inode creation
|
||||||
|
|
||||||
|
if (tree != NULL) {
|
||||||
// does the file already exist?
|
// does the file already exist?
|
||||||
off_t offset;
|
off_t offset;
|
||||||
if (tree->Find((uint8 *)name, (uint16)strlen(name), &offset) == B_OK) {
|
if (tree->Find((uint8 *)name, (uint16)strlen(name), &offset) == B_OK) {
|
||||||
@ -2193,7 +2201,7 @@ Inode::Create(Transaction &transaction, Inode *parent, const char *name, int32 m
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Add a link to the inode from the parent, depending on its type
|
// Add a link to the inode from the parent, depending on its type
|
||||||
// (the INODE_NOT_READY flag is set, so it is safe to make the inode
|
// (the vnode is not published yet, so it is safe to make the inode
|
||||||
// accessable to the file system here)
|
// accessable to the file system here)
|
||||||
if (tree) {
|
if (tree) {
|
||||||
status = tree->Insert(transaction, name, inode->ID());
|
status = tree->Insert(transaction, name, inode->ID());
|
||||||
@ -2245,16 +2253,6 @@ Inode::Create(Transaction &transaction, Inode *parent, const char *name, int32 m
|
|||||||
// initialized inode, and we want to keep it
|
// initialized inode, and we want to keep it
|
||||||
allocator.Keep();
|
allocator.Keep();
|
||||||
|
|
||||||
// We hold the volume lock to make sure that bfs_read_vnode()
|
|
||||||
// won't succeed in the meantime (between the call right
|
|
||||||
// above and below)!
|
|
||||||
|
|
||||||
if ((status = new_vnode(volume->ID(), inode->ID(), inode)) != B_OK) {
|
|
||||||
// this is a really fatal error, and we can't recover from that
|
|
||||||
FATAL(("new_vnode() failed with: %s\n", strerror(status)));
|
|
||||||
DIE(("new_vnode() failed for inode!"));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (inode->IsFile() || inode->IsAttribute())
|
if (inode->IsFile() || inode->IsAttribute())
|
||||||
inode->SetFileCache(file_cache_create(volume->ID(), inode->ID(), inode->Size(), volume->Device()));
|
inode->SetFileCache(file_cache_create(volume->ID(), inode->ID(), inode->Size(), volume->Device()));
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user