xref: /aosp_15_r20/external/e2fsprogs/lib/ext2fs/bmove.c (revision 6a54128f25917bfc36a8a6e9d722c04a0b4641b6)
1*6a54128fSAndroid Build Coastguard Worker /*
2*6a54128fSAndroid Build Coastguard Worker  * bmove.c --- Move blocks around to make way for a particular
3*6a54128fSAndroid Build Coastguard Worker  * 	filesystem structure.
4*6a54128fSAndroid Build Coastguard Worker  *
5*6a54128fSAndroid Build Coastguard Worker  * Copyright (C) 1997 Theodore Ts'o.
6*6a54128fSAndroid Build Coastguard Worker  *
7*6a54128fSAndroid Build Coastguard Worker  * %Begin-Header%
8*6a54128fSAndroid Build Coastguard Worker  * This file may be redistributed under the terms of the GNU Library
9*6a54128fSAndroid Build Coastguard Worker  * General Public License, version 2.
10*6a54128fSAndroid Build Coastguard Worker  * %End-Header%
11*6a54128fSAndroid Build Coastguard Worker  */
12*6a54128fSAndroid Build Coastguard Worker 
13*6a54128fSAndroid Build Coastguard Worker #include "config.h"
14*6a54128fSAndroid Build Coastguard Worker #include <stdio.h>
15*6a54128fSAndroid Build Coastguard Worker #include <string.h>
16*6a54128fSAndroid Build Coastguard Worker #if HAVE_UNISTD_H
17*6a54128fSAndroid Build Coastguard Worker #include <unistd.h>
18*6a54128fSAndroid Build Coastguard Worker #endif
19*6a54128fSAndroid Build Coastguard Worker #if HAVE_SYS_TYPES_H
20*6a54128fSAndroid Build Coastguard Worker #include <sys/types.h>
21*6a54128fSAndroid Build Coastguard Worker #endif
22*6a54128fSAndroid Build Coastguard Worker #if HAVE_SYS_TIME_H
23*6a54128fSAndroid Build Coastguard Worker #include <sys/time.h>
24*6a54128fSAndroid Build Coastguard Worker #endif
25*6a54128fSAndroid Build Coastguard Worker 
26*6a54128fSAndroid Build Coastguard Worker #include "ext2_fs.h"
27*6a54128fSAndroid Build Coastguard Worker #include "ext2fsP.h"
28*6a54128fSAndroid Build Coastguard Worker 
29*6a54128fSAndroid Build Coastguard Worker struct process_block_struct {
30*6a54128fSAndroid Build Coastguard Worker 	ext2_ino_t		ino;
31*6a54128fSAndroid Build Coastguard Worker 	struct ext2_inode *	inode;
32*6a54128fSAndroid Build Coastguard Worker 	ext2fs_block_bitmap	reserve;
33*6a54128fSAndroid Build Coastguard Worker 	ext2fs_block_bitmap	alloc_map;
34*6a54128fSAndroid Build Coastguard Worker 	errcode_t		error;
35*6a54128fSAndroid Build Coastguard Worker 	char			*buf;
36*6a54128fSAndroid Build Coastguard Worker 	int			add_dir;
37*6a54128fSAndroid Build Coastguard Worker 	int			flags;
38*6a54128fSAndroid Build Coastguard Worker };
39*6a54128fSAndroid Build Coastguard Worker 
process_block(ext2_filsys fs,blk64_t * block_nr,e2_blkcnt_t blockcnt,blk64_t ref_block,int ref_offset,void * priv_data)40*6a54128fSAndroid Build Coastguard Worker static int process_block(ext2_filsys fs, blk64_t *block_nr,
41*6a54128fSAndroid Build Coastguard Worker 			 e2_blkcnt_t blockcnt, blk64_t ref_block,
42*6a54128fSAndroid Build Coastguard Worker 			 int ref_offset, void *priv_data)
43*6a54128fSAndroid Build Coastguard Worker {
44*6a54128fSAndroid Build Coastguard Worker 	struct process_block_struct *pb;
45*6a54128fSAndroid Build Coastguard Worker 	errcode_t	retval;
46*6a54128fSAndroid Build Coastguard Worker 	int		ret;
47*6a54128fSAndroid Build Coastguard Worker 	blk64_t		block, orig;
48*6a54128fSAndroid Build Coastguard Worker 
49*6a54128fSAndroid Build Coastguard Worker 	pb = (struct process_block_struct *) priv_data;
50*6a54128fSAndroid Build Coastguard Worker 	block = orig = *block_nr;
51*6a54128fSAndroid Build Coastguard Worker 	ret = 0;
52*6a54128fSAndroid Build Coastguard Worker 
53*6a54128fSAndroid Build Coastguard Worker 	/*
54*6a54128fSAndroid Build Coastguard Worker 	 * Let's see if this is one which we need to relocate
55*6a54128fSAndroid Build Coastguard Worker 	 */
56*6a54128fSAndroid Build Coastguard Worker 	if (ext2fs_test_block_bitmap2(pb->reserve, block)) {
57*6a54128fSAndroid Build Coastguard Worker 		do {
58*6a54128fSAndroid Build Coastguard Worker 			if (++block >= ext2fs_blocks_count(fs->super))
59*6a54128fSAndroid Build Coastguard Worker 				block = fs->super->s_first_data_block;
60*6a54128fSAndroid Build Coastguard Worker 			if (block == orig) {
61*6a54128fSAndroid Build Coastguard Worker 				pb->error = EXT2_ET_BLOCK_ALLOC_FAIL;
62*6a54128fSAndroid Build Coastguard Worker 				return BLOCK_ABORT;
63*6a54128fSAndroid Build Coastguard Worker 			}
64*6a54128fSAndroid Build Coastguard Worker 		} while (ext2fs_test_block_bitmap2(pb->reserve, block) ||
65*6a54128fSAndroid Build Coastguard Worker 			 ext2fs_test_block_bitmap2(pb->alloc_map, block));
66*6a54128fSAndroid Build Coastguard Worker 
67*6a54128fSAndroid Build Coastguard Worker 		retval = io_channel_read_blk64(fs->io, orig, 1, pb->buf);
68*6a54128fSAndroid Build Coastguard Worker 		if (retval) {
69*6a54128fSAndroid Build Coastguard Worker 			pb->error = retval;
70*6a54128fSAndroid Build Coastguard Worker 			return BLOCK_ABORT;
71*6a54128fSAndroid Build Coastguard Worker 		}
72*6a54128fSAndroid Build Coastguard Worker 		retval = io_channel_write_blk64(fs->io, block, 1, pb->buf);
73*6a54128fSAndroid Build Coastguard Worker 		if (retval) {
74*6a54128fSAndroid Build Coastguard Worker 			pb->error = retval;
75*6a54128fSAndroid Build Coastguard Worker 			return BLOCK_ABORT;
76*6a54128fSAndroid Build Coastguard Worker 		}
77*6a54128fSAndroid Build Coastguard Worker 		*block_nr = block;
78*6a54128fSAndroid Build Coastguard Worker 		ext2fs_mark_block_bitmap2(pb->alloc_map, block);
79*6a54128fSAndroid Build Coastguard Worker 		ret = BLOCK_CHANGED;
80*6a54128fSAndroid Build Coastguard Worker 		if (pb->flags & EXT2_BMOVE_DEBUG)
81*6a54128fSAndroid Build Coastguard Worker 			printf("ino=%u, blockcnt=%lld, %llu->%llu\n",
82*6a54128fSAndroid Build Coastguard Worker 			       (unsigned) pb->ino, blockcnt,
83*6a54128fSAndroid Build Coastguard Worker 			       (unsigned long long) orig,
84*6a54128fSAndroid Build Coastguard Worker 			       (unsigned long long) block);
85*6a54128fSAndroid Build Coastguard Worker 	}
86*6a54128fSAndroid Build Coastguard Worker 	if (pb->add_dir) {
87*6a54128fSAndroid Build Coastguard Worker 		retval = ext2fs_add_dir_block2(fs->dblist, pb->ino,
88*6a54128fSAndroid Build Coastguard Worker 					       block, blockcnt);
89*6a54128fSAndroid Build Coastguard Worker 		if (retval) {
90*6a54128fSAndroid Build Coastguard Worker 			pb->error = retval;
91*6a54128fSAndroid Build Coastguard Worker 			ret |= BLOCK_ABORT;
92*6a54128fSAndroid Build Coastguard Worker 		}
93*6a54128fSAndroid Build Coastguard Worker 	}
94*6a54128fSAndroid Build Coastguard Worker 	return ret;
95*6a54128fSAndroid Build Coastguard Worker }
96*6a54128fSAndroid Build Coastguard Worker 
ext2fs_move_blocks(ext2_filsys fs,ext2fs_block_bitmap reserve,ext2fs_block_bitmap alloc_map,int flags)97*6a54128fSAndroid Build Coastguard Worker errcode_t ext2fs_move_blocks(ext2_filsys fs,
98*6a54128fSAndroid Build Coastguard Worker 			     ext2fs_block_bitmap reserve,
99*6a54128fSAndroid Build Coastguard Worker 			     ext2fs_block_bitmap alloc_map,
100*6a54128fSAndroid Build Coastguard Worker 			     int flags)
101*6a54128fSAndroid Build Coastguard Worker {
102*6a54128fSAndroid Build Coastguard Worker 	ext2_ino_t	ino;
103*6a54128fSAndroid Build Coastguard Worker 	struct ext2_inode inode;
104*6a54128fSAndroid Build Coastguard Worker 	errcode_t	retval;
105*6a54128fSAndroid Build Coastguard Worker 	struct process_block_struct pb;
106*6a54128fSAndroid Build Coastguard Worker 	ext2_inode_scan	scan;
107*6a54128fSAndroid Build Coastguard Worker 	char		*block_buf;
108*6a54128fSAndroid Build Coastguard Worker 
109*6a54128fSAndroid Build Coastguard Worker 	retval = ext2fs_open_inode_scan(fs, 0, &scan);
110*6a54128fSAndroid Build Coastguard Worker 	if (retval)
111*6a54128fSAndroid Build Coastguard Worker 		return retval;
112*6a54128fSAndroid Build Coastguard Worker 
113*6a54128fSAndroid Build Coastguard Worker 	pb.reserve = reserve;
114*6a54128fSAndroid Build Coastguard Worker 	pb.error = 0;
115*6a54128fSAndroid Build Coastguard Worker 	pb.alloc_map = alloc_map ? alloc_map : fs->block_map;
116*6a54128fSAndroid Build Coastguard Worker 	pb.flags = flags;
117*6a54128fSAndroid Build Coastguard Worker 
118*6a54128fSAndroid Build Coastguard Worker 	retval = ext2fs_get_array(4, fs->blocksize, &block_buf);
119*6a54128fSAndroid Build Coastguard Worker 	if (retval)
120*6a54128fSAndroid Build Coastguard Worker 		return retval;
121*6a54128fSAndroid Build Coastguard Worker 	pb.buf = block_buf + fs->blocksize * 3;
122*6a54128fSAndroid Build Coastguard Worker 
123*6a54128fSAndroid Build Coastguard Worker 	/*
124*6a54128fSAndroid Build Coastguard Worker 	 * If GET_DBLIST is set in the flags field, then we should
125*6a54128fSAndroid Build Coastguard Worker 	 * gather directory block information while we're doing the
126*6a54128fSAndroid Build Coastguard Worker 	 * block move.
127*6a54128fSAndroid Build Coastguard Worker 	 */
128*6a54128fSAndroid Build Coastguard Worker 	if (flags & EXT2_BMOVE_GET_DBLIST) {
129*6a54128fSAndroid Build Coastguard Worker 		if (fs->dblist) {
130*6a54128fSAndroid Build Coastguard Worker 			ext2fs_free_dblist(fs->dblist);
131*6a54128fSAndroid Build Coastguard Worker 			fs->dblist = NULL;
132*6a54128fSAndroid Build Coastguard Worker 		}
133*6a54128fSAndroid Build Coastguard Worker 		retval = ext2fs_init_dblist(fs, 0);
134*6a54128fSAndroid Build Coastguard Worker 		if (retval)
135*6a54128fSAndroid Build Coastguard Worker 			return retval;
136*6a54128fSAndroid Build Coastguard Worker 	}
137*6a54128fSAndroid Build Coastguard Worker 
138*6a54128fSAndroid Build Coastguard Worker 	retval = ext2fs_get_next_inode(scan, &ino, &inode);
139*6a54128fSAndroid Build Coastguard Worker 	if (retval)
140*6a54128fSAndroid Build Coastguard Worker 		return retval;
141*6a54128fSAndroid Build Coastguard Worker 
142*6a54128fSAndroid Build Coastguard Worker 	while (ino) {
143*6a54128fSAndroid Build Coastguard Worker 		if ((inode.i_links_count == 0) ||
144*6a54128fSAndroid Build Coastguard Worker 		    !ext2fs_inode_has_valid_blocks2(fs, &inode))
145*6a54128fSAndroid Build Coastguard Worker 			goto next;
146*6a54128fSAndroid Build Coastguard Worker 
147*6a54128fSAndroid Build Coastguard Worker 		pb.ino = ino;
148*6a54128fSAndroid Build Coastguard Worker 		pb.inode = &inode;
149*6a54128fSAndroid Build Coastguard Worker 
150*6a54128fSAndroid Build Coastguard Worker 		pb.add_dir = (LINUX_S_ISDIR(inode.i_mode) &&
151*6a54128fSAndroid Build Coastguard Worker 			      flags & EXT2_BMOVE_GET_DBLIST);
152*6a54128fSAndroid Build Coastguard Worker 
153*6a54128fSAndroid Build Coastguard Worker 		retval = ext2fs_block_iterate3(fs, ino, 0, block_buf,
154*6a54128fSAndroid Build Coastguard Worker 					       process_block, &pb);
155*6a54128fSAndroid Build Coastguard Worker 		if (retval)
156*6a54128fSAndroid Build Coastguard Worker 			return retval;
157*6a54128fSAndroid Build Coastguard Worker 		if (pb.error)
158*6a54128fSAndroid Build Coastguard Worker 			return pb.error;
159*6a54128fSAndroid Build Coastguard Worker 
160*6a54128fSAndroid Build Coastguard Worker 	next:
161*6a54128fSAndroid Build Coastguard Worker 		retval = ext2fs_get_next_inode(scan, &ino, &inode);
162*6a54128fSAndroid Build Coastguard Worker 		if (retval == EXT2_ET_BAD_BLOCK_IN_INODE_TABLE)
163*6a54128fSAndroid Build Coastguard Worker 			goto next;
164*6a54128fSAndroid Build Coastguard Worker 	}
165*6a54128fSAndroid Build Coastguard Worker 	return 0;
166*6a54128fSAndroid Build Coastguard Worker }
167*6a54128fSAndroid Build Coastguard Worker 
168