mirror of
				https://github.com/smaeul/u-boot.git
				synced 2025-10-25 10:08:21 +01:00 
			
		
		
		
	If the same block is updated multiple times in a row during a single file system operation, gd_index is decremented to use the same journal entry again. Avoid loosing the already allocated buffer. Signed-off-by: Stefan Brüns <stefan.bruens@rwth-aachen.de>
		
			
				
	
	
		
			661 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			661 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * (C) Copyright 2011 - 2012 Samsung Electronics
 | |
|  * EXT4 filesystem implementation in Uboot by
 | |
|  * Uma Shankar <uma.shankar@samsung.com>
 | |
|  * Manjunatha C Achar <a.manjunatha@samsung.com>
 | |
|  *
 | |
|  * Journal data structures and headers for Journaling feature of ext4
 | |
|  * have been referred from JBD2 (Journaling Block device 2)
 | |
|  * implementation in Linux Kernel.
 | |
|  * Written by Stephen C. Tweedie <sct@redhat.com>
 | |
|  *
 | |
|  * Copyright 1998-2000 Red Hat, Inc --- All Rights Reserved
 | |
|  * SPDX-License-Identifier:	GPL-2.0+
 | |
|  */
 | |
| 
 | |
| #include <common.h>
 | |
| #include <ext4fs.h>
 | |
| #include <malloc.h>
 | |
| #include <ext_common.h>
 | |
| #include "ext4_common.h"
 | |
| 
 | |
| static struct revoke_blk_list *revk_blk_list;
 | |
| static struct revoke_blk_list *prev_node;
 | |
| static int first_node = true;
 | |
| 
 | |
| int gindex;
 | |
| int gd_index;
 | |
| int jrnl_blk_idx;
 | |
| struct journal_log *journal_ptr[MAX_JOURNAL_ENTRIES];
 | |
| struct dirty_blocks *dirty_block_ptr[MAX_JOURNAL_ENTRIES];
 | |
| 
 | |
| int ext4fs_init_journal(void)
 | |
| {
 | |
| 	int i;
 | |
| 	char *temp = NULL;
 | |
| 	struct ext_filesystem *fs = get_fs();
 | |
| 
 | |
| 	/* init globals */
 | |
| 	revk_blk_list = NULL;
 | |
| 	prev_node = NULL;
 | |
| 	gindex = 0;
 | |
| 	gd_index = 0;
 | |
| 	jrnl_blk_idx = 1;
 | |
| 
 | |
| 	for (i = 0; i < MAX_JOURNAL_ENTRIES; i++) {
 | |
| 		journal_ptr[i] = zalloc(sizeof(struct journal_log));
 | |
| 		if (!journal_ptr[i])
 | |
| 			goto fail;
 | |
| 		dirty_block_ptr[i] = zalloc(sizeof(struct dirty_blocks));
 | |
| 		if (!dirty_block_ptr[i])
 | |
| 			goto fail;
 | |
| 		journal_ptr[i]->buf = NULL;
 | |
| 		journal_ptr[i]->blknr = -1;
 | |
| 
 | |
| 		dirty_block_ptr[i]->buf = NULL;
 | |
| 		dirty_block_ptr[i]->blknr = -1;
 | |
| 	}
 | |
| 
 | |
| 	if (fs->blksz == 4096) {
 | |
| 		temp = zalloc(fs->blksz);
 | |
| 		if (!temp)
 | |
| 			goto fail;
 | |
| 		journal_ptr[gindex]->buf = zalloc(fs->blksz);
 | |
| 		if (!journal_ptr[gindex]->buf)
 | |
| 			goto fail;
 | |
| 		ext4fs_devread(0, 0, fs->blksz, temp);
 | |
| 		memcpy(temp + SUPERBLOCK_SIZE, fs->sb, SUPERBLOCK_SIZE);
 | |
| 		memcpy(journal_ptr[gindex]->buf, temp, fs->blksz);
 | |
| 		journal_ptr[gindex++]->blknr = 0;
 | |
| 		free(temp);
 | |
| 	} else {
 | |
| 		journal_ptr[gindex]->buf = zalloc(fs->blksz);
 | |
| 		if (!journal_ptr[gindex]->buf)
 | |
| 			goto fail;
 | |
| 		memcpy(journal_ptr[gindex]->buf, fs->sb, SUPERBLOCK_SIZE);
 | |
| 		journal_ptr[gindex++]->blknr = 1;
 | |
| 	}
 | |
| 
 | |
| 	/* Check the file system state using journal super block */
 | |
| 	if (ext4fs_check_journal_state(SCAN))
 | |
| 		goto fail;
 | |
| 	/* Check the file system state using journal super block */
 | |
| 	if (ext4fs_check_journal_state(RECOVER))
 | |
| 		goto fail;
 | |
| 
 | |
| 	return 0;
 | |
| fail:
 | |
| 	return -1;
 | |
| }
 | |
| 
 | |
| void ext4fs_dump_metadata(void)
 | |
| {
 | |
| 	struct ext_filesystem *fs = get_fs();
 | |
| 	int i;
 | |
| 	for (i = 0; i < MAX_JOURNAL_ENTRIES; i++) {
 | |
| 		if (dirty_block_ptr[i]->blknr == -1)
 | |
| 			break;
 | |
| 		put_ext4((uint64_t) ((uint64_t)dirty_block_ptr[i]->blknr *
 | |
| 				(uint64_t)fs->blksz), dirty_block_ptr[i]->buf,
 | |
| 								fs->blksz);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void ext4fs_free_journal(void)
 | |
| {
 | |
| 	int i;
 | |
| 	for (i = 0; i < MAX_JOURNAL_ENTRIES; i++) {
 | |
| 		if (dirty_block_ptr[i]->blknr == -1)
 | |
| 			break;
 | |
| 		if (dirty_block_ptr[i]->buf)
 | |
| 			free(dirty_block_ptr[i]->buf);
 | |
| 	}
 | |
| 
 | |
| 	for (i = 0; i < MAX_JOURNAL_ENTRIES; i++) {
 | |
| 		if (journal_ptr[i]->blknr == -1)
 | |
| 			break;
 | |
| 		if (journal_ptr[i]->buf)
 | |
| 			free(journal_ptr[i]->buf);
 | |
| 	}
 | |
| 
 | |
| 	for (i = 0; i < MAX_JOURNAL_ENTRIES; i++) {
 | |
| 		if (journal_ptr[i])
 | |
| 			free(journal_ptr[i]);
 | |
| 		if (dirty_block_ptr[i])
 | |
| 			free(dirty_block_ptr[i]);
 | |
| 	}
 | |
| 	gindex = 0;
 | |
| 	gd_index = 0;
 | |
| 	jrnl_blk_idx = 1;
 | |
| }
 | |
| 
 | |
| int ext4fs_log_gdt(char *gd_table)
 | |
| {
 | |
| 	struct ext_filesystem *fs = get_fs();
 | |
| 	short i;
 | |
| 	long int var = fs->gdtable_blkno;
 | |
| 	for (i = 0; i < fs->no_blk_pergdt; i++) {
 | |
| 		journal_ptr[gindex]->buf = zalloc(fs->blksz);
 | |
| 		if (!journal_ptr[gindex]->buf)
 | |
| 			return -ENOMEM;
 | |
| 		memcpy(journal_ptr[gindex]->buf, gd_table, fs->blksz);
 | |
| 		gd_table += fs->blksz;
 | |
| 		journal_ptr[gindex++]->blknr = var++;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * This function stores the backup copy of meta data in RAM
 | |
|  * journal_buffer -- Buffer containing meta data
 | |
|  * blknr -- Block number on disk of the meta data buffer
 | |
|  */
 | |
| int ext4fs_log_journal(char *journal_buffer, uint32_t blknr)
 | |
| {
 | |
| 	struct ext_filesystem *fs = get_fs();
 | |
| 	short i;
 | |
| 
 | |
| 	if (!journal_buffer) {
 | |
| 		printf("Invalid input arguments %s\n", __func__);
 | |
| 		return -EINVAL;
 | |
| 	}
 | |
| 
 | |
| 	for (i = 0; i < MAX_JOURNAL_ENTRIES; i++) {
 | |
| 		if (journal_ptr[i]->blknr == -1)
 | |
| 			break;
 | |
| 		if (journal_ptr[i]->blknr == blknr)
 | |
| 			return 0;
 | |
| 	}
 | |
| 
 | |
| 	journal_ptr[gindex]->buf = zalloc(fs->blksz);
 | |
| 	if (!journal_ptr[gindex]->buf)
 | |
| 		return -ENOMEM;
 | |
| 
 | |
| 	memcpy(journal_ptr[gindex]->buf, journal_buffer, fs->blksz);
 | |
| 	journal_ptr[gindex++]->blknr = blknr;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * This function stores the modified meta data in RAM
 | |
|  * metadata_buffer -- Buffer containing meta data
 | |
|  * blknr -- Block number on disk of the meta data buffer
 | |
|  */
 | |
| int ext4fs_put_metadata(char *metadata_buffer, uint32_t blknr)
 | |
| {
 | |
| 	struct ext_filesystem *fs = get_fs();
 | |
| 	if (!metadata_buffer) {
 | |
| 		printf("Invalid input arguments %s\n", __func__);
 | |
| 		return -EINVAL;
 | |
| 	}
 | |
| 	if (dirty_block_ptr[gd_index]->buf)
 | |
| 		assert(dirty_block_ptr[gd_index]->blknr == blknr);
 | |
| 	else
 | |
| 		dirty_block_ptr[gd_index]->buf = zalloc(fs->blksz);
 | |
| 
 | |
| 	if (!dirty_block_ptr[gd_index]->buf)
 | |
| 		return -ENOMEM;
 | |
| 	memcpy(dirty_block_ptr[gd_index]->buf, metadata_buffer, fs->blksz);
 | |
| 	dirty_block_ptr[gd_index++]->blknr = blknr;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| void print_revoke_blks(char *revk_blk)
 | |
| {
 | |
| 	int offset;
 | |
| 	int max;
 | |
| 	long int blocknr;
 | |
| 	struct journal_revoke_header_t *header;
 | |
| 
 | |
| 	if (revk_blk == NULL)
 | |
| 		return;
 | |
| 
 | |
| 	header = (struct journal_revoke_header_t *) revk_blk;
 | |
| 	offset = sizeof(struct journal_revoke_header_t);
 | |
| 	max = be32_to_cpu(header->r_count);
 | |
| 	printf("total bytes %d\n", max);
 | |
| 
 | |
| 	while (offset < max) {
 | |
| 		blocknr = be32_to_cpu(*((__be32 *)(revk_blk + offset)));
 | |
| 		printf("revoke blknr is %ld\n", blocknr);
 | |
| 		offset += 4;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static struct revoke_blk_list *_get_node(void)
 | |
| {
 | |
| 	struct revoke_blk_list *tmp_node;
 | |
| 	tmp_node = zalloc(sizeof(struct revoke_blk_list));
 | |
| 	if (tmp_node == NULL)
 | |
| 		return NULL;
 | |
| 	tmp_node->content = NULL;
 | |
| 	tmp_node->next = NULL;
 | |
| 
 | |
| 	return tmp_node;
 | |
| }
 | |
| 
 | |
| void ext4fs_push_revoke_blk(char *buffer)
 | |
| {
 | |
| 	struct revoke_blk_list *node = NULL;
 | |
| 	struct ext_filesystem *fs = get_fs();
 | |
| 	if (buffer == NULL) {
 | |
| 		printf("buffer ptr is NULL\n");
 | |
| 		return;
 | |
| 	}
 | |
| 	node = _get_node();
 | |
| 	if (!node) {
 | |
| 		printf("_get_node: malloc failed\n");
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	node->content = zalloc(fs->blksz);
 | |
| 	if (node->content == NULL)
 | |
| 		return;
 | |
| 	memcpy(node->content, buffer, fs->blksz);
 | |
| 
 | |
| 	if (first_node == true) {
 | |
| 		revk_blk_list = node;
 | |
| 		prev_node = node;
 | |
| 		 first_node = false;
 | |
| 	} else {
 | |
| 		prev_node->next = node;
 | |
| 		prev_node = node;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void ext4fs_free_revoke_blks(void)
 | |
| {
 | |
| 	struct revoke_blk_list *tmp_node = revk_blk_list;
 | |
| 	struct revoke_blk_list *next_node = NULL;
 | |
| 
 | |
| 	while (tmp_node != NULL) {
 | |
| 		if (tmp_node->content)
 | |
| 			free(tmp_node->content);
 | |
| 		tmp_node = tmp_node->next;
 | |
| 	}
 | |
| 
 | |
| 	tmp_node = revk_blk_list;
 | |
| 	while (tmp_node != NULL) {
 | |
| 		next_node = tmp_node->next;
 | |
| 		free(tmp_node);
 | |
| 		tmp_node = next_node;
 | |
| 	}
 | |
| 
 | |
| 	revk_blk_list = NULL;
 | |
| 	prev_node = NULL;
 | |
| 	first_node = true;
 | |
| }
 | |
| 
 | |
| int check_blknr_for_revoke(long int blknr, int sequence_no)
 | |
| {
 | |
| 	struct journal_revoke_header_t *header;
 | |
| 	int offset;
 | |
| 	int max;
 | |
| 	long int blocknr;
 | |
| 	char *revk_blk;
 | |
| 	struct revoke_blk_list *tmp_revk_node = revk_blk_list;
 | |
| 	while (tmp_revk_node != NULL) {
 | |
| 		revk_blk = tmp_revk_node->content;
 | |
| 
 | |
| 		header = (struct journal_revoke_header_t *) revk_blk;
 | |
| 		if (sequence_no < be32_to_cpu(header->r_header.h_sequence)) {
 | |
| 			offset = sizeof(struct journal_revoke_header_t);
 | |
| 			max = be32_to_cpu(header->r_count);
 | |
| 
 | |
| 			while (offset < max) {
 | |
| 				blocknr = be32_to_cpu(*((__be32 *)
 | |
| 						  (revk_blk + offset)));
 | |
| 				if (blocknr == blknr)
 | |
| 					goto found;
 | |
| 				offset += 4;
 | |
| 			}
 | |
| 		}
 | |
| 		tmp_revk_node = tmp_revk_node->next;
 | |
| 	}
 | |
| 
 | |
| 	return -1;
 | |
| 
 | |
| found:
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * This function parses the journal blocks and replays the
 | |
|  * suceessful transactions. A transaction is successfull
 | |
|  * if commit block is found for a descriptor block
 | |
|  * The tags in descriptor block contain the disk block
 | |
|  * numbers of the metadata  to be replayed
 | |
|  */
 | |
| void recover_transaction(int prev_desc_logical_no)
 | |
| {
 | |
| 	struct ext2_inode inode_journal;
 | |
| 	struct ext_filesystem *fs = get_fs();
 | |
| 	struct journal_header_t *jdb;
 | |
| 	long int blknr;
 | |
| 	char *p_jdb;
 | |
| 	int ofs, flags;
 | |
| 	int i;
 | |
| 	struct ext3_journal_block_tag *tag;
 | |
| 	char *temp_buff = zalloc(fs->blksz);
 | |
| 	char *metadata_buff = zalloc(fs->blksz);
 | |
| 	if (!temp_buff || !metadata_buff)
 | |
| 		goto fail;
 | |
| 	i = prev_desc_logical_no;
 | |
| 	ext4fs_read_inode(ext4fs_root, EXT2_JOURNAL_INO,
 | |
| 			  (struct ext2_inode *)&inode_journal);
 | |
| 	blknr = read_allocated_block((struct ext2_inode *)
 | |
| 				     &inode_journal, i);
 | |
| 	ext4fs_devread((lbaint_t)blknr * fs->sect_perblk, 0, fs->blksz,
 | |
| 		       temp_buff);
 | |
| 	p_jdb = (char *)temp_buff;
 | |
| 	jdb = (struct journal_header_t *) temp_buff;
 | |
| 	ofs = sizeof(struct journal_header_t);
 | |
| 
 | |
| 	do {
 | |
| 		tag = (struct ext3_journal_block_tag *)&p_jdb[ofs];
 | |
| 		ofs += sizeof(struct ext3_journal_block_tag);
 | |
| 
 | |
| 		if (ofs > fs->blksz)
 | |
| 			break;
 | |
| 
 | |
| 		flags = be32_to_cpu(tag->flags);
 | |
| 		if (!(flags & EXT3_JOURNAL_FLAG_SAME_UUID))
 | |
| 			ofs += 16;
 | |
| 
 | |
| 		i++;
 | |
| 		debug("\t\ttag %u\n", be32_to_cpu(tag->block));
 | |
| 		if (revk_blk_list != NULL) {
 | |
| 			if (check_blknr_for_revoke(be32_to_cpu(tag->block),
 | |
| 				be32_to_cpu(jdb->h_sequence)) == 0)
 | |
| 				continue;
 | |
| 		}
 | |
| 		blknr = read_allocated_block(&inode_journal, i);
 | |
| 		ext4fs_devread((lbaint_t)blknr * fs->sect_perblk, 0,
 | |
| 			       fs->blksz, metadata_buff);
 | |
| 		put_ext4((uint64_t)((uint64_t)be32_to_cpu(tag->block) * (uint64_t)fs->blksz),
 | |
| 			 metadata_buff, (uint32_t) fs->blksz);
 | |
| 	} while (!(flags & EXT3_JOURNAL_FLAG_LAST_TAG));
 | |
| fail:
 | |
| 	free(temp_buff);
 | |
| 	free(metadata_buff);
 | |
| }
 | |
| 
 | |
| void print_jrnl_status(int recovery_flag)
 | |
| {
 | |
| 	if (recovery_flag == RECOVER)
 | |
| 		printf("Journal Recovery Completed\n");
 | |
| 	else
 | |
| 		printf("Journal Scan Completed\n");
 | |
| }
 | |
| 
 | |
| int ext4fs_check_journal_state(int recovery_flag)
 | |
| {
 | |
| 	int i;
 | |
| 	int DB_FOUND = NO;
 | |
| 	long int blknr;
 | |
| 	int transaction_state = TRANSACTION_COMPLETE;
 | |
| 	int prev_desc_logical_no = 0;
 | |
| 	int curr_desc_logical_no = 0;
 | |
| 	int ofs, flags;
 | |
| 	struct ext2_inode inode_journal;
 | |
| 	struct journal_superblock_t *jsb = NULL;
 | |
| 	struct journal_header_t *jdb = NULL;
 | |
| 	char *p_jdb = NULL;
 | |
| 	struct ext3_journal_block_tag *tag = NULL;
 | |
| 	char *temp_buff = NULL;
 | |
| 	char *temp_buff1 = NULL;
 | |
| 	struct ext_filesystem *fs = get_fs();
 | |
| 
 | |
| 	temp_buff = zalloc(fs->blksz);
 | |
| 	if (!temp_buff)
 | |
| 		return -ENOMEM;
 | |
| 	temp_buff1 = zalloc(fs->blksz);
 | |
| 	if (!temp_buff1) {
 | |
| 		free(temp_buff);
 | |
| 		return -ENOMEM;
 | |
| 	}
 | |
| 
 | |
| 	ext4fs_read_inode(ext4fs_root, EXT2_JOURNAL_INO, &inode_journal);
 | |
| 	blknr = read_allocated_block(&inode_journal, EXT2_JOURNAL_SUPERBLOCK);
 | |
| 	ext4fs_devread((lbaint_t)blknr * fs->sect_perblk, 0, fs->blksz,
 | |
| 		       temp_buff);
 | |
| 	jsb = (struct journal_superblock_t *) temp_buff;
 | |
| 
 | |
| 	if (le32_to_cpu(fs->sb->feature_incompat) & EXT3_FEATURE_INCOMPAT_RECOVER) {
 | |
| 		if (recovery_flag == RECOVER)
 | |
| 			printf("Recovery required\n");
 | |
| 	} else {
 | |
| 		if (recovery_flag == RECOVER)
 | |
| 			printf("File System is consistent\n");
 | |
| 		goto end;
 | |
| 	}
 | |
| 
 | |
| 	if (be32_to_cpu(jsb->s_start) == 0)
 | |
| 		goto end;
 | |
| 
 | |
| 	if (!(jsb->s_feature_compat &
 | |
| 				cpu_to_be32(JBD2_FEATURE_COMPAT_CHECKSUM)))
 | |
| 		jsb->s_feature_compat |=
 | |
| 				cpu_to_be32(JBD2_FEATURE_COMPAT_CHECKSUM);
 | |
| 
 | |
| 	i = be32_to_cpu(jsb->s_first);
 | |
| 	while (1) {
 | |
| 		blknr = read_allocated_block(&inode_journal, i);
 | |
| 		memset(temp_buff1, '\0', fs->blksz);
 | |
| 		ext4fs_devread((lbaint_t)blknr * fs->sect_perblk,
 | |
| 			       0, fs->blksz, temp_buff1);
 | |
| 		jdb = (struct journal_header_t *) temp_buff1;
 | |
| 
 | |
| 		if (be32_to_cpu(jdb->h_blocktype) ==
 | |
| 		    EXT3_JOURNAL_DESCRIPTOR_BLOCK) {
 | |
| 			if (be32_to_cpu(jdb->h_sequence) !=
 | |
| 			    be32_to_cpu(jsb->s_sequence)) {
 | |
| 				print_jrnl_status(recovery_flag);
 | |
| 				break;
 | |
| 			}
 | |
| 
 | |
| 			curr_desc_logical_no = i;
 | |
| 			if (transaction_state == TRANSACTION_COMPLETE)
 | |
| 				transaction_state = TRANSACTION_RUNNING;
 | |
| 			else
 | |
| 				return -1;
 | |
| 			p_jdb = (char *)temp_buff1;
 | |
| 			ofs = sizeof(struct journal_header_t);
 | |
| 			do {
 | |
| 				tag = (struct ext3_journal_block_tag *)
 | |
| 				    &p_jdb[ofs];
 | |
| 				ofs += sizeof(struct ext3_journal_block_tag);
 | |
| 				if (ofs > fs->blksz)
 | |
| 					break;
 | |
| 				flags = be32_to_cpu(tag->flags);
 | |
| 				if (!(flags & EXT3_JOURNAL_FLAG_SAME_UUID))
 | |
| 					ofs += 16;
 | |
| 				i++;
 | |
| 				debug("\t\ttag %u\n", be32_to_cpu(tag->block));
 | |
| 			} while (!(flags & EXT3_JOURNAL_FLAG_LAST_TAG));
 | |
| 			i++;
 | |
| 			DB_FOUND = YES;
 | |
| 		} else if (be32_to_cpu(jdb->h_blocktype) ==
 | |
| 				EXT3_JOURNAL_COMMIT_BLOCK) {
 | |
| 			if (be32_to_cpu(jdb->h_sequence) !=
 | |
| 			     be32_to_cpu(jsb->s_sequence)) {
 | |
| 				print_jrnl_status(recovery_flag);
 | |
| 				break;
 | |
| 			}
 | |
| 
 | |
| 			if (transaction_state == TRANSACTION_RUNNING ||
 | |
| 					(DB_FOUND == NO)) {
 | |
| 				transaction_state = TRANSACTION_COMPLETE;
 | |
| 				i++;
 | |
| 				jsb->s_sequence =
 | |
| 					cpu_to_be32(be32_to_cpu(
 | |
| 						jsb->s_sequence) + 1);
 | |
| 			}
 | |
| 			prev_desc_logical_no = curr_desc_logical_no;
 | |
| 			if ((recovery_flag == RECOVER) && (DB_FOUND == YES))
 | |
| 				recover_transaction(prev_desc_logical_no);
 | |
| 
 | |
| 			DB_FOUND = NO;
 | |
| 		} else if (be32_to_cpu(jdb->h_blocktype) ==
 | |
| 				EXT3_JOURNAL_REVOKE_BLOCK) {
 | |
| 			if (be32_to_cpu(jdb->h_sequence) !=
 | |
| 			    be32_to_cpu(jsb->s_sequence)) {
 | |
| 				print_jrnl_status(recovery_flag);
 | |
| 				break;
 | |
| 			}
 | |
| 			if (recovery_flag == SCAN)
 | |
| 				ext4fs_push_revoke_blk((char *)jdb);
 | |
| 			i++;
 | |
| 		} else {
 | |
| 			debug("Else Case\n");
 | |
| 			if (be32_to_cpu(jdb->h_sequence) !=
 | |
| 			    be32_to_cpu(jsb->s_sequence)) {
 | |
| 				print_jrnl_status(recovery_flag);
 | |
| 				break;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| end:
 | |
| 	if (recovery_flag == RECOVER) {
 | |
| 		uint32_t new_feature_incompat;
 | |
| 		jsb->s_start = cpu_to_be32(1);
 | |
| 		jsb->s_sequence = cpu_to_be32(be32_to_cpu(jsb->s_sequence) + 1);
 | |
| 		/* get the superblock */
 | |
| 		ext4_read_superblock((char *)fs->sb);
 | |
| 		new_feature_incompat = le32_to_cpu(fs->sb->feature_incompat);
 | |
| 		new_feature_incompat |= EXT3_FEATURE_INCOMPAT_RECOVER;
 | |
| 		fs->sb->feature_incompat = cpu_to_le32(new_feature_incompat);
 | |
| 
 | |
| 		/* Update the super block */
 | |
| 		put_ext4((uint64_t) (SUPERBLOCK_SIZE),
 | |
| 			 (struct ext2_sblock *)fs->sb,
 | |
| 			 (uint32_t) SUPERBLOCK_SIZE);
 | |
| 		ext4_read_superblock((char *)fs->sb);
 | |
| 
 | |
| 		blknr = read_allocated_block(&inode_journal,
 | |
| 					 EXT2_JOURNAL_SUPERBLOCK);
 | |
| 		put_ext4((uint64_t) ((uint64_t)blknr * (uint64_t)fs->blksz),
 | |
| 			 (struct journal_superblock_t *)temp_buff,
 | |
| 			 (uint32_t) fs->blksz);
 | |
| 		ext4fs_free_revoke_blks();
 | |
| 	}
 | |
| 	free(temp_buff);
 | |
| 	free(temp_buff1);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static void update_descriptor_block(long int blknr)
 | |
| {
 | |
| 	int i;
 | |
| 	long int jsb_blknr;
 | |
| 	struct journal_header_t jdb;
 | |
| 	struct ext3_journal_block_tag tag;
 | |
| 	struct ext2_inode inode_journal;
 | |
| 	struct journal_superblock_t *jsb = NULL;
 | |
| 	char *buf = NULL;
 | |
| 	char *temp = NULL;
 | |
| 	struct ext_filesystem *fs = get_fs();
 | |
| 	char *temp_buff = zalloc(fs->blksz);
 | |
| 	if (!temp_buff)
 | |
| 		return;
 | |
| 
 | |
| 	ext4fs_read_inode(ext4fs_root, EXT2_JOURNAL_INO, &inode_journal);
 | |
| 	jsb_blknr = read_allocated_block(&inode_journal,
 | |
| 					 EXT2_JOURNAL_SUPERBLOCK);
 | |
| 	ext4fs_devread((lbaint_t)jsb_blknr * fs->sect_perblk, 0, fs->blksz,
 | |
| 		       temp_buff);
 | |
| 	jsb = (struct journal_superblock_t *) temp_buff;
 | |
| 
 | |
| 	jdb.h_blocktype = cpu_to_be32(EXT3_JOURNAL_DESCRIPTOR_BLOCK);
 | |
| 	jdb.h_magic = cpu_to_be32(EXT3_JOURNAL_MAGIC_NUMBER);
 | |
| 	jdb.h_sequence = jsb->s_sequence;
 | |
| 	buf = zalloc(fs->blksz);
 | |
| 	if (!buf) {
 | |
| 		free(temp_buff);
 | |
| 		return;
 | |
| 	}
 | |
| 	temp = buf;
 | |
| 	memcpy(buf, &jdb, sizeof(struct journal_header_t));
 | |
| 	temp += sizeof(struct journal_header_t);
 | |
| 
 | |
| 	for (i = 0; i < MAX_JOURNAL_ENTRIES; i++) {
 | |
| 		if (journal_ptr[i]->blknr == -1)
 | |
| 			break;
 | |
| 
 | |
| 		tag.block = cpu_to_be32(journal_ptr[i]->blknr);
 | |
| 		tag.flags = cpu_to_be32(EXT3_JOURNAL_FLAG_SAME_UUID);
 | |
| 		memcpy(temp, &tag, sizeof(struct ext3_journal_block_tag));
 | |
| 		temp = temp + sizeof(struct ext3_journal_block_tag);
 | |
| 	}
 | |
| 
 | |
| 	tag.block = cpu_to_be32(journal_ptr[--i]->blknr);
 | |
| 	tag.flags = cpu_to_be32(EXT3_JOURNAL_FLAG_LAST_TAG);
 | |
| 	memcpy(temp - sizeof(struct ext3_journal_block_tag), &tag,
 | |
| 	       sizeof(struct ext3_journal_block_tag));
 | |
| 	put_ext4((uint64_t) ((uint64_t)blknr * (uint64_t)fs->blksz), buf, (uint32_t) fs->blksz);
 | |
| 
 | |
| 	free(temp_buff);
 | |
| 	free(buf);
 | |
| }
 | |
| 
 | |
| static void update_commit_block(long int blknr)
 | |
| {
 | |
| 	struct journal_header_t jdb;
 | |
| 	struct ext_filesystem *fs = get_fs();
 | |
| 	char *buf = NULL;
 | |
| 	struct ext2_inode inode_journal;
 | |
| 	struct journal_superblock_t *jsb;
 | |
| 	long int jsb_blknr;
 | |
| 	char *temp_buff = zalloc(fs->blksz);
 | |
| 	if (!temp_buff)
 | |
| 		return;
 | |
| 
 | |
| 	ext4fs_read_inode(ext4fs_root, EXT2_JOURNAL_INO,
 | |
| 			  &inode_journal);
 | |
| 	jsb_blknr = read_allocated_block(&inode_journal,
 | |
| 					 EXT2_JOURNAL_SUPERBLOCK);
 | |
| 	ext4fs_devread((lbaint_t)jsb_blknr * fs->sect_perblk, 0, fs->blksz,
 | |
| 		       temp_buff);
 | |
| 	jsb = (struct journal_superblock_t *) temp_buff;
 | |
| 
 | |
| 	jdb.h_blocktype = cpu_to_be32(EXT3_JOURNAL_COMMIT_BLOCK);
 | |
| 	jdb.h_magic = cpu_to_be32(EXT3_JOURNAL_MAGIC_NUMBER);
 | |
| 	jdb.h_sequence = jsb->s_sequence;
 | |
| 	buf = zalloc(fs->blksz);
 | |
| 	if (!buf) {
 | |
| 		free(temp_buff);
 | |
| 		return;
 | |
| 	}
 | |
| 	memcpy(buf, &jdb, sizeof(struct journal_header_t));
 | |
| 	put_ext4((uint64_t) ((uint64_t)blknr * (uint64_t)fs->blksz), buf, (uint32_t) fs->blksz);
 | |
| 
 | |
| 	free(temp_buff);
 | |
| 	free(buf);
 | |
| }
 | |
| 
 | |
| void ext4fs_update_journal(void)
 | |
| {
 | |
| 	struct ext2_inode inode_journal;
 | |
| 	struct ext_filesystem *fs = get_fs();
 | |
| 	long int blknr;
 | |
| 	int i;
 | |
| 	ext4fs_read_inode(ext4fs_root, EXT2_JOURNAL_INO, &inode_journal);
 | |
| 	blknr = read_allocated_block(&inode_journal, jrnl_blk_idx++);
 | |
| 	update_descriptor_block(blknr);
 | |
| 	for (i = 0; i < MAX_JOURNAL_ENTRIES; i++) {
 | |
| 		if (journal_ptr[i]->blknr == -1)
 | |
| 			break;
 | |
| 		blknr = read_allocated_block(&inode_journal, jrnl_blk_idx++);
 | |
| 		put_ext4((uint64_t) ((uint64_t)blknr * (uint64_t)fs->blksz),
 | |
| 			 journal_ptr[i]->buf, fs->blksz);
 | |
| 	}
 | |
| 	blknr = read_allocated_block(&inode_journal, jrnl_blk_idx++);
 | |
| 	update_commit_block(blknr);
 | |
| 	printf("update journal finished\n");
 | |
| }
 |