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