diff options
Diffstat (limited to 'fs/jbd2/journal.c')
| -rw-r--r-- | fs/jbd2/journal.c | 43 | 
1 files changed, 36 insertions, 7 deletions
| diff --git a/fs/jbd2/journal.c b/fs/jbd2/journal.c index 81e622681c82..de73a9516a54 100644 --- a/fs/jbd2/journal.c +++ b/fs/jbd2/journal.c @@ -805,10 +805,13 @@ int jbd2_journal_bmap(journal_t *journal, unsigned long blocknr,   * But we don't bother doing that, so there will be coherency problems with   * mmaps of blockdevs which hold live JBD-controlled filesystems.   */ -struct buffer_head *jbd2_journal_get_descriptor_buffer(journal_t *journal) +struct buffer_head * +jbd2_journal_get_descriptor_buffer(transaction_t *transaction, int type)  { +	journal_t *journal = transaction->t_journal;  	struct buffer_head *bh;  	unsigned long long blocknr; +	journal_header_t *header;  	int err;  	err = jbd2_journal_next_log_block(journal, &blocknr); @@ -821,12 +824,31 @@ struct buffer_head *jbd2_journal_get_descriptor_buffer(journal_t *journal)  		return NULL;  	lock_buffer(bh);  	memset(bh->b_data, 0, journal->j_blocksize); +	header = (journal_header_t *)bh->b_data; +	header->h_magic = cpu_to_be32(JBD2_MAGIC_NUMBER); +	header->h_blocktype = cpu_to_be32(type); +	header->h_sequence = cpu_to_be32(transaction->t_tid);  	set_buffer_uptodate(bh);  	unlock_buffer(bh);  	BUFFER_TRACE(bh, "return this buffer");  	return bh;  } +void jbd2_descriptor_block_csum_set(journal_t *j, struct buffer_head *bh) +{ +	struct jbd2_journal_block_tail *tail; +	__u32 csum; + +	if (!jbd2_journal_has_csum_v2or3(j)) +		return; + +	tail = (struct jbd2_journal_block_tail *)(bh->b_data + j->j_blocksize - +			sizeof(struct jbd2_journal_block_tail)); +	tail->t_checksum = 0; +	csum = jbd2_chksum(j, j->j_csum_seed, bh->b_data, j->j_blocksize); +	tail->t_checksum = cpu_to_be32(csum); +} +  /*   * Return tid of the oldest transaction in the journal and block in the journal   * where the transaction starts. @@ -1408,11 +1430,12 @@ out:  /**   * jbd2_mark_journal_empty() - Mark on disk journal as empty.   * @journal: The journal to update. + * @write_op: With which operation should we write the journal sb   *   * Update a journal's dynamic superblock fields to show that journal is empty.   * Write updated superblock to disk waiting for IO to complete.   */ -static void jbd2_mark_journal_empty(journal_t *journal) +static void jbd2_mark_journal_empty(journal_t *journal, int write_op)  {  	journal_superblock_t *sb = journal->j_superblock; @@ -1430,7 +1453,7 @@ static void jbd2_mark_journal_empty(journal_t *journal)  	sb->s_start    = cpu_to_be32(0);  	read_unlock(&journal->j_state_lock); -	jbd2_write_superblock(journal, WRITE_FUA); +	jbd2_write_superblock(journal, write_op);  	/* Log is no longer empty */  	write_lock(&journal->j_state_lock); @@ -1716,7 +1739,13 @@ int jbd2_journal_destroy(journal_t *journal)  	if (journal->j_sb_buffer) {  		if (!is_journal_aborted(journal)) {  			mutex_lock(&journal->j_checkpoint_mutex); -			jbd2_mark_journal_empty(journal); + +			write_lock(&journal->j_state_lock); +			journal->j_tail_sequence = +				++journal->j_transaction_sequence; +			write_unlock(&journal->j_state_lock); + +			jbd2_mark_journal_empty(journal, WRITE_FLUSH_FUA);  			mutex_unlock(&journal->j_checkpoint_mutex);  		} else  			err = -EIO; @@ -1975,7 +2004,7 @@ int jbd2_journal_flush(journal_t *journal)  	 * the magic code for a fully-recovered superblock.  Any future  	 * commits of data to the journal will restore the current  	 * s_start value. */ -	jbd2_mark_journal_empty(journal); +	jbd2_mark_journal_empty(journal, WRITE_FUA);  	mutex_unlock(&journal->j_checkpoint_mutex);  	write_lock(&journal->j_state_lock);  	J_ASSERT(!journal->j_running_transaction); @@ -2021,7 +2050,7 @@ int jbd2_journal_wipe(journal_t *journal, int write)  	if (write) {  		/* Lock to make assertions happy... */  		mutex_lock(&journal->j_checkpoint_mutex); -		jbd2_mark_journal_empty(journal); +		jbd2_mark_journal_empty(journal, WRITE_FUA);  		mutex_unlock(&journal->j_checkpoint_mutex);  	} @@ -2565,7 +2594,7 @@ void jbd2_journal_release_jbd_inode(journal_t *journal,  restart:  	spin_lock(&journal->j_list_lock);  	/* Is commit writing out inode - we have to wait */ -	if (test_bit(__JI_COMMIT_RUNNING, &jinode->i_flags)) { +	if (jinode->i_flags & JI_COMMIT_RUNNING) {  		wait_queue_head_t *wq;  		DEFINE_WAIT_BIT(wait, &jinode->i_flags, __JI_COMMIT_RUNNING);  		wq = bit_waitqueue(&jinode->i_flags, __JI_COMMIT_RUNNING); | 
