1 /*
2 * block.c --- iterate over all blocks in an inode
3 *
4 * Copyright (C) 1993, 1994, 1995, 1996 Theodore Ts'o.
5 *
6 * %Begin-Header%
7 * This file may be redistributed under the terms of the GNU Library
8 * General Public License, version 2.
9 * %End-Header%
10 */
11
12 #include "config.h"
13 #include <stdio.h>
14 #include <string.h>
15 #if HAVE_UNISTD_H
16 #include <unistd.h>
17 #endif
18
19 #include "ext2_fs.h"
20 #include "ext2fs.h"
21
22 struct block_context {
23 ext2_filsys fs;
24 int (*func)(ext2_filsys fs,
25 blk64_t *blocknr,
26 e2_blkcnt_t bcount,
27 blk64_t ref_blk,
28 int ref_offset,
29 void *priv_data);
30 e2_blkcnt_t bcount;
31 int bsize;
32 int flags;
33 errcode_t errcode;
34 char *ind_buf;
35 char *dind_buf;
36 char *tind_buf;
37 void *priv_data;
38 };
39
40 #define check_for_ro_violation_return(ctx, ret) \
41 do { \
42 if (((ctx)->flags & BLOCK_FLAG_READ_ONLY) && \
43 ((ret) & BLOCK_CHANGED)) { \
44 (ctx)->errcode = EXT2_ET_RO_BLOCK_ITERATE; \
45 ret |= BLOCK_ABORT | BLOCK_ERROR; \
46 return ret; \
47 } \
48 } while (0)
49
50 #define check_for_ro_violation_goto(ctx, ret, label) \
51 do { \
52 if (((ctx)->flags & BLOCK_FLAG_READ_ONLY) && \
53 ((ret) & BLOCK_CHANGED)) { \
54 (ctx)->errcode = EXT2_ET_RO_BLOCK_ITERATE; \
55 ret |= BLOCK_ABORT | BLOCK_ERROR; \
56 goto label; \
57 } \
58 } while (0)
59
block_iterate_ind(blk_t * ind_block,blk_t ref_block,int ref_offset,struct block_context * ctx)60 static int block_iterate_ind(blk_t *ind_block, blk_t ref_block,
61 int ref_offset, struct block_context *ctx)
62 {
63 int ret = 0, changed = 0;
64 int i, flags, limit, offset;
65 blk_t *block_nr;
66 blk64_t blk64;
67
68 limit = ctx->fs->blocksize >> 2;
69 if (!(ctx->flags & BLOCK_FLAG_DEPTH_TRAVERSE) &&
70 !(ctx->flags & BLOCK_FLAG_DATA_ONLY)) {
71 blk64 = *ind_block;
72 ret = (*ctx->func)(ctx->fs, &blk64,
73 BLOCK_COUNT_IND, ref_block,
74 ref_offset, ctx->priv_data);
75 *ind_block = blk64;
76 }
77 check_for_ro_violation_return(ctx, ret);
78 if (!*ind_block || (ret & BLOCK_ABORT)) {
79 ctx->bcount += limit;
80 return ret;
81 }
82 if (*ind_block >= ext2fs_blocks_count(ctx->fs->super) ||
83 *ind_block < ctx->fs->super->s_first_data_block) {
84 ctx->errcode = EXT2_ET_BAD_IND_BLOCK;
85 ret |= BLOCK_ERROR;
86 return ret;
87 }
88 ctx->errcode = ext2fs_read_ind_block(ctx->fs, *ind_block,
89 ctx->ind_buf);
90 if (ctx->errcode) {
91 ret |= BLOCK_ERROR;
92 return ret;
93 }
94
95 block_nr = (blk_t *) ctx->ind_buf;
96 offset = 0;
97 if (ctx->flags & BLOCK_FLAG_APPEND) {
98 for (i = 0; i < limit; i++, ctx->bcount++, block_nr++) {
99 blk64 = *block_nr;
100 flags = (*ctx->func)(ctx->fs, &blk64, ctx->bcount,
101 *ind_block, offset,
102 ctx->priv_data);
103 *block_nr = blk64;
104 changed |= flags;
105 if (flags & BLOCK_ABORT) {
106 ret |= BLOCK_ABORT;
107 break;
108 }
109 offset += sizeof(blk_t);
110 }
111 } else {
112 for (i = 0; i < limit; i++, ctx->bcount++, block_nr++) {
113 if (*block_nr == 0)
114 goto skip_sparse;
115 blk64 = *block_nr;
116 flags = (*ctx->func)(ctx->fs, &blk64, ctx->bcount,
117 *ind_block, offset,
118 ctx->priv_data);
119 *block_nr = blk64;
120 changed |= flags;
121 if (flags & BLOCK_ABORT) {
122 ret |= BLOCK_ABORT;
123 break;
124 }
125 skip_sparse:
126 offset += sizeof(blk_t);
127 }
128 }
129 check_for_ro_violation_return(ctx, changed);
130 if (changed & BLOCK_CHANGED) {
131 ctx->errcode = ext2fs_write_ind_block(ctx->fs, *ind_block,
132 ctx->ind_buf);
133 if (ctx->errcode)
134 ret |= BLOCK_ERROR | BLOCK_ABORT;
135 }
136 if ((ctx->flags & BLOCK_FLAG_DEPTH_TRAVERSE) &&
137 !(ctx->flags & BLOCK_FLAG_DATA_ONLY) &&
138 !(ret & BLOCK_ABORT)) {
139 blk64 = *ind_block;
140 ret |= (*ctx->func)(ctx->fs, &blk64,
141 BLOCK_COUNT_IND, ref_block,
142 ref_offset, ctx->priv_data);
143 *ind_block = blk64;
144 }
145 check_for_ro_violation_return(ctx, ret);
146 return ret;
147 }
148
block_iterate_dind(blk_t * dind_block,blk_t ref_block,int ref_offset,struct block_context * ctx)149 static int block_iterate_dind(blk_t *dind_block, blk_t ref_block,
150 int ref_offset, struct block_context *ctx)
151 {
152 int ret = 0, changed = 0;
153 int i, flags, limit, offset;
154 blk_t *block_nr;
155 blk64_t blk64;
156
157 limit = ctx->fs->blocksize >> 2;
158 if (!(ctx->flags & (BLOCK_FLAG_DEPTH_TRAVERSE |
159 BLOCK_FLAG_DATA_ONLY))) {
160 blk64 = *dind_block;
161 ret = (*ctx->func)(ctx->fs, &blk64,
162 BLOCK_COUNT_DIND, ref_block,
163 ref_offset, ctx->priv_data);
164 *dind_block = blk64;
165 }
166 check_for_ro_violation_return(ctx, ret);
167 if (!*dind_block || (ret & BLOCK_ABORT)) {
168 ctx->bcount += limit*limit;
169 return ret;
170 }
171 if (*dind_block >= ext2fs_blocks_count(ctx->fs->super) ||
172 *dind_block < ctx->fs->super->s_first_data_block) {
173 ctx->errcode = EXT2_ET_BAD_DIND_BLOCK;
174 ret |= BLOCK_ERROR;
175 return ret;
176 }
177 ctx->errcode = ext2fs_read_ind_block(ctx->fs, *dind_block,
178 ctx->dind_buf);
179 if (ctx->errcode) {
180 ret |= BLOCK_ERROR;
181 return ret;
182 }
183
184 block_nr = (blk_t *) ctx->dind_buf;
185 offset = 0;
186 if (ctx->flags & BLOCK_FLAG_APPEND) {
187 for (i = 0; i < limit; i++, block_nr++) {
188 flags = block_iterate_ind(block_nr,
189 *dind_block, offset,
190 ctx);
191 changed |= flags;
192 if (flags & (BLOCK_ABORT | BLOCK_ERROR)) {
193 ret |= flags & (BLOCK_ABORT | BLOCK_ERROR);
194 break;
195 }
196 offset += sizeof(blk_t);
197 }
198 } else {
199 for (i = 0; i < limit; i++, block_nr++) {
200 if (*block_nr == 0) {
201 ctx->bcount += limit;
202 continue;
203 }
204 flags = block_iterate_ind(block_nr,
205 *dind_block, offset,
206 ctx);
207 changed |= flags;
208 if (flags & (BLOCK_ABORT | BLOCK_ERROR)) {
209 ret |= flags & (BLOCK_ABORT | BLOCK_ERROR);
210 break;
211 }
212 offset += sizeof(blk_t);
213 }
214 }
215 check_for_ro_violation_return(ctx, changed);
216 if (changed & BLOCK_CHANGED) {
217 ctx->errcode = ext2fs_write_ind_block(ctx->fs, *dind_block,
218 ctx->dind_buf);
219 if (ctx->errcode)
220 ret |= BLOCK_ERROR | BLOCK_ABORT;
221 }
222 if ((ctx->flags & BLOCK_FLAG_DEPTH_TRAVERSE) &&
223 !(ctx->flags & BLOCK_FLAG_DATA_ONLY) &&
224 !(ret & BLOCK_ABORT)) {
225 blk64 = *dind_block;
226 ret |= (*ctx->func)(ctx->fs, &blk64,
227 BLOCK_COUNT_DIND, ref_block,
228 ref_offset, ctx->priv_data);
229 *dind_block = blk64;
230 }
231 check_for_ro_violation_return(ctx, ret);
232 return ret;
233 }
234
block_iterate_tind(blk_t * tind_block,blk_t ref_block,int ref_offset,struct block_context * ctx)235 static int block_iterate_tind(blk_t *tind_block, blk_t ref_block,
236 int ref_offset, struct block_context *ctx)
237 {
238 int ret = 0, changed = 0;
239 int i, flags, limit, offset;
240 blk_t *block_nr;
241 blk64_t blk64;
242
243 limit = ctx->fs->blocksize >> 2;
244 if (!(ctx->flags & (BLOCK_FLAG_DEPTH_TRAVERSE |
245 BLOCK_FLAG_DATA_ONLY))) {
246 blk64 = *tind_block;
247 ret = (*ctx->func)(ctx->fs, &blk64,
248 BLOCK_COUNT_TIND, ref_block,
249 ref_offset, ctx->priv_data);
250 *tind_block = blk64;
251 }
252 check_for_ro_violation_return(ctx, ret);
253 if (!*tind_block || (ret & BLOCK_ABORT)) {
254 ctx->bcount += limit*limit*limit;
255 return ret;
256 }
257 if (*tind_block >= ext2fs_blocks_count(ctx->fs->super) ||
258 *tind_block < ctx->fs->super->s_first_data_block) {
259 ctx->errcode = EXT2_ET_BAD_TIND_BLOCK;
260 ret |= BLOCK_ERROR;
261 return ret;
262 }
263 ctx->errcode = ext2fs_read_ind_block(ctx->fs, *tind_block,
264 ctx->tind_buf);
265 if (ctx->errcode) {
266 ret |= BLOCK_ERROR;
267 return ret;
268 }
269
270 block_nr = (blk_t *) ctx->tind_buf;
271 offset = 0;
272 if (ctx->flags & BLOCK_FLAG_APPEND) {
273 for (i = 0; i < limit; i++, block_nr++) {
274 flags = block_iterate_dind(block_nr,
275 *tind_block,
276 offset, ctx);
277 changed |= flags;
278 if (flags & (BLOCK_ABORT | BLOCK_ERROR)) {
279 ret |= flags & (BLOCK_ABORT | BLOCK_ERROR);
280 break;
281 }
282 offset += sizeof(blk_t);
283 }
284 } else {
285 for (i = 0; i < limit; i++, block_nr++) {
286 if (*block_nr == 0) {
287 ctx->bcount += limit*limit;
288 continue;
289 }
290 flags = block_iterate_dind(block_nr,
291 *tind_block,
292 offset, ctx);
293 changed |= flags;
294 if (flags & (BLOCK_ABORT | BLOCK_ERROR)) {
295 ret |= flags & (BLOCK_ABORT | BLOCK_ERROR);
296 break;
297 }
298 offset += sizeof(blk_t);
299 }
300 }
301 check_for_ro_violation_return(ctx, changed);
302 if (changed & BLOCK_CHANGED) {
303 ctx->errcode = ext2fs_write_ind_block(ctx->fs, *tind_block,
304 ctx->tind_buf);
305 if (ctx->errcode)
306 ret |= BLOCK_ERROR | BLOCK_ABORT;
307 }
308 if ((ctx->flags & BLOCK_FLAG_DEPTH_TRAVERSE) &&
309 !(ctx->flags & BLOCK_FLAG_DATA_ONLY) &&
310 !(ret & BLOCK_ABORT)) {
311 blk64 = *tind_block;
312 ret |= (*ctx->func)(ctx->fs, &blk64,
313 BLOCK_COUNT_TIND, ref_block,
314 ref_offset, ctx->priv_data);
315 *tind_block = blk64;
316 }
317 check_for_ro_violation_return(ctx, ret);
318 return ret;
319 }
320
ext2fs_block_iterate3(ext2_filsys fs,ext2_ino_t ino,int flags,char * block_buf,int (* func)(ext2_filsys fs,blk64_t * blocknr,e2_blkcnt_t blockcnt,blk64_t ref_blk,int ref_offset,void * priv_data),void * priv_data)321 errcode_t ext2fs_block_iterate3(ext2_filsys fs,
322 ext2_ino_t ino,
323 int flags,
324 char *block_buf,
325 int (*func)(ext2_filsys fs,
326 blk64_t *blocknr,
327 e2_blkcnt_t blockcnt,
328 blk64_t ref_blk,
329 int ref_offset,
330 void *priv_data),
331 void *priv_data)
332 {
333 int i;
334 int r, ret = 0;
335 struct ext2_inode inode;
336 errcode_t retval;
337 struct block_context ctx;
338 int limit;
339 blk64_t blk64;
340
341 EXT2_CHECK_MAGIC(fs, EXT2_ET_MAGIC_EXT2FS_FILSYS);
342
343 ctx.errcode = ext2fs_read_inode(fs, ino, &inode);
344 if (ctx.errcode)
345 return ctx.errcode;
346
347 /*
348 * An inode with inline data has no blocks over which to
349 * iterate, so return an error code indicating this fact.
350 */
351 if (inode.i_flags & EXT4_INLINE_DATA_FL)
352 return EXT2_ET_INLINE_DATA_CANT_ITERATE;
353
354 /*
355 * Check to see if we need to limit large files
356 */
357 if (flags & BLOCK_FLAG_NO_LARGE) {
358 if (!LINUX_S_ISDIR(inode.i_mode) &&
359 (inode.i_size_high != 0))
360 return EXT2_ET_FILE_TOO_BIG;
361 }
362
363 limit = fs->blocksize >> 2;
364
365 ctx.fs = fs;
366 ctx.func = func;
367 ctx.priv_data = priv_data;
368 ctx.flags = flags;
369 ctx.bcount = 0;
370 if (block_buf) {
371 ctx.ind_buf = block_buf;
372 } else {
373 retval = ext2fs_get_array(3, fs->blocksize, &ctx.ind_buf);
374 if (retval)
375 return retval;
376 }
377 ctx.dind_buf = ctx.ind_buf + fs->blocksize;
378 ctx.tind_buf = ctx.dind_buf + fs->blocksize;
379
380 /*
381 * Iterate over the HURD translator block (if present)
382 */
383 if ((fs->super->s_creator_os == EXT2_OS_HURD) &&
384 !(flags & BLOCK_FLAG_DATA_ONLY)) {
385 if (inode.osd1.hurd1.h_i_translator) {
386 blk64 = inode.osd1.hurd1.h_i_translator;
387 ret |= (*ctx.func)(fs, &blk64,
388 BLOCK_COUNT_TRANSLATOR,
389 0, 0, priv_data);
390 inode.osd1.hurd1.h_i_translator = (blk_t) blk64;
391 if (ret & BLOCK_ABORT)
392 goto abort_exit;
393 check_for_ro_violation_goto(&ctx, ret, abort_exit);
394 }
395 }
396
397 if (inode.i_flags & EXT4_EXTENTS_FL) {
398 ext2_extent_handle_t handle;
399 struct ext2fs_extent extent, next;
400 e2_blkcnt_t blockcnt = 0;
401 blk64_t blk, new_blk;
402 int op = EXT2_EXTENT_ROOT;
403 int uninit;
404 unsigned int j;
405
406 ctx.errcode = ext2fs_extent_open2(fs, ino, &inode, &handle);
407 if (ctx.errcode)
408 goto abort_exit;
409
410 while (1) {
411 if (op == EXT2_EXTENT_CURRENT)
412 ctx.errcode = 0;
413 else
414 ctx.errcode = ext2fs_extent_get(handle, op,
415 &extent);
416 if (ctx.errcode) {
417 if (ctx.errcode != EXT2_ET_EXTENT_NO_NEXT)
418 break;
419 ctx.errcode = 0;
420 if (!(flags & BLOCK_FLAG_APPEND))
421 break;
422 next_block_set:
423 blk = 0;
424 r = (*ctx.func)(fs, &blk, blockcnt,
425 0, 0, priv_data);
426 ret |= r;
427 check_for_ro_violation_goto(&ctx, ret,
428 extent_done);
429 if (r & BLOCK_CHANGED) {
430 ctx.errcode =
431 ext2fs_extent_set_bmap(handle,
432 (blk64_t) blockcnt++,
433 (blk64_t) blk, 0);
434 if (ctx.errcode || (ret & BLOCK_ABORT))
435 break;
436 if (blk)
437 goto next_block_set;
438 }
439 break;
440 }
441
442 op = EXT2_EXTENT_NEXT;
443 blk = extent.e_pblk;
444 if (!(extent.e_flags & EXT2_EXTENT_FLAGS_LEAF)) {
445 if (ctx.flags & BLOCK_FLAG_DATA_ONLY)
446 continue;
447 if ((!(extent.e_flags &
448 EXT2_EXTENT_FLAGS_SECOND_VISIT) &&
449 !(ctx.flags & BLOCK_FLAG_DEPTH_TRAVERSE)) ||
450 ((extent.e_flags &
451 EXT2_EXTENT_FLAGS_SECOND_VISIT) &&
452 (ctx.flags & BLOCK_FLAG_DEPTH_TRAVERSE))) {
453 ret |= (*ctx.func)(fs, &blk,
454 -1, 0, 0, priv_data);
455 if (ret & BLOCK_CHANGED) {
456 extent.e_pblk = blk;
457 ctx.errcode =
458 ext2fs_extent_replace(handle, 0, &extent);
459 if (ctx.errcode)
460 break;
461 }
462 if (ret & BLOCK_ABORT)
463 break;
464 }
465 continue;
466 }
467 uninit = 0;
468 if (extent.e_flags & EXT2_EXTENT_FLAGS_UNINIT)
469 uninit = EXT2_EXTENT_SET_BMAP_UNINIT;
470
471 /*
472 * Get the next extent before we start messing
473 * with the current extent
474 */
475 retval = ext2fs_extent_get(handle, op, &next);
476
477 #if 0
478 printf("lblk %llu pblk %llu len %d blockcnt %llu\n",
479 extent.e_lblk, extent.e_pblk,
480 extent.e_len, blockcnt);
481 #endif
482 if (extent.e_lblk + extent.e_len <= (blk64_t) blockcnt)
483 continue;
484 if (extent.e_lblk > (blk64_t) blockcnt)
485 blockcnt = extent.e_lblk;
486 j = blockcnt - extent.e_lblk;
487 blk += j;
488 for (blockcnt = extent.e_lblk, j = 0;
489 j < extent.e_len;
490 blk++, blockcnt++, j++) {
491 new_blk = blk;
492 r = (*ctx.func)(fs, &new_blk, blockcnt,
493 0, 0, priv_data);
494 ret |= r;
495 check_for_ro_violation_goto(&ctx, ret,
496 extent_done);
497 if (r & BLOCK_CHANGED) {
498 ctx.errcode =
499 ext2fs_extent_set_bmap(handle,
500 (blk64_t) blockcnt,
501 new_blk, uninit);
502 if (ctx.errcode)
503 goto extent_done;
504 }
505 if (ret & BLOCK_ABORT)
506 goto extent_done;
507 }
508 if (retval == 0) {
509 extent = next;
510 op = EXT2_EXTENT_CURRENT;
511 }
512 }
513
514 extent_done:
515 ext2fs_extent_free(handle);
516 ret |= BLOCK_ERROR; /* ctx.errcode is always valid here */
517 goto errout;
518 }
519
520 /*
521 * Iterate over normal data blocks
522 */
523 for (i = 0; i < EXT2_NDIR_BLOCKS ; i++, ctx.bcount++) {
524 if (inode.i_block[i] || (flags & BLOCK_FLAG_APPEND)) {
525 blk64 = inode.i_block[i];
526 ret |= (*ctx.func)(fs, &blk64, ctx.bcount, 0, i,
527 priv_data);
528 inode.i_block[i] = (blk_t) blk64;
529 if (ret & BLOCK_ABORT)
530 goto abort_exit;
531 }
532 }
533 check_for_ro_violation_goto(&ctx, ret, abort_exit);
534 if (inode.i_block[EXT2_IND_BLOCK] || (flags & BLOCK_FLAG_APPEND)) {
535 ret |= block_iterate_ind(&inode.i_block[EXT2_IND_BLOCK],
536 0, EXT2_IND_BLOCK, &ctx);
537 if (ret & BLOCK_ABORT)
538 goto abort_exit;
539 } else
540 ctx.bcount += limit;
541 if (inode.i_block[EXT2_DIND_BLOCK] || (flags & BLOCK_FLAG_APPEND)) {
542 ret |= block_iterate_dind(&inode.i_block[EXT2_DIND_BLOCK],
543 0, EXT2_DIND_BLOCK, &ctx);
544 if (ret & BLOCK_ABORT)
545 goto abort_exit;
546 } else
547 ctx.bcount += limit * limit;
548 if (inode.i_block[EXT2_TIND_BLOCK] || (flags & BLOCK_FLAG_APPEND)) {
549 ret |= block_iterate_tind(&inode.i_block[EXT2_TIND_BLOCK],
550 0, EXT2_TIND_BLOCK, &ctx);
551 if (ret & BLOCK_ABORT)
552 goto abort_exit;
553 }
554
555 abort_exit:
556 if (ret & BLOCK_CHANGED) {
557 retval = ext2fs_write_inode(fs, ino, &inode);
558 if (retval) {
559 ret |= BLOCK_ERROR;
560 ctx.errcode = retval;
561 }
562 }
563 errout:
564 if (!block_buf)
565 ext2fs_free_mem(&ctx.ind_buf);
566
567 return (ret & BLOCK_ERROR) ? ctx.errcode : 0;
568 }
569
570 /*
571 * Emulate the old ext2fs_block_iterate function!
572 */
573
574 struct xlate64 {
575 int (*func)(ext2_filsys fs,
576 blk_t *blocknr,
577 e2_blkcnt_t blockcnt,
578 blk_t ref_blk,
579 int ref_offset,
580 void *priv_data);
581 void *real_private;
582 };
583
xlate64_func(ext2_filsys fs,blk64_t * blocknr,e2_blkcnt_t blockcnt,blk64_t ref_blk,int ref_offset,void * priv_data)584 static int xlate64_func(ext2_filsys fs, blk64_t *blocknr,
585 e2_blkcnt_t blockcnt, blk64_t ref_blk,
586 int ref_offset, void *priv_data)
587 {
588 struct xlate64 *xl = (struct xlate64 *) priv_data;
589 int ret;
590 blk_t block32 = *blocknr;
591
592 ret = (*xl->func)(fs, &block32, blockcnt, (blk_t) ref_blk, ref_offset,
593 xl->real_private);
594 *blocknr = block32;
595 return ret;
596 }
597
ext2fs_block_iterate2(ext2_filsys fs,ext2_ino_t ino,int flags,char * block_buf,int (* func)(ext2_filsys fs,blk_t * blocknr,e2_blkcnt_t blockcnt,blk_t ref_blk,int ref_offset,void * priv_data),void * priv_data)598 errcode_t ext2fs_block_iterate2(ext2_filsys fs,
599 ext2_ino_t ino,
600 int flags,
601 char *block_buf,
602 int (*func)(ext2_filsys fs,
603 blk_t *blocknr,
604 e2_blkcnt_t blockcnt,
605 blk_t ref_blk,
606 int ref_offset,
607 void *priv_data),
608 void *priv_data)
609 {
610 struct xlate64 xl;
611
612 xl.real_private = priv_data;
613 xl.func = func;
614
615 return ext2fs_block_iterate3(fs, ino, flags, block_buf,
616 xlate64_func, &xl);
617 }
618
619
620 struct xlate {
621 int (*func)(ext2_filsys fs,
622 blk_t *blocknr,
623 int bcount,
624 void *priv_data);
625 void *real_private;
626 };
627
628 #ifdef __TURBOC__
629 #pragma argsused
630 #endif
xlate_func(ext2_filsys fs,blk_t * blocknr,e2_blkcnt_t blockcnt,blk_t ref_block EXT2FS_ATTR ((unused)),int ref_offset EXT2FS_ATTR ((unused)),void * priv_data)631 static int xlate_func(ext2_filsys fs, blk_t *blocknr, e2_blkcnt_t blockcnt,
632 blk_t ref_block EXT2FS_ATTR((unused)),
633 int ref_offset EXT2FS_ATTR((unused)),
634 void *priv_data)
635 {
636 struct xlate *xl = (struct xlate *) priv_data;
637
638 return (*xl->func)(fs, blocknr, (int) blockcnt, xl->real_private);
639 }
640
ext2fs_block_iterate(ext2_filsys fs,ext2_ino_t ino,int flags,char * block_buf,int (* func)(ext2_filsys fs,blk_t * blocknr,int blockcnt,void * priv_data),void * priv_data)641 errcode_t ext2fs_block_iterate(ext2_filsys fs,
642 ext2_ino_t ino,
643 int flags,
644 char *block_buf,
645 int (*func)(ext2_filsys fs,
646 blk_t *blocknr,
647 int blockcnt,
648 void *priv_data),
649 void *priv_data)
650 {
651 struct xlate xl;
652
653 xl.real_private = priv_data;
654 xl.func = func;
655
656 return ext2fs_block_iterate2(fs, ino, BLOCK_FLAG_NO_LARGE | flags,
657 block_buf, xlate_func, &xl);
658 }
659
660