Commit a5789b07 authored by Jan Kara's avatar Jan Kara Committed by Paul Moore

audit: Fix possible spurious -ENOSPC error

When an inode is tagged with a tree, tag_chunk() checks whether there is
audit_tree_group mark attached to the inode and adds one if not. However
nothing protects another tag_chunk() to add the mark between we've
checked and try to add the fsnotify mark thus resulting in an error from
fsnotify_add_mark() and consequently an ENOSPC error from tag_chunk().

Fix the problem by holding mark_mutex over the whole check-insert code
sequence.
Reviewed-by: default avatarRichard Guy Briggs <rgb@redhat.com>
Signed-off-by: default avatarJan Kara <jack@suse.cz>
Signed-off-by: default avatarPaul Moore <paul@paul-moore.com>
parent 9f16d2e6
...@@ -342,25 +342,29 @@ static void untag_chunk(struct node *p) ...@@ -342,25 +342,29 @@ static void untag_chunk(struct node *p)
spin_lock(&hash_lock); spin_lock(&hash_lock);
} }
/* Call with group->mark_mutex held, releases it */
static int create_chunk(struct inode *inode, struct audit_tree *tree) static int create_chunk(struct inode *inode, struct audit_tree *tree)
{ {
struct fsnotify_mark *entry; struct fsnotify_mark *entry;
struct audit_chunk *chunk = alloc_chunk(1); struct audit_chunk *chunk = alloc_chunk(1);
if (!chunk)
if (!chunk) {
mutex_unlock(&audit_tree_group->mark_mutex);
return -ENOMEM; return -ENOMEM;
}
entry = &chunk->mark; entry = &chunk->mark;
if (fsnotify_add_inode_mark(entry, inode, 0)) { if (fsnotify_add_inode_mark_locked(entry, inode, 0)) {
mutex_unlock(&audit_tree_group->mark_mutex);
fsnotify_put_mark(entry); fsnotify_put_mark(entry);
return -ENOSPC; return -ENOSPC;
} }
mutex_lock(&entry->group->mark_mutex);
spin_lock(&hash_lock); spin_lock(&hash_lock);
if (tree->goner) { if (tree->goner) {
spin_unlock(&hash_lock); spin_unlock(&hash_lock);
chunk->dead = 1; chunk->dead = 1;
mutex_unlock(&entry->group->mark_mutex); mutex_unlock(&audit_tree_group->mark_mutex);
fsnotify_destroy_mark(entry, audit_tree_group); fsnotify_destroy_mark(entry, audit_tree_group);
fsnotify_put_mark(entry); fsnotify_put_mark(entry);
return 0; return 0;
...@@ -375,7 +379,7 @@ static int create_chunk(struct inode *inode, struct audit_tree *tree) ...@@ -375,7 +379,7 @@ static int create_chunk(struct inode *inode, struct audit_tree *tree)
} }
insert_hash(chunk); insert_hash(chunk);
spin_unlock(&hash_lock); spin_unlock(&hash_lock);
mutex_unlock(&entry->group->mark_mutex); mutex_unlock(&audit_tree_group->mark_mutex);
fsnotify_put_mark(entry); /* drop initial reference */ fsnotify_put_mark(entry); /* drop initial reference */
return 0; return 0;
} }
...@@ -389,6 +393,7 @@ static int tag_chunk(struct inode *inode, struct audit_tree *tree) ...@@ -389,6 +393,7 @@ static int tag_chunk(struct inode *inode, struct audit_tree *tree)
struct node *p; struct node *p;
int n; int n;
mutex_lock(&audit_tree_group->mark_mutex);
old_entry = fsnotify_find_mark(&inode->i_fsnotify_marks, old_entry = fsnotify_find_mark(&inode->i_fsnotify_marks,
audit_tree_group); audit_tree_group);
if (!old_entry) if (!old_entry)
...@@ -401,6 +406,7 @@ static int tag_chunk(struct inode *inode, struct audit_tree *tree) ...@@ -401,6 +406,7 @@ static int tag_chunk(struct inode *inode, struct audit_tree *tree)
for (n = 0; n < old->count; n++) { for (n = 0; n < old->count; n++) {
if (old->owners[n].owner == tree) { if (old->owners[n].owner == tree) {
spin_unlock(&hash_lock); spin_unlock(&hash_lock);
mutex_unlock(&audit_tree_group->mark_mutex);
fsnotify_put_mark(old_entry); fsnotify_put_mark(old_entry);
return 0; return 0;
} }
...@@ -409,20 +415,20 @@ static int tag_chunk(struct inode *inode, struct audit_tree *tree) ...@@ -409,20 +415,20 @@ static int tag_chunk(struct inode *inode, struct audit_tree *tree)
chunk = alloc_chunk(old->count + 1); chunk = alloc_chunk(old->count + 1);
if (!chunk) { if (!chunk) {
mutex_unlock(&audit_tree_group->mark_mutex);
fsnotify_put_mark(old_entry); fsnotify_put_mark(old_entry);
return -ENOMEM; return -ENOMEM;
} }
chunk_entry = &chunk->mark; chunk_entry = &chunk->mark;
mutex_lock(&old_entry->group->mark_mutex);
/* /*
* mark_mutex protects mark from getting detached and thus also from * mark_mutex protects mark from getting detached and thus also from
* mark->connector->obj getting NULL. * mark->connector->obj getting NULL.
*/ */
if (!(old_entry->flags & FSNOTIFY_MARK_FLAG_ATTACHED)) { if (!(old_entry->flags & FSNOTIFY_MARK_FLAG_ATTACHED)) {
/* old_entry is being shot, lets just lie */ /* old_entry is being shot, lets just lie */
mutex_unlock(&old_entry->group->mark_mutex); mutex_unlock(&audit_tree_group->mark_mutex);
fsnotify_put_mark(old_entry); fsnotify_put_mark(old_entry);
fsnotify_put_mark(&chunk->mark); fsnotify_put_mark(&chunk->mark);
return -ENOENT; return -ENOENT;
...@@ -430,7 +436,7 @@ static int tag_chunk(struct inode *inode, struct audit_tree *tree) ...@@ -430,7 +436,7 @@ static int tag_chunk(struct inode *inode, struct audit_tree *tree)
if (fsnotify_add_mark_locked(chunk_entry, old_entry->connector->obj, if (fsnotify_add_mark_locked(chunk_entry, old_entry->connector->obj,
FSNOTIFY_OBJ_TYPE_INODE, 1)) { FSNOTIFY_OBJ_TYPE_INODE, 1)) {
mutex_unlock(&old_entry->group->mark_mutex); mutex_unlock(&audit_tree_group->mark_mutex);
fsnotify_put_mark(chunk_entry); fsnotify_put_mark(chunk_entry);
fsnotify_put_mark(old_entry); fsnotify_put_mark(old_entry);
return -ENOSPC; return -ENOSPC;
...@@ -440,7 +446,7 @@ static int tag_chunk(struct inode *inode, struct audit_tree *tree) ...@@ -440,7 +446,7 @@ static int tag_chunk(struct inode *inode, struct audit_tree *tree)
if (tree->goner) { if (tree->goner) {
spin_unlock(&hash_lock); spin_unlock(&hash_lock);
chunk->dead = 1; chunk->dead = 1;
mutex_unlock(&old_entry->group->mark_mutex); mutex_unlock(&audit_tree_group->mark_mutex);
fsnotify_destroy_mark(chunk_entry, audit_tree_group); fsnotify_destroy_mark(chunk_entry, audit_tree_group);
...@@ -471,7 +477,7 @@ static int tag_chunk(struct inode *inode, struct audit_tree *tree) ...@@ -471,7 +477,7 @@ static int tag_chunk(struct inode *inode, struct audit_tree *tree)
list_add(&tree->same_root, &chunk->trees); list_add(&tree->same_root, &chunk->trees);
} }
spin_unlock(&hash_lock); spin_unlock(&hash_lock);
mutex_unlock(&old_entry->group->mark_mutex); mutex_unlock(&audit_tree_group->mark_mutex);
fsnotify_destroy_mark(old_entry, audit_tree_group); fsnotify_destroy_mark(old_entry, audit_tree_group);
fsnotify_put_mark(chunk_entry); /* drop initial reference */ fsnotify_put_mark(chunk_entry); /* drop initial reference */
fsnotify_put_mark(old_entry); /* pair to fsnotify_find mark_entry */ fsnotify_put_mark(old_entry); /* pair to fsnotify_find mark_entry */
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment