Skip to content

Commit 5d3ee20

Browse files
Dmitry Monakhovtytso
Dmitry Monakhov
authored andcommitted
ext4: fix journal callback list traversal
It is incorrect to use list_for_each_entry_safe() for journal callback traversial because ->next may be removed by other task: ->ext4_mb_free_metadata() ->ext4_mb_free_metadata() ->ext4_journal_callback_del() This results in the following issue: WARNING: at lib/list_debug.c:62 __list_del_entry+0x1c0/0x250() Hardware name: list_del corruption. prev->next should be ffff88019a4ec198, but was 6b6b6b6b6b6b6b6b Modules linked in: cpufreq_ondemand acpi_cpufreq freq_table mperf coretemp kvm_intel kvm crc32c_intel ghash_clmulni_intel microcode sg xhci_hcd button sd_mod crc_t10dif aesni_intel ablk_helper cryptd lrw aes_x86_64 xts gf128mul ahci libahci pata_acpi ata_generic dm_mirror dm_region_hash dm_log dm_mod Pid: 16400, comm: jbd2/dm-1-8 Tainted: G W 3.8.0-rc3+ #107 Call Trace: [<ffffffff8106fb0d>] warn_slowpath_common+0xad/0xf0 [<ffffffff8106fc06>] warn_slowpath_fmt+0x46/0x50 [<ffffffff813637e9>] ? ext4_journal_commit_callback+0x99/0xc0 [<ffffffff8148cae0>] __list_del_entry+0x1c0/0x250 [<ffffffff813637bf>] ext4_journal_commit_callback+0x6f/0xc0 [<ffffffff813ca336>] jbd2_journal_commit_transaction+0x23a6/0x2570 [<ffffffff8108aa42>] ? try_to_del_timer_sync+0x82/0xa0 [<ffffffff8108b491>] ? del_timer_sync+0x91/0x1e0 [<ffffffff813d3ecf>] kjournald2+0x19f/0x6a0 [<ffffffff810ad630>] ? wake_up_bit+0x40/0x40 [<ffffffff813d3d30>] ? bit_spin_lock+0x80/0x80 [<ffffffff810ac6be>] kthread+0x10e/0x120 [<ffffffff810ac5b0>] ? __init_kthread_worker+0x70/0x70 [<ffffffff818ff6ac>] ret_from_fork+0x7c/0xb0 [<ffffffff810ac5b0>] ? __init_kthread_worker+0x70/0x70 This patch fix the issue as follows: - ext4_journal_commit_callback() make list truly traversial safe simply by always starting from list_head - fix race between two ext4_journal_callback_del() and ext4_journal_callback_try_del() Signed-off-by: Dmitry Monakhov <[email protected]> Signed-off-by: "Theodore Ts'o" <[email protected]> Reviewed-by: Jan Kara <[email protected]> Cc: [email protected]
1 parent 794446c commit 5d3ee20

File tree

3 files changed

+14
-7
lines changed

3 files changed

+14
-7
lines changed

fs/ext4/ext4_jbd2.h

Lines changed: 5 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -194,16 +194,20 @@ static inline void ext4_journal_callback_add(handle_t *handle,
194194
* ext4_journal_callback_del: delete a registered callback
195195
* @handle: active journal transaction handle on which callback was registered
196196
* @jce: registered journal callback entry to unregister
197+
* Return true if object was sucessfully removed
197198
*/
198-
static inline void ext4_journal_callback_del(handle_t *handle,
199+
static inline bool ext4_journal_callback_try_del(handle_t *handle,
199200
struct ext4_journal_cb_entry *jce)
200201
{
202+
bool deleted;
201203
struct ext4_sb_info *sbi =
202204
EXT4_SB(handle->h_transaction->t_journal->j_private);
203205

204206
spin_lock(&sbi->s_md_lock);
207+
deleted = !list_empty(&jce->jce_list);
205208
list_del_init(&jce->jce_list);
206209
spin_unlock(&sbi->s_md_lock);
210+
return deleted;
207211
}
208212

209213
int

fs/ext4/mballoc.c

Lines changed: 4 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -4423,22 +4423,22 @@ ext4_mb_free_metadata(handle_t *handle, struct ext4_buddy *e4b,
44234423
node = rb_prev(new_node);
44244424
if (node) {
44254425
entry = rb_entry(node, struct ext4_free_data, efd_node);
4426-
if (can_merge(entry, new_entry)) {
4426+
if (can_merge(entry, new_entry) &&
4427+
ext4_journal_callback_try_del(handle, &entry->efd_jce)) {
44274428
new_entry->efd_start_cluster = entry->efd_start_cluster;
44284429
new_entry->efd_count += entry->efd_count;
44294430
rb_erase(node, &(db->bb_free_root));
4430-
ext4_journal_callback_del(handle, &entry->efd_jce);
44314431
kmem_cache_free(ext4_free_data_cachep, entry);
44324432
}
44334433
}
44344434

44354435
node = rb_next(new_node);
44364436
if (node) {
44374437
entry = rb_entry(node, struct ext4_free_data, efd_node);
4438-
if (can_merge(new_entry, entry)) {
4438+
if (can_merge(new_entry, entry) &&
4439+
ext4_journal_callback_try_del(handle, &entry->efd_jce)) {
44394440
new_entry->efd_count += entry->efd_count;
44404441
rb_erase(node, &(db->bb_free_root));
4441-
ext4_journal_callback_del(handle, &entry->efd_jce);
44424442
kmem_cache_free(ext4_free_data_cachep, entry);
44434443
}
44444444
}

fs/ext4/super.c

Lines changed: 5 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -353,10 +353,13 @@ static void ext4_journal_commit_callback(journal_t *journal, transaction_t *txn)
353353
struct super_block *sb = journal->j_private;
354354
struct ext4_sb_info *sbi = EXT4_SB(sb);
355355
int error = is_journal_aborted(journal);
356-
struct ext4_journal_cb_entry *jce, *tmp;
356+
struct ext4_journal_cb_entry *jce;
357357

358+
BUG_ON(txn->t_state == T_FINISHED);
358359
spin_lock(&sbi->s_md_lock);
359-
list_for_each_entry_safe(jce, tmp, &txn->t_private_list, jce_list) {
360+
while (!list_empty(&txn->t_private_list)) {
361+
jce = list_entry(txn->t_private_list.next,
362+
struct ext4_journal_cb_entry, jce_list);
360363
list_del_init(&jce->jce_list);
361364
spin_unlock(&sbi->s_md_lock);
362365
jce->jce_func(sb, jce, error);

0 commit comments

Comments
 (0)