Whamcloud - gitweb
e2fsck: clean up xattr checking code, add test
[tools/e2fsprogs.git] / lib / ext2fs / ext2fs.h
1 /*
2  * ext2fs.h --- ext2fs
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 #ifndef _EXT2FS_EXT2FS_H
13 #define _EXT2FS_EXT2FS_H
14
15 #ifdef __GNUC__
16 #define EXT2FS_ATTR(x) __attribute__(x)
17 #else
18 #define EXT2FS_ATTR(x)
19 #endif
20
21 #ifdef __cplusplus
22 extern "C" {
23 #endif
24
25 /*
26  * Non-GNU C compilers won't necessarily understand inline
27  */
28 #if (!defined(__GNUC__) && !defined(__WATCOMC__))
29 #define NO_INLINE_FUNCS
30 #endif
31
32 /*
33  * Where the master copy of the superblock is located, and how big
34  * superblocks are supposed to be.  We define SUPERBLOCK_SIZE because
35  * the size of the superblock structure is not necessarily trustworthy
36  * (some versions have the padding set up so that the superblock is
37  * 1032 bytes long).
38  */
39 #define SUPERBLOCK_OFFSET       1024
40 #define SUPERBLOCK_SIZE         1024
41
42 #define UUID_STR_SIZE 37
43
44 /*
45  * The last ext2fs revision level that this version of the library is
46  * able to support.
47  */
48 #define EXT2_LIB_CURRENT_REV    EXT2_DYNAMIC_REV
49
50 #ifdef HAVE_SYS_TYPES_H
51 #include <sys/types.h>
52 #endif
53
54 #include <stdio.h>
55 #include <stdlib.h>
56 #include <string.h>
57 #include <errno.h>
58
59 #if EXT2_FLAT_INCLUDES
60 #include "e2_types.h"
61 #include "ext2_fs.h"
62 #include "ext3_extents.h"
63 #else
64 #include <ext2fs/ext2_types.h>
65 #include <ext2fs/ext2_fs.h>
66 #include <ext2fs/ext3_extents.h>
67 #endif /* EXT2_FLAT_INCLUDES */
68
69 #ifdef __CHECK_ENDIAN__
70 #define __bitwise __attribute__((bitwise))
71 #else
72 #define __bitwise
73 #endif
74
75 typedef __u32 __bitwise         ext2_ino_t;
76 typedef __u32 __bitwise         blk_t;
77 typedef __u64 __bitwise         blk64_t;
78 typedef __u32 __bitwise         dgrp_t;
79 typedef __u32 __bitwise         ext2_off_t;
80 typedef __u64 __bitwise         ext2_off64_t;
81 typedef __s64 __bitwise         e2_blkcnt_t;
82 typedef __u32 __bitwise         ext2_dirhash_t;
83
84 #if EXT2_FLAT_INCLUDES
85 #include "com_err.h"
86 #include "ext2_io.h"
87 #include "ext2_err.h"
88 #include "ext2_ext_attr.h"
89 #else
90 #include <et/com_err.h>
91 #include <ext2fs/ext2_io.h>
92 #include <ext2fs/ext2_err.h>
93 #include <ext2fs/ext2_ext_attr.h>
94 #endif
95
96 /*
97  * Portability help for Microsoft Visual C++
98  */
99 #ifdef _MSC_VER
100 #define EXT2_QSORT_TYPE int __cdecl
101 #else
102 #define EXT2_QSORT_TYPE int
103 #endif
104
105 typedef struct struct_ext2_filsys *ext2_filsys;
106
107 #define EXT2FS_MARK_ERROR       0
108 #define EXT2FS_UNMARK_ERROR     1
109 #define EXT2FS_TEST_ERROR       2
110
111 typedef struct ext2fs_struct_generic_bitmap *ext2fs_generic_bitmap;
112 typedef struct ext2fs_struct_generic_bitmap *ext2fs_inode_bitmap;
113 typedef struct ext2fs_struct_generic_bitmap *ext2fs_block_bitmap;
114
115 #define EXT2_FIRST_INODE(s)     EXT2_FIRST_INO(s)
116
117
118 /*
119  * Badblocks list definitions
120  */
121
122 typedef struct ext2_struct_u32_list *ext2_badblocks_list;
123 typedef struct ext2_struct_u32_iterate *ext2_badblocks_iterate;
124
125 typedef struct ext2_struct_u32_list *ext2_u32_list;
126 typedef struct ext2_struct_u32_iterate *ext2_u32_iterate;
127
128 /* old */
129 typedef struct ext2_struct_u32_list *badblocks_list;
130 typedef struct ext2_struct_u32_iterate *badblocks_iterate;
131
132 #define BADBLOCKS_FLAG_DIRTY    1
133
134 /*
135  * ext2_dblist structure and abstractions (see dblist.c)
136  */
137 struct ext2_db_entry2 {
138         ext2_ino_t      ino;
139         blk64_t blk;
140         e2_blkcnt_t     blockcnt;
141 };
142
143 /* Ye Olde 32-bit version */
144 struct ext2_db_entry {
145         ext2_ino_t      ino;
146         blk_t   blk;
147         int     blockcnt;
148 };
149
150 typedef struct ext2_struct_dblist *ext2_dblist;
151
152 #define DBLIST_ABORT    1
153
154 /*
155  * ext2_fileio definitions
156  */
157
158 #define EXT2_FILE_WRITE         0x0001
159 #define EXT2_FILE_CREATE        0x0002
160
161 #define EXT2_FILE_MASK          0x00FF
162
163 #define EXT2_FILE_BUF_DIRTY     0x4000
164 #define EXT2_FILE_BUF_VALID     0x2000
165
166 typedef struct ext2_file *ext2_file_t;
167
168 #define EXT2_SEEK_SET   0
169 #define EXT2_SEEK_CUR   1
170 #define EXT2_SEEK_END   2
171
172 /*
173  * Flags for the ext2_filsys structure and for ext2fs_open()
174  */
175 #define EXT2_FLAG_RW                    0x01
176 #define EXT2_FLAG_CHANGED               0x02
177 #define EXT2_FLAG_DIRTY                 0x04
178 #define EXT2_FLAG_VALID                 0x08
179 #define EXT2_FLAG_IB_DIRTY              0x10
180 #define EXT2_FLAG_BB_DIRTY              0x20
181 #define EXT2_FLAG_SWAP_BYTES            0x40
182 #define EXT2_FLAG_SWAP_BYTES_READ       0x80
183 #define EXT2_FLAG_SWAP_BYTES_WRITE      0x100
184 #define EXT2_FLAG_MASTER_SB_ONLY        0x200
185 #define EXT2_FLAG_FORCE                 0x400
186 #define EXT2_FLAG_SUPER_ONLY            0x800
187 #define EXT2_FLAG_JOURNAL_DEV_OK        0x1000
188 #define EXT2_FLAG_IMAGE_FILE            0x2000
189 #define EXT2_FLAG_EXCLUSIVE             0x4000
190 #define EXT2_FLAG_SOFTSUPP_FEATURES     0x8000
191 #define EXT2_FLAG_NOFREE_ON_ERROR       0x10000
192 #define EXT2_FLAG_64BITS                0x20000
193 #define EXT2_FLAG_PRINT_PROGRESS        0x40000
194 #define EXT2_FLAG_DIRECT_IO             0x80000
195 #define EXT2_FLAG_SKIP_MMP              0x100000
196
197 /*
198  * Special flag in the ext2 inode i_flag field that means that this is
199  * a new inode.  (So that ext2_write_inode() can clear extra fields.)
200  */
201 #define EXT2_NEW_INODE_FL       0x80000000
202
203 /*
204  * Flags for mkjournal
205  */
206 #define EXT2_MKJOURNAL_V1_SUPER 0x0000001 /* create V1 superblock (deprecated) */
207 #define EXT2_MKJOURNAL_LAZYINIT 0x0000002 /* don't zero journal inode before use*/
208 #define EXT2_MKJOURNAL_NO_MNT_CHECK 0x0000004 /* don't check mount status */
209
210 struct opaque_ext2_group_desc;
211
212 struct struct_ext2_filsys {
213         errcode_t                       magic;
214         io_channel                      io;
215         int                             flags;
216         char *                          device_name;
217         struct ext2_super_block *       super;
218         unsigned int                    blocksize;
219         int                             fragsize;
220         dgrp_t                          group_desc_count;
221         unsigned long                   desc_blocks;
222         struct opaque_ext2_group_desc * group_desc;
223         unsigned int                    inode_blocks_per_group;
224         ext2fs_inode_bitmap             inode_map;
225         ext2fs_block_bitmap             block_map;
226         /* XXX FIXME-64: not 64-bit safe, but not used? */
227         errcode_t (*get_blocks)(ext2_filsys fs, ext2_ino_t ino, blk_t *blocks);
228         errcode_t (*check_directory)(ext2_filsys fs, ext2_ino_t ino);
229         errcode_t (*write_bitmaps)(ext2_filsys fs);
230         errcode_t (*read_inode)(ext2_filsys fs, ext2_ino_t ino,
231                                 struct ext2_inode *inode);
232         errcode_t (*write_inode)(ext2_filsys fs, ext2_ino_t ino,
233                                 struct ext2_inode *inode);
234         ext2_badblocks_list             badblocks;
235         ext2_dblist                     dblist;
236         __u32                           stride; /* for mke2fs */
237         struct ext2_super_block *       orig_super;
238         struct ext2_image_hdr *         image_header;
239         __u32                           umask;
240         time_t                          now;
241         int                             cluster_ratio_bits;
242         __u16                           default_bitmap_type;
243         __u16                           pad;
244         /*
245          * Reserved for future expansion
246          */
247         __u32                           reserved[5];
248
249         /*
250          * Reserved for the use of the calling application.
251          */
252         void *                          priv_data;
253
254         /*
255          * Inode cache
256          */
257         struct ext2_inode_cache         *icache;
258         io_channel                      image_io;
259
260         /*
261          * More callback functions
262          */
263         errcode_t (*get_alloc_block)(ext2_filsys fs, blk64_t goal,
264                                      blk64_t *ret);
265         void (*block_alloc_stats)(ext2_filsys fs, blk64_t blk, int inuse);
266
267         /*
268          * Buffers for Multiple mount protection(MMP) block.
269          */
270         void *mmp_buf;
271         void *mmp_cmp;
272         int mmp_fd;
273
274         /*
275          * Time at which e2fsck last updated the MMP block.
276          */
277         long mmp_last_written;
278 };
279
280 #if EXT2_FLAT_INCLUDES
281 #include "e2_bitops.h"
282 #else
283 #include <ext2fs/bitops.h>
284 #endif
285
286 /*
287  * 64-bit bitmap backend types
288  */
289 #define EXT2FS_BMAP64_BITARRAY  1
290 #define EXT2FS_BMAP64_RBTREE    2
291 #define EXT2FS_BMAP64_AUTODIR   3
292
293 /*
294  * Return flags for the block iterator functions
295  */
296 #define BLOCK_CHANGED   1
297 #define BLOCK_ABORT     2
298 #define BLOCK_ERROR     4
299
300 /*
301  * Block interate flags
302  *
303  * BLOCK_FLAG_APPEND, or BLOCK_FLAG_HOLE, indicates that the interator
304  * function should be called on blocks where the block number is zero.
305  * This is used by ext2fs_expand_dir() to be able to add a new block
306  * to an inode.  It can also be used for programs that want to be able
307  * to deal with files that contain "holes".
308  *
309  * BLOCK_FLAG_DEPTH_TRAVERSE indicates that the iterator function for
310  * the indirect, doubly indirect, etc. blocks should be called after
311  * all of the blocks containined in the indirect blocks are processed.
312  * This is useful if you are going to be deallocating blocks from an
313  * inode.
314  *
315  * BLOCK_FLAG_DATA_ONLY indicates that the iterator function should be
316  * called for data blocks only.
317  *
318  * BLOCK_FLAG_READ_ONLY is a promise by the caller that it will not
319  * modify returned block number.
320  *
321  * BLOCK_FLAG_NO_LARGE is for internal use only.  It informs
322  * ext2fs_block_iterate2 that large files won't be accepted.
323  */
324 #define BLOCK_FLAG_APPEND       1
325 #define BLOCK_FLAG_HOLE         1
326 #define BLOCK_FLAG_DEPTH_TRAVERSE       2
327 #define BLOCK_FLAG_DATA_ONLY    4
328 #define BLOCK_FLAG_READ_ONLY    8
329
330 #define BLOCK_FLAG_NO_LARGE     0x1000
331
332 /*
333  * Magic "block count" return values for the block iterator function.
334  */
335 #define BLOCK_COUNT_IND         (-1)
336 #define BLOCK_COUNT_DIND        (-2)
337 #define BLOCK_COUNT_TIND        (-3)
338 #define BLOCK_COUNT_TRANSLATOR  (-4)
339
340 #if 0
341 /*
342  * Flags for ext2fs_move_blocks
343  */
344 #define EXT2_BMOVE_GET_DBLIST   0x0001
345 #define EXT2_BMOVE_DEBUG        0x0002
346 #endif
347
348 /*
349  * Generic (non-filesystem layout specific) extents structure
350  */
351
352 #define EXT2_EXTENT_FLAGS_LEAF          0x0001
353 #define EXT2_EXTENT_FLAGS_UNINIT        0x0002
354 #define EXT2_EXTENT_FLAGS_SECOND_VISIT  0x0004
355
356 struct ext2fs_extent {
357         blk64_t e_pblk;         /* first physical block */
358         blk64_t e_lblk;         /* first logical block extent covers */
359         __u32   e_len;          /* number of blocks covered by extent */
360         __u32   e_flags;        /* extent flags */
361 };
362
363 typedef struct ext2_extent_handle *ext2_extent_handle_t;
364 typedef struct ext2_extent_path *ext2_extent_path_t;
365
366 /*
367  * Flags used by ext2fs_extent_get()
368  */
369 #define EXT2_EXTENT_CURRENT     0x0000
370 #define EXT2_EXTENT_MOVE_MASK   0x000F
371 #define EXT2_EXTENT_ROOT        0x0001
372 #define EXT2_EXTENT_LAST_LEAF   0x0002
373 #define EXT2_EXTENT_FIRST_SIB   0x0003
374 #define EXT2_EXTENT_LAST_SIB    0x0004
375 #define EXT2_EXTENT_NEXT_SIB    0x0005
376 #define EXT2_EXTENT_PREV_SIB    0x0006
377 #define EXT2_EXTENT_NEXT_LEAF   0x0007
378 #define EXT2_EXTENT_PREV_LEAF   0x0008
379 #define EXT2_EXTENT_NEXT        0x0009
380 #define EXT2_EXTENT_PREV        0x000A
381 #define EXT2_EXTENT_UP          0x000B
382 #define EXT2_EXTENT_DOWN        0x000C
383 #define EXT2_EXTENT_DOWN_AND_LAST 0x000D
384
385 /*
386  * Flags used by ext2fs_extent_insert()
387  */
388 #define EXT2_EXTENT_INSERT_AFTER        0x0001 /* insert after handle loc'n */
389 #define EXT2_EXTENT_INSERT_NOSPLIT      0x0002 /* insert may not cause split */
390
391 /*
392  * Flags used by ext2fs_extent_delete()
393  */
394 #define EXT2_EXTENT_DELETE_KEEP_EMPTY   0x001 /* keep node if last extnt gone */
395
396 /*
397  * Flags used by ext2fs_extent_set_bmap()
398  */
399 #define EXT2_EXTENT_SET_BMAP_UNINIT     0x0001
400
401 /*
402  * Data structure returned by ext2fs_extent_get_info()
403  */
404 struct ext2_extent_info {
405         int             curr_entry;
406         int             curr_level;
407         int             num_entries;
408         int             max_entries;
409         int             max_depth;
410         int             bytes_avail;
411         blk64_t         max_lblk;
412         blk64_t         max_pblk;
413         __u32           max_len;
414         __u32           max_uninit_len;
415 };
416
417 /*
418  * Flags for directory block reading and writing functions
419  */
420 #define EXT2_DIRBLOCK_V2_STRUCT 0x0001
421
422 /*
423  * Return flags for the directory iterator functions
424  */
425 #define DIRENT_CHANGED  1
426 #define DIRENT_ABORT    2
427 #define DIRENT_ERROR    3
428
429 /*
430  * Directory iterator flags
431  */
432
433 #define DIRENT_FLAG_INCLUDE_EMPTY       1
434 #define DIRENT_FLAG_INCLUDE_REMOVED     2
435
436 #define DIRENT_DOT_FILE         1
437 #define DIRENT_DOT_DOT_FILE     2
438 #define DIRENT_OTHER_FILE       3
439 #define DIRENT_DELETED_FILE     4
440
441 /*
442  * Inode scan definitions
443  */
444 typedef struct ext2_struct_inode_scan *ext2_inode_scan;
445
446 /*
447  * ext2fs_scan flags
448  */
449 #define EXT2_SF_CHK_BADBLOCKS   0x0001
450 #define EXT2_SF_BAD_INODE_BLK   0x0002
451 #define EXT2_SF_BAD_EXTRA_BYTES 0x0004
452 #define EXT2_SF_SKIP_MISSING_ITABLE     0x0008
453 #define EXT2_SF_DO_LAZY         0x0010
454
455 /*
456  * ext2fs_check_if_mounted flags
457  */
458 #define EXT2_MF_MOUNTED         1
459 #define EXT2_MF_ISROOT          2
460 #define EXT2_MF_READONLY        4
461 #define EXT2_MF_SWAP            8
462 #define EXT2_MF_BUSY            16
463
464 /*
465  * Ext2/linux mode flags.  We define them here so that we don't need
466  * to depend on the OS's sys/stat.h, since we may be compiling on a
467  * non-Linux system.
468  */
469 #define LINUX_S_IFMT  00170000
470 #define LINUX_S_IFSOCK 0140000
471 #define LINUX_S_IFLNK    0120000
472 #define LINUX_S_IFREG  0100000
473 #define LINUX_S_IFBLK  0060000
474 #define LINUX_S_IFDIR  0040000
475 #define LINUX_S_IFCHR  0020000
476 #define LINUX_S_IFIFO  0010000
477 #define LINUX_S_ISUID  0004000
478 #define LINUX_S_ISGID  0002000
479 #define LINUX_S_ISVTX  0001000
480
481 #define LINUX_S_IRWXU 00700
482 #define LINUX_S_IRUSR 00400
483 #define LINUX_S_IWUSR 00200
484 #define LINUX_S_IXUSR 00100
485
486 #define LINUX_S_IRWXG 00070
487 #define LINUX_S_IRGRP 00040
488 #define LINUX_S_IWGRP 00020
489 #define LINUX_S_IXGRP 00010
490
491 #define LINUX_S_IRWXO 00007
492 #define LINUX_S_IROTH 00004
493 #define LINUX_S_IWOTH 00002
494 #define LINUX_S_IXOTH 00001
495
496 #define LINUX_S_ISLNK(m)        (((m) & LINUX_S_IFMT) == LINUX_S_IFLNK)
497 #define LINUX_S_ISREG(m)        (((m) & LINUX_S_IFMT) == LINUX_S_IFREG)
498 #define LINUX_S_ISDIR(m)        (((m) & LINUX_S_IFMT) == LINUX_S_IFDIR)
499 #define LINUX_S_ISCHR(m)        (((m) & LINUX_S_IFMT) == LINUX_S_IFCHR)
500 #define LINUX_S_ISBLK(m)        (((m) & LINUX_S_IFMT) == LINUX_S_IFBLK)
501 #define LINUX_S_ISFIFO(m)       (((m) & LINUX_S_IFMT) == LINUX_S_IFIFO)
502 #define LINUX_S_ISSOCK(m)       (((m) & LINUX_S_IFMT) == LINUX_S_IFSOCK)
503
504 /*
505  * ext2 size of an inode
506  */
507 #define EXT2_I_SIZE(i)  ((i)->i_size | ((__u64) (i)->i_size_high << 32))
508
509 /*
510  * ext2_icount_t abstraction
511  */
512 #define EXT2_ICOUNT_OPT_INCREMENT       0x01
513
514 typedef struct ext2_icount *ext2_icount_t;
515
516 /*
517  * Flags for ext2fs_bmap
518  */
519 #define BMAP_ALLOC      0x0001
520 #define BMAP_SET        0x0002
521
522 /*
523  * Returned flags from ext2fs_bmap
524  */
525 #define BMAP_RET_UNINIT 0x0001
526
527 /*
528  * Flags for imager.c functions
529  */
530 #define IMAGER_FLAG_INODEMAP    1
531 #define IMAGER_FLAG_SPARSEWRITE 2
532
533 /*
534  * For checking structure magic numbers...
535  */
536
537 #define EXT2_CHECK_MAGIC(struct, code) \
538           if ((struct)->magic != (code)) return (code)
539
540 /*
541  * Flags for returning status of ext2fs_expand_extra_isize()
542  */
543 #define EXT2_EXPAND_EISIZE_UNSAFE       0x0001
544 #define EXT2_EXPAND_EISIZE_NEW_BLOCK    0x0002
545 #define EXT2_EXPAND_EISIZE_NOSPC        0x0004
546
547 /*
548  * For ext2 compression support
549  */
550 #define EXT2FS_COMPRESSED_BLKADDR ((blk_t) -1)
551 #define HOLE_BLKADDR(_b) ((_b) == 0 || (_b) == EXT2FS_COMPRESSED_BLKADDR)
552
553 /*
554  * Features supported by this version of the library
555  */
556 #define EXT2_LIB_FEATURE_COMPAT_SUPP    (EXT2_FEATURE_COMPAT_DIR_PREALLOC|\
557                                          EXT2_FEATURE_COMPAT_IMAGIC_INODES|\
558                                          EXT3_FEATURE_COMPAT_HAS_JOURNAL|\
559                                          EXT2_FEATURE_COMPAT_RESIZE_INODE|\
560                                          EXT2_FEATURE_COMPAT_DIR_INDEX|\
561                                          EXT2_FEATURE_COMPAT_EXT_ATTR|\
562                                          EXT4_FEATURE_COMPAT_SPARSE_SUPER2)
563
564 /* This #ifdef is temporary until compression is fully supported */
565 #ifdef ENABLE_COMPRESSION
566 #ifndef I_KNOW_THAT_COMPRESSION_IS_EXPERIMENTAL
567 /* If the below warning bugs you, then have
568    `CPPFLAGS=-DI_KNOW_THAT_COMPRESSION_IS_EXPERIMENTAL' in your
569    environment at configure time. */
570  #warning "Compression support is experimental"
571 #endif
572 #define EXT2_LIB_FEATURE_INCOMPAT_SUPP  (EXT2_FEATURE_INCOMPAT_FILETYPE|\
573                                          EXT2_FEATURE_INCOMPAT_COMPRESSION|\
574                                          EXT3_FEATURE_INCOMPAT_JOURNAL_DEV|\
575                                          EXT2_FEATURE_INCOMPAT_META_BG|\
576                                          EXT3_FEATURE_INCOMPAT_RECOVER|\
577                                          EXT3_FEATURE_INCOMPAT_EXTENTS|\
578                                          EXT4_FEATURE_INCOMPAT_FLEX_BG|\
579                                          EXT4_FEATURE_INCOMPAT_MMP|\
580                                          EXT4_FEATURE_INCOMPAT_64BIT)
581 #else
582 #define EXT2_LIB_FEATURE_INCOMPAT_SUPP  (EXT2_FEATURE_INCOMPAT_FILETYPE|\
583                                          EXT3_FEATURE_INCOMPAT_JOURNAL_DEV|\
584                                          EXT2_FEATURE_INCOMPAT_META_BG|\
585                                          EXT3_FEATURE_INCOMPAT_RECOVER|\
586                                          EXT3_FEATURE_INCOMPAT_EXTENTS|\
587                                          EXT4_FEATURE_INCOMPAT_FLEX_BG|\
588                                          EXT4_FEATURE_INCOMPAT_MMP|\
589                                          EXT4_FEATURE_INCOMPAT_64BIT)
590 #endif
591 #ifdef CONFIG_QUOTA
592 #define EXT2_LIB_FEATURE_RO_COMPAT_SUPP (EXT2_FEATURE_RO_COMPAT_SPARSE_SUPER|\
593                                          EXT4_FEATURE_RO_COMPAT_HUGE_FILE|\
594                                          EXT2_FEATURE_RO_COMPAT_LARGE_FILE|\
595                                          EXT4_FEATURE_RO_COMPAT_DIR_NLINK|\
596                                          EXT4_FEATURE_RO_COMPAT_EXTRA_ISIZE|\
597                                          EXT4_FEATURE_RO_COMPAT_GDT_CSUM|\
598                                          EXT4_FEATURE_RO_COMPAT_BIGALLOC|\
599                                          EXT4_FEATURE_RO_COMPAT_QUOTA)
600 #else
601 #define EXT2_LIB_FEATURE_RO_COMPAT_SUPP (EXT2_FEATURE_RO_COMPAT_SPARSE_SUPER|\
602                                          EXT4_FEATURE_RO_COMPAT_HUGE_FILE|\
603                                          EXT2_FEATURE_RO_COMPAT_LARGE_FILE|\
604                                          EXT4_FEATURE_RO_COMPAT_DIR_NLINK|\
605                                          EXT4_FEATURE_RO_COMPAT_EXTRA_ISIZE|\
606                                          EXT4_FEATURE_RO_COMPAT_GDT_CSUM|\
607                                          EXT4_FEATURE_RO_COMPAT_BIGALLOC)
608 #endif
609
610 /*
611  * These features are only allowed if EXT2_FLAG_SOFTSUPP_FEATURES is passed
612  * to ext2fs_openfs()
613  */
614 #define EXT2_LIB_SOFTSUPP_INCOMPAT      (0)
615 #define EXT2_LIB_SOFTSUPP_RO_COMPAT     (EXT4_FEATURE_RO_COMPAT_REPLICA)
616
617
618 /* Translate a block number to a cluster number */
619 #define EXT2FS_CLUSTER_RATIO(fs)        (1 << (fs)->cluster_ratio_bits)
620 #define EXT2FS_CLUSTER_MASK(fs)         (EXT2FS_CLUSTER_RATIO(fs) - 1)
621 #define EXT2FS_B2C(fs, blk)             ((blk) >> (fs)->cluster_ratio_bits)
622 /* Translate a cluster number to a block number */
623 #define EXT2FS_C2B(fs, cluster)         ((cluster) << (fs)->cluster_ratio_bits)
624 /* Translate # of blks to # of clusters */
625 #define EXT2FS_NUM_B2C(fs, blks)        (((blks) + EXT2FS_CLUSTER_MASK(fs)) >> \
626                                          (fs)->cluster_ratio_bits)
627
628 #if defined(HAVE_FSTAT64) && !defined(__OSX_AVAILABLE_BUT_DEPRECATED)
629 typedef struct stat64 ext2fs_struct_stat;
630 #else
631 typedef struct stat ext2fs_struct_stat;
632 #endif
633
634 /*
635  * For ext2fs_close2() and ext2fs_flush2(), this flag allows you to
636  * avoid the fsync call.
637  */
638 #define EXT2_FLAG_FLUSH_NO_SYNC          1
639
640 /*
641  * function prototypes
642  */
643
644 /* The LARGE_FILE feature should be set if we have stored files 2GB+ in size */
645 static inline int ext2fs_needs_large_file_feature(unsigned long long file_size)
646 {
647         return file_size >= 0x80000000ULL;
648 }
649
650 /* alloc.c */
651 extern errcode_t ext2fs_new_inode(ext2_filsys fs, ext2_ino_t dir, int mode,
652                                   ext2fs_inode_bitmap map, ext2_ino_t *ret);
653 extern errcode_t ext2fs_new_block(ext2_filsys fs, blk_t goal,
654                                   ext2fs_block_bitmap map, blk_t *ret);
655 extern errcode_t ext2fs_new_block2(ext2_filsys fs, blk64_t goal,
656                                    ext2fs_block_bitmap map, blk64_t *ret);
657 extern errcode_t ext2fs_get_free_blocks(ext2_filsys fs, blk_t start,
658                                         blk_t finish, int num,
659                                         ext2fs_block_bitmap map,
660                                         blk_t *ret);
661 extern errcode_t ext2fs_get_free_blocks2(ext2_filsys fs, blk64_t start,
662                                          blk64_t finish, int num,
663                                          ext2fs_block_bitmap map,
664                                          blk64_t *ret);
665 extern errcode_t ext2fs_alloc_block(ext2_filsys fs, blk_t goal,
666                                     char *block_buf, blk_t *ret);
667 extern errcode_t ext2fs_alloc_block2(ext2_filsys fs, blk64_t goal,
668                                      char *block_buf, blk64_t *ret);
669 extern void ext2fs_set_alloc_block_callback(ext2_filsys fs,
670                                             errcode_t (*func)(ext2_filsys fs,
671                                                               blk64_t goal,
672                                                               blk64_t *ret),
673                                             errcode_t (**old)(ext2_filsys fs,
674                                                               blk64_t goal,
675                                                               blk64_t *ret));
676
677 /* alloc_sb.c */
678 extern int ext2fs_reserve_super_and_bgd(ext2_filsys fs,
679                                         dgrp_t group,
680                                         ext2fs_block_bitmap bmap);
681 extern void ext2fs_set_block_alloc_stats_callback(ext2_filsys fs,
682                                                   void (*func)(ext2_filsys fs,
683                                                                blk64_t blk,
684                                                                int inuse),
685                                                   void (**old)(ext2_filsys fs,
686                                                                blk64_t blk,
687                                                                int inuse));
688
689 /* alloc_stats.c */
690 void ext2fs_inode_alloc_stats(ext2_filsys fs, ext2_ino_t ino, int inuse);
691 void ext2fs_inode_alloc_stats2(ext2_filsys fs, ext2_ino_t ino,
692                                int inuse, int isdir);
693 void ext2fs_block_alloc_stats(ext2_filsys fs, blk_t blk, int inuse);
694 void ext2fs_block_alloc_stats2(ext2_filsys fs, blk64_t blk, int inuse);
695 void ext2fs_block_alloc_stats_range(ext2_filsys fs, blk64_t blk,
696                                     blk_t num, int inuse);
697
698 /* alloc_tables.c */
699 extern errcode_t ext2fs_allocate_tables(ext2_filsys fs);
700 extern errcode_t ext2fs_allocate_group_table(ext2_filsys fs, dgrp_t group,
701                                              ext2fs_block_bitmap bmap);
702
703 /* badblocks.c */
704 extern errcode_t ext2fs_u32_list_create(ext2_u32_list *ret, int size);
705 extern errcode_t ext2fs_u32_list_add(ext2_u32_list bb, __u32 blk);
706 extern int ext2fs_u32_list_find(ext2_u32_list bb, __u32 blk);
707 extern int ext2fs_u32_list_test(ext2_u32_list bb, blk_t blk);
708 extern errcode_t ext2fs_u32_list_iterate_begin(ext2_u32_list bb,
709                                                ext2_u32_iterate *ret);
710 extern int ext2fs_u32_list_iterate(ext2_u32_iterate iter, blk_t *blk);
711 extern void ext2fs_u32_list_iterate_end(ext2_u32_iterate iter);
712 extern errcode_t ext2fs_u32_copy(ext2_u32_list src, ext2_u32_list *dest);
713 extern int ext2fs_u32_list_equal(ext2_u32_list bb1, ext2_u32_list bb2);
714
715 extern errcode_t ext2fs_badblocks_list_create(ext2_badblocks_list *ret,
716                                             int size);
717 extern errcode_t ext2fs_badblocks_list_add(ext2_badblocks_list bb,
718                                            blk_t blk);
719 extern int ext2fs_badblocks_list_test(ext2_badblocks_list bb,
720                                     blk_t blk);
721 extern int ext2fs_u32_list_del(ext2_u32_list bb, __u32 blk);
722 extern void ext2fs_badblocks_list_del(ext2_u32_list bb, __u32 blk);
723 extern errcode_t
724         ext2fs_badblocks_list_iterate_begin(ext2_badblocks_list bb,
725                                             ext2_badblocks_iterate *ret);
726 extern int ext2fs_badblocks_list_iterate(ext2_badblocks_iterate iter,
727                                          blk_t *blk);
728 extern void ext2fs_badblocks_list_iterate_end(ext2_badblocks_iterate iter);
729 extern errcode_t ext2fs_badblocks_copy(ext2_badblocks_list src,
730                                        ext2_badblocks_list *dest);
731 extern int ext2fs_badblocks_equal(ext2_badblocks_list bb1,
732                                   ext2_badblocks_list bb2);
733 extern int ext2fs_u32_list_count(ext2_u32_list bb);
734
735 /* bb_compat */
736 extern errcode_t badblocks_list_create(badblocks_list *ret, int size);
737 extern errcode_t badblocks_list_add(badblocks_list bb, blk_t blk);
738 extern int badblocks_list_test(badblocks_list bb, blk_t blk);
739 extern errcode_t badblocks_list_iterate_begin(badblocks_list bb,
740                                               badblocks_iterate *ret);
741 extern int badblocks_list_iterate(badblocks_iterate iter, blk_t *blk);
742 extern void badblocks_list_iterate_end(badblocks_iterate iter);
743 extern void badblocks_list_free(badblocks_list bb);
744
745 /* bb_inode.c */
746 extern errcode_t ext2fs_update_bb_inode(ext2_filsys fs,
747                                         ext2_badblocks_list bb_list);
748
749 /* bitmaps.c */
750 extern void ext2fs_free_block_bitmap(ext2fs_block_bitmap bitmap);
751 extern void ext2fs_free_inode_bitmap(ext2fs_inode_bitmap bitmap);
752 extern errcode_t ext2fs_copy_bitmap(ext2fs_generic_bitmap src,
753                                     ext2fs_generic_bitmap *dest);
754 extern errcode_t ext2fs_write_inode_bitmap(ext2_filsys fs);
755 extern errcode_t ext2fs_write_block_bitmap (ext2_filsys fs);
756 extern errcode_t ext2fs_read_inode_bitmap (ext2_filsys fs);
757 extern errcode_t ext2fs_read_block_bitmap(ext2_filsys fs);
758 extern errcode_t ext2fs_allocate_block_bitmap(ext2_filsys fs,
759                                               const char *descr,
760                                               ext2fs_block_bitmap *ret);
761 extern errcode_t ext2fs_allocate_subcluster_bitmap(ext2_filsys fs,
762                                                    const char *descr,
763                                                    ext2fs_block_bitmap *ret);
764 extern int ext2fs_get_bitmap_granularity(ext2fs_block_bitmap bitmap);
765 extern errcode_t ext2fs_allocate_inode_bitmap(ext2_filsys fs,
766                                               const char *descr,
767                                               ext2fs_inode_bitmap *ret);
768 extern errcode_t ext2fs_fudge_inode_bitmap_end(ext2fs_inode_bitmap bitmap,
769                                                ext2_ino_t end, ext2_ino_t *oend);
770 extern errcode_t ext2fs_fudge_block_bitmap_end(ext2fs_block_bitmap bitmap,
771                                                blk_t end, blk_t *oend);
772 extern errcode_t ext2fs_fudge_block_bitmap_end2(ext2fs_block_bitmap bitmap,
773                                          blk64_t end, blk64_t *oend);
774 extern void ext2fs_clear_inode_bitmap(ext2fs_inode_bitmap bitmap);
775 extern void ext2fs_clear_block_bitmap(ext2fs_block_bitmap bitmap);
776 extern errcode_t ext2fs_read_bitmaps(ext2_filsys fs);
777 extern errcode_t ext2fs_write_bitmaps(ext2_filsys fs);
778 extern errcode_t ext2fs_resize_inode_bitmap(__u32 new_end, __u32 new_real_end,
779                                             ext2fs_inode_bitmap bmap);
780 extern errcode_t ext2fs_resize_inode_bitmap2(__u64 new_end,
781                                              __u64 new_real_end,
782                                              ext2fs_inode_bitmap bmap);
783 extern errcode_t ext2fs_resize_block_bitmap(__u32 new_end, __u32 new_real_end,
784                                             ext2fs_block_bitmap bmap);
785 extern errcode_t ext2fs_resize_block_bitmap2(__u64 new_end,
786                                              __u64 new_real_end,
787                                              ext2fs_block_bitmap bmap);
788 extern errcode_t ext2fs_compare_block_bitmap(ext2fs_block_bitmap bm1,
789                                              ext2fs_block_bitmap bm2);
790 extern errcode_t ext2fs_compare_inode_bitmap(ext2fs_inode_bitmap bm1,
791                                              ext2fs_inode_bitmap bm2);
792 extern errcode_t ext2fs_set_inode_bitmap_range(ext2fs_inode_bitmap bmap,
793                                         ext2_ino_t start, unsigned int num,
794                                         void *in);
795 extern errcode_t ext2fs_set_inode_bitmap_range2(ext2fs_inode_bitmap bmap,
796                                          __u64 start, size_t num,
797                                          void *in);
798 extern errcode_t ext2fs_get_inode_bitmap_range(ext2fs_inode_bitmap bmap,
799                                         ext2_ino_t start, unsigned int num,
800                                         void *out);
801 extern errcode_t ext2fs_get_inode_bitmap_range2(ext2fs_inode_bitmap bmap,
802                                          __u64 start, size_t num,
803                                          void *out);
804 extern errcode_t ext2fs_set_block_bitmap_range(ext2fs_block_bitmap bmap,
805                                         blk_t start, unsigned int num,
806                                         void *in);
807 extern errcode_t ext2fs_set_block_bitmap_range2(ext2fs_block_bitmap bmap,
808                                          blk64_t start, size_t num,
809                                          void *in);
810 extern errcode_t ext2fs_get_block_bitmap_range(ext2fs_block_bitmap bmap,
811                                         blk_t start, unsigned int num,
812                                         void *out);
813 extern errcode_t ext2fs_get_block_bitmap_range2(ext2fs_block_bitmap bmap,
814                                          blk64_t start, size_t num,
815                                          void *out);
816
817 /* blknum.c */
818 extern dgrp_t ext2fs_group_of_blk2(ext2_filsys fs, blk64_t);
819 extern blk64_t ext2fs_group_first_block2(ext2_filsys fs, dgrp_t group);
820 extern blk64_t ext2fs_group_last_block2(ext2_filsys fs, dgrp_t group);
821 extern int ext2fs_group_blocks_count(ext2_filsys fs, dgrp_t group);
822 extern blk64_t ext2fs_inode_data_blocks2(ext2_filsys fs,
823                                          struct ext2_inode *inode);
824 extern blk64_t ext2fs_inode_i_blocks(ext2_filsys fs,
825                                          struct ext2_inode *inode);
826 extern blk64_t ext2fs_blocks_count(struct ext2_super_block *super);
827 extern void ext2fs_blocks_count_set(struct ext2_super_block *super,
828                                     blk64_t blk);
829 extern void ext2fs_blocks_count_add(struct ext2_super_block *super,
830                                     blk64_t blk);
831 extern blk64_t ext2fs_r_blocks_count(struct ext2_super_block *super);
832 extern void ext2fs_r_blocks_count_set(struct ext2_super_block *super,
833                                       blk64_t blk);
834 extern void ext2fs_r_blocks_count_add(struct ext2_super_block *super,
835                                       blk64_t blk);
836 extern blk64_t ext2fs_free_blocks_count(struct ext2_super_block *super);
837 extern void ext2fs_free_blocks_count_set(struct ext2_super_block *super,
838                                          blk64_t blk);
839 extern void ext2fs_free_blocks_count_add(struct ext2_super_block *super,
840                                          blk64_t blk);
841 /* Block group descriptor accessor functions */
842 extern struct ext2_group_desc *ext2fs_group_desc(ext2_filsys fs,
843                                           struct opaque_ext2_group_desc *gdp,
844                                           dgrp_t group);
845 extern blk64_t ext2fs_block_bitmap_loc(ext2_filsys fs, dgrp_t group);
846 extern void ext2fs_block_bitmap_loc_set(ext2_filsys fs, dgrp_t group,
847                                         blk64_t blk);
848 extern blk64_t ext2fs_inode_bitmap_loc(ext2_filsys fs, dgrp_t group);
849 extern void ext2fs_inode_bitmap_loc_set(ext2_filsys fs, dgrp_t group,
850                                         blk64_t blk);
851 extern blk64_t ext2fs_inode_table_loc(ext2_filsys fs, dgrp_t group);
852 extern void ext2fs_inode_table_loc_set(ext2_filsys fs, dgrp_t group,
853                                        blk64_t blk);
854 extern __u32 ext2fs_bg_free_blocks_count(ext2_filsys fs, dgrp_t group);
855 extern void ext2fs_bg_free_blocks_count_set(ext2_filsys fs, dgrp_t group,
856                                          __u32 n);
857 extern __u32 ext2fs_bg_free_inodes_count(ext2_filsys fs, dgrp_t group);
858 extern void ext2fs_bg_free_inodes_count_set(ext2_filsys fs, dgrp_t group,
859                                          __u32 n);
860 extern __u32 ext2fs_bg_used_dirs_count(ext2_filsys fs, dgrp_t group);
861 extern void ext2fs_bg_used_dirs_count_set(ext2_filsys fs, dgrp_t group,
862                                        __u32 n);
863 extern __u32 ext2fs_bg_itable_unused(ext2_filsys fs, dgrp_t group);
864 extern void ext2fs_bg_itable_unused_set(ext2_filsys fs, dgrp_t group,
865                                      __u32 n);
866 extern __u16 ext2fs_bg_flags(ext2_filsys fs, dgrp_t group);
867 extern void ext2fs_bg_flags_zap(ext2_filsys fs, dgrp_t group);
868 extern int ext2fs_bg_flags_test(ext2_filsys fs, dgrp_t group, __u16 bg_flag);
869 extern void ext2fs_bg_flags_set(ext2_filsys fs, dgrp_t group, __u16 bg_flags);
870 extern void ext2fs_bg_flags_clear(ext2_filsys fs, dgrp_t group, __u16 bg_flags);
871 extern __u16 ext2fs_bg_checksum(ext2_filsys fs, dgrp_t group);
872 extern void ext2fs_bg_checksum_set(ext2_filsys fs, dgrp_t group, __u16 checksum);
873 extern blk64_t ext2fs_file_acl_block(ext2_filsys fs,
874                                      const struct ext2_inode *inode);
875 extern void ext2fs_file_acl_block_set(ext2_filsys fs,
876                                       struct ext2_inode *inode, blk64_t blk);
877 extern errcode_t ext2fs_inode_size_set(ext2_filsys fs, struct ext2_inode *inode,
878                                        ext2_off64_t size);
879
880 /* block.c */
881 extern errcode_t ext2fs_block_iterate(ext2_filsys fs,
882                                       ext2_ino_t        ino,
883                                       int       flags,
884                                       char *block_buf,
885                                       int (*func)(ext2_filsys fs,
886                                                   blk_t *blocknr,
887                                                   int   blockcnt,
888                                                   void  *priv_data),
889                                       void *priv_data);
890 errcode_t ext2fs_block_iterate2(ext2_filsys fs,
891                                 ext2_ino_t      ino,
892                                 int     flags,
893                                 char *block_buf,
894                                 int (*func)(ext2_filsys fs,
895                                             blk_t       *blocknr,
896                                             e2_blkcnt_t blockcnt,
897                                             blk_t       ref_blk,
898                                             int         ref_offset,
899                                             void        *priv_data),
900                                 void *priv_data);
901 errcode_t ext2fs_block_iterate3(ext2_filsys fs,
902                                 ext2_ino_t ino,
903                                 int     flags,
904                                 char *block_buf,
905                                 int (*func)(ext2_filsys fs,
906                                             blk64_t     *blocknr,
907                                             e2_blkcnt_t blockcnt,
908                                             blk64_t     ref_blk,
909                                             int         ref_offset,
910                                             void        *priv_data),
911                                 void *priv_data);
912
913 /* bmap.c */
914 extern errcode_t ext2fs_bmap(ext2_filsys fs, ext2_ino_t ino,
915                              struct ext2_inode *inode,
916                              char *block_buf, int bmap_flags,
917                              blk_t block, blk_t *phys_blk);
918 extern errcode_t ext2fs_bmap2(ext2_filsys fs, ext2_ino_t ino,
919                               struct ext2_inode *inode,
920                               char *block_buf, int bmap_flags, blk64_t block,
921                               int *ret_flags, blk64_t *phys_blk);
922 errcode_t ext2fs_map_cluster_block(ext2_filsys fs, ext2_ino_t ino,
923                                    struct ext2_inode *inode, blk64_t lblk,
924                                    blk64_t *pblk);
925
926 #if 0
927 /* bmove.c */
928 extern errcode_t ext2fs_move_blocks(ext2_filsys fs,
929                                     ext2fs_block_bitmap reserve,
930                                     ext2fs_block_bitmap alloc_map,
931                                     int flags);
932 #endif
933
934 /* check_desc.c */
935 extern errcode_t ext2fs_check_desc(ext2_filsys fs);
936
937 /* closefs.c */
938 extern errcode_t ext2fs_close(ext2_filsys fs);
939 extern errcode_t ext2fs_close2(ext2_filsys fs, int flags);
940 extern errcode_t ext2fs_close_free(ext2_filsys *fs);
941 extern errcode_t ext2fs_flush(ext2_filsys fs);
942 extern errcode_t ext2fs_flush2(ext2_filsys fs, int flags);
943 extern int ext2fs_bg_has_super(ext2_filsys fs, dgrp_t group_block);
944 extern errcode_t ext2fs_super_and_bgd_loc2(ext2_filsys fs,
945                                     dgrp_t group,
946                                     blk64_t *ret_super_blk,
947                                     blk64_t *ret_old_desc_blk,
948                                     blk64_t *ret_new_desc_blk,
949                                     blk_t *ret_used_blks);
950 extern int ext2fs_super_and_bgd_loc(ext2_filsys fs,
951                                     dgrp_t group,
952                                     blk_t *ret_super_blk,
953                                     blk_t *ret_old_desc_blk,
954                                     blk_t *ret_new_desc_blk,
955                                     int *ret_meta_bg);
956 extern void ext2fs_update_dynamic_rev(ext2_filsys fs);
957
958 /* crc32c.c */
959 extern __u32 ext2fs_crc32c_be(__u32 crc, unsigned char const *p, size_t len);
960 extern __u32 ext2fs_crc32c_le(__u32 crc, unsigned char const *p, size_t len);
961
962 /* csum.c */
963 extern void ext2fs_group_desc_csum_set(ext2_filsys fs, dgrp_t group);
964 extern int ext2fs_group_desc_csum_verify(ext2_filsys fs, dgrp_t group);
965 extern errcode_t ext2fs_set_gdt_csum(ext2_filsys fs);
966 extern __u16 ext2fs_group_desc_csum(ext2_filsys fs, dgrp_t group);
967
968 /* dblist.c */
969
970 extern errcode_t ext2fs_get_num_dirs(ext2_filsys fs, ext2_ino_t *ret_num_dirs);
971 extern errcode_t ext2fs_init_dblist(ext2_filsys fs, ext2_dblist *ret_dblist);
972 extern errcode_t ext2fs_add_dir_block(ext2_dblist dblist, ext2_ino_t ino,
973                                       blk_t blk, int blockcnt);
974 extern errcode_t ext2fs_add_dir_block2(ext2_dblist dblist, ext2_ino_t ino,
975                                        blk64_t blk, e2_blkcnt_t blockcnt);
976 extern void ext2fs_dblist_sort(ext2_dblist dblist,
977                                EXT2_QSORT_TYPE (*sortfunc)(const void *,
978                                                            const void *));
979 extern void ext2fs_dblist_sort2(ext2_dblist dblist,
980                                 EXT2_QSORT_TYPE (*sortfunc)(const void *,
981                                                             const void *));
982 extern errcode_t ext2fs_dblist_iterate(ext2_dblist dblist,
983         int (*func)(ext2_filsys fs, struct ext2_db_entry *db_info,
984                     void        *priv_data),
985        void *priv_data);
986 extern errcode_t ext2fs_dblist_iterate2(ext2_dblist dblist,
987         int (*func)(ext2_filsys fs, struct ext2_db_entry2 *db_info,
988                     void        *priv_data),
989        void *priv_data);
990 extern errcode_t ext2fs_set_dir_block(ext2_dblist dblist, ext2_ino_t ino,
991                                       blk_t blk, int blockcnt);
992 extern errcode_t ext2fs_set_dir_block2(ext2_dblist dblist, ext2_ino_t ino,
993                                        blk64_t blk, e2_blkcnt_t blockcnt);
994 extern errcode_t ext2fs_copy_dblist(ext2_dblist src,
995                                     ext2_dblist *dest);
996 extern int ext2fs_dblist_count(ext2_dblist dblist);
997 extern blk64_t ext2fs_dblist_count2(ext2_dblist dblist);
998 extern errcode_t ext2fs_dblist_get_last(ext2_dblist dblist,
999                                         struct ext2_db_entry **entry);
1000 extern errcode_t ext2fs_dblist_get_last2(ext2_dblist dblist,
1001                                         struct ext2_db_entry2 **entry);
1002 extern errcode_t ext2fs_dblist_drop_last(ext2_dblist dblist);
1003
1004 /* dblist_dir.c */
1005 extern errcode_t
1006         ext2fs_dblist_dir_iterate(ext2_dblist dblist,
1007                                   int   flags,
1008                                   char  *block_buf,
1009                                   int (*func)(ext2_ino_t        dir,
1010                                               int               entry,
1011                                               struct ext2_dir_entry *dirent,
1012                                               int       offset,
1013                                               int       blocksize,
1014                                               char      *buf,
1015                                               void      *priv_data),
1016                                   void *priv_data);
1017
1018 /* dirblock.c */
1019 extern errcode_t ext2fs_read_dir_block(ext2_filsys fs, blk_t block,
1020                                        void *buf);
1021 extern errcode_t ext2fs_read_dir_block2(ext2_filsys fs, blk_t block,
1022                                         void *buf, int flags);
1023 extern errcode_t ext2fs_read_dir_block3(ext2_filsys fs, blk64_t block,
1024                                         void *buf, int flags);
1025 extern errcode_t ext2fs_write_dir_block(ext2_filsys fs, blk_t block,
1026                                         void *buf);
1027 extern errcode_t ext2fs_write_dir_block2(ext2_filsys fs, blk_t block,
1028                                          void *buf, int flags);
1029 extern errcode_t ext2fs_write_dir_block3(ext2_filsys fs, blk64_t block,
1030                                          void *buf, int flags);
1031
1032 /* dirhash.c */
1033 extern errcode_t ext2fs_dirhash(int version, const char *name, int len,
1034                                 const __u32 *seed,
1035                                 ext2_dirhash_t *ret_hash,
1036                                 ext2_dirhash_t *ret_minor_hash);
1037
1038
1039 /* dir_iterate.c */
1040 extern errcode_t ext2fs_get_rec_len(ext2_filsys fs,
1041                                     struct ext2_dir_entry *dirent,
1042                                     unsigned int *rec_len);
1043 extern errcode_t ext2fs_set_rec_len(ext2_filsys fs,
1044                                     unsigned int len,
1045                                     struct ext2_dir_entry *dirent);
1046 extern errcode_t ext2fs_dir_iterate(ext2_filsys fs,
1047                               ext2_ino_t dir,
1048                               int flags,
1049                               char *block_buf,
1050                               int (*func)(struct ext2_dir_entry *dirent,
1051                                           int   offset,
1052                                           int   blocksize,
1053                                           char  *buf,
1054                                           void  *priv_data),
1055                               void *priv_data);
1056 extern errcode_t ext2fs_dir_iterate2(ext2_filsys fs,
1057                               ext2_ino_t dir,
1058                               int flags,
1059                               char *block_buf,
1060                               int (*func)(ext2_ino_t    dir,
1061                                           int   entry,
1062                                           struct ext2_dir_entry *dirent,
1063                                           int   offset,
1064                                           int   blocksize,
1065                                           char  *buf,
1066                                           void  *priv_data),
1067                               void *priv_data);
1068
1069 /* dupfs.c */
1070 extern errcode_t ext2fs_dup_handle(ext2_filsys src, ext2_filsys *dest);
1071
1072 /* expanddir.c */
1073 extern errcode_t ext2fs_expand_dir(ext2_filsys fs, ext2_ino_t dir);
1074
1075 /* ext_attr.c */
1076 extern errcode_t ext2fs_attr_get(ext2_filsys fs, struct ext2_inode *inode,
1077                                  int name_index, const char *name, char *buffer,
1078                                  size_t buffer_size, int *easize);
1079
1080 extern __u32 ext2fs_ext_attr_hash_entry(struct ext2_ext_attr_entry *entry,
1081                                         void *data);
1082 int ext2fs_attr_get_next_attr(struct ext2_ext_attr_entry *entry, int name_index,
1083                               char *buffer, int buffer_size, int start);
1084 errcode_t ext2fs_attr_set(ext2_filsys fs, ext2_ino_t ino,
1085                           struct ext2_inode *inode,
1086                           int name_index, const char *name, const char *value,
1087                           int value_len, int flags);
1088 extern errcode_t ext2fs_expand_extra_isize(ext2_filsys fs, ext2_ino_t ino,
1089                                            struct ext2_inode_large *inode,
1090                                            int new_extra_isize, int *ret,
1091                                            int *needed_size);
1092 extern errcode_t ext2fs_read_ext_attr(ext2_filsys fs, blk_t block, void *buf);
1093 extern errcode_t ext2fs_read_ext_attr2(ext2_filsys fs, blk64_t block,
1094                                        void *buf);
1095 extern errcode_t ext2fs_write_ext_attr(ext2_filsys fs, blk_t block,
1096                                        void *buf);
1097 extern errcode_t ext2fs_write_ext_attr2(ext2_filsys fs, blk64_t block,
1098                                        void *buf);
1099 extern errcode_t ext2fs_adjust_ea_refcount(ext2_filsys fs, blk_t blk,
1100                                            char *block_buf,
1101                                            int adjust, __u32 *newcount);
1102 extern errcode_t ext2fs_adjust_ea_refcount2(ext2_filsys fs, blk64_t blk,
1103                                            char *block_buf,
1104                                            int adjust, __u32 *newcount);
1105
1106 /* extent.c */
1107 extern errcode_t ext2fs_extent_header_verify(void *ptr, int size);
1108 extern errcode_t ext2fs_extent_open(ext2_filsys fs, ext2_ino_t ino,
1109                                     ext2_extent_handle_t *handle);
1110 extern errcode_t ext2fs_extent_open2(ext2_filsys fs, ext2_ino_t ino,
1111                                         struct ext2_inode *inode,
1112                                         ext2_extent_handle_t *ret_handle);
1113 extern void ext2fs_extent_free(ext2_extent_handle_t handle);
1114 extern errcode_t ext2fs_extent_get(ext2_extent_handle_t handle,
1115                                    int flags, struct ext2fs_extent *extent);
1116 extern errcode_t ext2fs_extent_node_split(ext2_extent_handle_t handle);
1117 extern errcode_t ext2fs_extent_replace(ext2_extent_handle_t handle, int flags,
1118                                        struct ext2fs_extent *extent);
1119 extern errcode_t ext2fs_extent_insert(ext2_extent_handle_t handle, int flags,
1120                                       struct ext2fs_extent *extent);
1121 extern errcode_t ext2fs_extent_set_bmap(ext2_extent_handle_t handle,
1122                                         blk64_t logical, blk64_t physical,
1123                                         int flags);
1124 extern errcode_t ext2fs_extent_delete(ext2_extent_handle_t handle, int flags);
1125 extern errcode_t ext2fs_extent_get_info(ext2_extent_handle_t handle,
1126                                         struct ext2_extent_info *info);
1127 extern errcode_t ext2fs_extent_goto(ext2_extent_handle_t handle,
1128                                     blk64_t blk);
1129 extern errcode_t ext2fs_extent_goto2(ext2_extent_handle_t handle,
1130                                      int leaf_level, blk64_t blk);
1131 extern errcode_t ext2fs_extent_fix_parents(ext2_extent_handle_t handle);
1132
1133 /* fileio.c */
1134 extern errcode_t ext2fs_file_open2(ext2_filsys fs, ext2_ino_t ino,
1135                                    struct ext2_inode *inode,
1136                                    int flags, ext2_file_t *ret);
1137 extern errcode_t ext2fs_file_open(ext2_filsys fs, ext2_ino_t ino,
1138                                   int flags, ext2_file_t *ret);
1139 extern ext2_filsys ext2fs_file_get_fs(ext2_file_t file);
1140 struct ext2_inode *ext2fs_file_get_inode(ext2_file_t file);
1141 extern ext2_ino_t ext2fs_file_get_inode_num(ext2_file_t file);
1142 extern errcode_t ext2fs_file_close(ext2_file_t file);
1143 extern errcode_t ext2fs_file_flush(ext2_file_t file);
1144 extern errcode_t ext2fs_file_read(ext2_file_t file, void *buf,
1145                                   unsigned int wanted, unsigned int *got);
1146 extern errcode_t ext2fs_file_write(ext2_file_t file, const void *buf,
1147                                    unsigned int nbytes, unsigned int *written);
1148 extern errcode_t ext2fs_file_llseek(ext2_file_t file, __u64 offset,
1149                                    int whence, __u64 *ret_pos);
1150 extern errcode_t ext2fs_file_lseek(ext2_file_t file, ext2_off_t offset,
1151                                    int whence, ext2_off_t *ret_pos);
1152 errcode_t ext2fs_file_get_lsize(ext2_file_t file, __u64 *ret_size);
1153 extern ext2_off_t ext2fs_file_get_size(ext2_file_t file);
1154 extern errcode_t ext2fs_file_set_size(ext2_file_t file, ext2_off_t size);
1155 extern errcode_t ext2fs_file_set_size2(ext2_file_t file, ext2_off64_t size);
1156
1157 /* finddev.c */
1158 extern char *ext2fs_find_block_device(dev_t device);
1159
1160 /* flushb.c */
1161 extern errcode_t ext2fs_sync_device(int fd, int flushb);
1162
1163 /* freefs.c */
1164 extern void ext2fs_free(ext2_filsys fs);
1165 extern void ext2fs_free_dblist(ext2_dblist dblist);
1166 extern void ext2fs_badblocks_list_free(ext2_badblocks_list bb);
1167 extern void ext2fs_u32_list_free(ext2_u32_list bb);
1168
1169 /* gen_bitmap.c */
1170 extern void ext2fs_free_generic_bitmap(ext2fs_inode_bitmap bitmap);
1171 extern errcode_t ext2fs_make_generic_bitmap(errcode_t magic, ext2_filsys fs,
1172                                             __u32 start, __u32 end,
1173                                             __u32 real_end,
1174                                             const char *descr, char *init_map,
1175                                             ext2fs_generic_bitmap *ret);
1176 extern errcode_t ext2fs_allocate_generic_bitmap(__u32 start,
1177                                                 __u32 end,
1178                                                 __u32 real_end,
1179                                                 const char *descr,
1180                                                 ext2fs_generic_bitmap *ret);
1181 extern errcode_t ext2fs_copy_generic_bitmap(ext2fs_generic_bitmap src,
1182                                             ext2fs_generic_bitmap *dest);
1183 extern void ext2fs_clear_generic_bitmap(ext2fs_generic_bitmap bitmap);
1184 extern errcode_t ext2fs_fudge_generic_bitmap_end(ext2fs_inode_bitmap bitmap,
1185                                                  errcode_t magic,
1186                                                  errcode_t neq,
1187                                                  ext2_ino_t end,
1188                                                  ext2_ino_t *oend);
1189 extern void ext2fs_set_generic_bitmap_padding(ext2fs_generic_bitmap map);
1190 extern errcode_t ext2fs_resize_generic_bitmap(errcode_t magic,
1191                                               __u32 new_end,
1192                                               __u32 new_real_end,
1193                                               ext2fs_generic_bitmap bmap);
1194 extern errcode_t ext2fs_compare_generic_bitmap(errcode_t magic, errcode_t neq,
1195                                                ext2fs_generic_bitmap bm1,
1196                                                ext2fs_generic_bitmap bm2);
1197 extern errcode_t ext2fs_get_generic_bitmap_range(ext2fs_generic_bitmap bmap,
1198                                                  errcode_t magic,
1199                                                  __u32 start, __u32 num,
1200                                                  void *out);
1201 extern errcode_t ext2fs_set_generic_bitmap_range(ext2fs_generic_bitmap bmap,
1202                                                  errcode_t magic,
1203                                                  __u32 start, __u32 num,
1204                                                  void *in);
1205 extern errcode_t ext2fs_find_first_zero_generic_bitmap(ext2fs_generic_bitmap bitmap,
1206                                                        __u32 start, __u32 end,
1207                                                        __u32 *out);
1208 extern errcode_t ext2fs_find_first_set_generic_bitmap(ext2fs_generic_bitmap bitmap,
1209                                                        __u32 start, __u32 end,
1210                                                        __u32 *out);
1211
1212 /* gen_bitmap64.c */
1213
1214 /* Generate and print bitmap usage statistics */
1215 #define BMAP_STATS
1216
1217 void ext2fs_free_generic_bmap(ext2fs_generic_bitmap bmap);
1218 errcode_t ext2fs_alloc_generic_bmap(ext2_filsys fs, errcode_t magic,
1219                                     int type, __u64 start, __u64 end,
1220                                     __u64 real_end,
1221                                     const char *descr,
1222                                     ext2fs_generic_bitmap *ret);
1223 errcode_t ext2fs_copy_generic_bmap(ext2fs_generic_bitmap src,
1224                                    ext2fs_generic_bitmap *dest);
1225 void ext2fs_clear_generic_bmap(ext2fs_generic_bitmap bitmap);
1226 errcode_t ext2fs_fudge_generic_bmap_end(ext2fs_generic_bitmap bitmap,
1227                                         errcode_t neq,
1228                                         __u64 end, __u64 *oend);
1229 void ext2fs_set_generic_bmap_padding(ext2fs_generic_bitmap bmap);
1230 errcode_t ext2fs_resize_generic_bmap(ext2fs_generic_bitmap bmap,
1231                                      __u64 new_end,
1232                                      __u64 new_real_end);
1233 errcode_t ext2fs_compare_generic_bmap(errcode_t neq,
1234                                       ext2fs_generic_bitmap bm1,
1235                                       ext2fs_generic_bitmap bm2);
1236 errcode_t ext2fs_get_generic_bmap_range(ext2fs_generic_bitmap bmap,
1237                                         __u64 start, unsigned int num,
1238                                         void *out);
1239 errcode_t ext2fs_set_generic_bmap_range(ext2fs_generic_bitmap bmap,
1240                                         __u64 start, unsigned int num,
1241                                         void *in);
1242 errcode_t ext2fs_convert_subcluster_bitmap(ext2_filsys fs,
1243                                            ext2fs_block_bitmap *bitmap);
1244
1245 /* getsize.c */
1246 extern errcode_t ext2fs_get_device_size(const char *file, int blocksize,
1247                                         blk_t *retblocks);
1248 extern errcode_t ext2fs_get_device_size2(const char *file, int blocksize,
1249                                         blk64_t *retblocks);
1250
1251 /* getsectsize.c */
1252 extern int ext2fs_get_dio_alignment(int fd);
1253 errcode_t ext2fs_get_device_sectsize(const char *file, int *sectsize);
1254 errcode_t ext2fs_get_device_phys_sectsize(const char *file, int *sectsize);
1255
1256 /* i_block.c */
1257 errcode_t ext2fs_iblk_add_blocks(ext2_filsys fs, struct ext2_inode *inode,
1258                                  blk64_t num_blocks);
1259 errcode_t ext2fs_iblk_sub_blocks(ext2_filsys fs, struct ext2_inode *inode,
1260                                  blk64_t num_blocks);
1261 errcode_t ext2fs_iblk_set(ext2_filsys fs, struct ext2_inode *inode, blk64_t b);
1262
1263 /* imager.c */
1264 extern errcode_t ext2fs_image_inode_write(ext2_filsys fs, int fd, int flags);
1265 extern errcode_t ext2fs_image_inode_read(ext2_filsys fs, int fd, int flags);
1266 extern errcode_t ext2fs_image_super_write(ext2_filsys fs, int fd, int flags);
1267 extern errcode_t ext2fs_image_super_read(ext2_filsys fs, int fd, int flags);
1268 extern errcode_t ext2fs_image_bitmap_write(ext2_filsys fs, int fd, int flags);
1269 extern errcode_t ext2fs_image_bitmap_read(ext2_filsys fs, int fd, int flags);
1270
1271 /* ind_block.c */
1272 errcode_t ext2fs_read_ind_block(ext2_filsys fs, blk_t blk, void *buf);
1273 errcode_t ext2fs_write_ind_block(ext2_filsys fs, blk_t blk, void *buf);
1274
1275 /* initialize.c */
1276 extern errcode_t ext2fs_initialize(const char *name, int flags,
1277                                    struct ext2_super_block *param,
1278                                    io_manager manager, ext2_filsys *ret_fs);
1279
1280 /* icount.c */
1281 extern void ext2fs_free_icount(ext2_icount_t icount);
1282 extern int ext2fs_icount_is_set(ext2_icount_t icount, ext2_ino_t ino);
1283 extern errcode_t ext2fs_create_icount_tdb(ext2_filsys fs, char *tdb_dir,
1284                                           int flags, ext2_icount_t *ret);
1285 extern errcode_t ext2fs_create_icount2(ext2_filsys fs, int flags,
1286                                        unsigned int size,
1287                                        ext2_icount_t hint, ext2_icount_t *ret);
1288 extern errcode_t ext2fs_create_icount(ext2_filsys fs, int flags,
1289                                       unsigned int size,
1290                                       ext2_icount_t *ret);
1291 extern errcode_t ext2fs_icount_fetch(ext2_icount_t icount, ext2_ino_t ino,
1292                                      __u16 *ret);
1293 extern errcode_t ext2fs_icount_increment(ext2_icount_t icount, ext2_ino_t ino,
1294                                          __u16 *ret);
1295 extern errcode_t ext2fs_icount_decrement(ext2_icount_t icount, ext2_ino_t ino,
1296                                          __u16 *ret);
1297 extern errcode_t ext2fs_icount_store(ext2_icount_t icount, ext2_ino_t ino,
1298                                      __u16 count);
1299 extern ext2_ino_t ext2fs_get_icount_size(ext2_icount_t icount);
1300 errcode_t ext2fs_icount_validate(ext2_icount_t icount, FILE *);
1301
1302 /* inline.c */
1303
1304 extern errcode_t ext2fs_get_memalign(unsigned long size,
1305                                      unsigned long align, void *ptr);
1306
1307 /* inode.c */
1308 extern errcode_t ext2fs_flush_icache(ext2_filsys fs);
1309 extern errcode_t ext2fs_get_next_inode_full(ext2_inode_scan scan,
1310                                             ext2_ino_t *ino,
1311                                             struct ext2_inode *inode,
1312                                             int bufsize);
1313 extern errcode_t ext2fs_open_inode_scan(ext2_filsys fs, int buffer_blocks,
1314                                   ext2_inode_scan *ret_scan);
1315 extern void ext2fs_close_inode_scan(ext2_inode_scan scan);
1316 extern errcode_t ext2fs_get_next_inode(ext2_inode_scan scan, ext2_ino_t *ino,
1317                                struct ext2_inode *inode);
1318 extern errcode_t ext2fs_inode_scan_goto_blockgroup(ext2_inode_scan scan,
1319                                                    int  group);
1320 extern void ext2fs_set_inode_callback
1321         (ext2_inode_scan scan,
1322          errcode_t (*done_group)(ext2_filsys fs,
1323                                  ext2_inode_scan scan,
1324                                  dgrp_t group,
1325                                  void * priv_data),
1326          void *done_group_data);
1327 extern int ext2fs_inode_scan_flags(ext2_inode_scan scan, int set_flags,
1328                                    int clear_flags);
1329 extern errcode_t ext2fs_read_inode_full(ext2_filsys fs, ext2_ino_t ino,
1330                                         struct ext2_inode * inode,
1331                                         int bufsize);
1332 extern errcode_t ext2fs_read_inode (ext2_filsys fs, ext2_ino_t ino,
1333                             struct ext2_inode * inode);
1334 extern errcode_t ext2fs_write_inode_full(ext2_filsys fs, ext2_ino_t ino,
1335                                          struct ext2_inode * inode,
1336                                          int bufsize);
1337 extern errcode_t ext2fs_write_inode(ext2_filsys fs, ext2_ino_t ino,
1338                             struct ext2_inode * inode);
1339 extern errcode_t ext2fs_write_new_inode(ext2_filsys fs, ext2_ino_t ino,
1340                             struct ext2_inode * inode);
1341 extern errcode_t ext2fs_get_blocks(ext2_filsys fs, ext2_ino_t ino, blk_t *blocks);
1342 extern errcode_t ext2fs_check_directory(ext2_filsys fs, ext2_ino_t ino);
1343
1344 /* inode_io.c */
1345 extern io_manager inode_io_manager;
1346 extern errcode_t ext2fs_inode_io_intern(ext2_filsys fs, ext2_ino_t ino,
1347                                         char **name);
1348 extern errcode_t ext2fs_inode_io_intern2(ext2_filsys fs, ext2_ino_t ino,
1349                                          struct ext2_inode *inode,
1350                                          char **name);
1351
1352 /* ismounted.c */
1353 extern errcode_t ext2fs_check_if_mounted(const char *file, int *mount_flags);
1354 extern errcode_t ext2fs_check_mount_point(const char *device, int *mount_flags,
1355                                           char *mtpt, int mtlen);
1356
1357 /* punch.c */
1358 /*
1359  * NOTE: This function removes from an inode the blocks "start", "end", and
1360  * every block in between.
1361  */
1362 extern errcode_t ext2fs_punch(ext2_filsys fs, ext2_ino_t ino,
1363                               struct ext2_inode *inode,
1364                               char *block_buf, blk64_t start,
1365                               blk64_t end);
1366
1367 /* namei.c */
1368 extern errcode_t ext2fs_lookup(ext2_filsys fs, ext2_ino_t dir, const char *name,
1369                          int namelen, char *buf, ext2_ino_t *inode);
1370 extern errcode_t ext2fs_namei(ext2_filsys fs, ext2_ino_t root, ext2_ino_t cwd,
1371                         const char *name, ext2_ino_t *inode);
1372 errcode_t ext2fs_namei_follow(ext2_filsys fs, ext2_ino_t root, ext2_ino_t cwd,
1373                               const char *name, ext2_ino_t *inode);
1374 extern errcode_t ext2fs_follow_link(ext2_filsys fs, ext2_ino_t root, ext2_ino_t cwd,
1375                         ext2_ino_t inode, ext2_ino_t *res_inode);
1376
1377 /* native.c */
1378 int ext2fs_native_flag(void);
1379
1380 /* newdir.c */
1381 extern errcode_t ext2fs_new_dir_block(ext2_filsys fs, ext2_ino_t dir_ino,
1382                                 ext2_ino_t parent_ino, char **block);
1383
1384 /* mkdir.c */
1385 extern errcode_t ext2fs_mkdir(ext2_filsys fs, ext2_ino_t parent, ext2_ino_t inum,
1386                               const char *name);
1387
1388 /* mkjournal.c */
1389 extern errcode_t ext2fs_zero_blocks(ext2_filsys fs, blk_t blk, int num,
1390                                     blk_t *ret_blk, int *ret_count);
1391 extern errcode_t ext2fs_zero_blocks2(ext2_filsys fs, blk64_t blk, int num,
1392                                      blk64_t *ret_blk, int *ret_count);
1393 extern errcode_t ext2fs_create_journal_superblock(ext2_filsys fs,
1394                                                   __u32 num_blocks, int flags,
1395                                                   char  **ret_jsb);
1396 extern errcode_t ext2fs_add_journal_device(ext2_filsys fs,
1397                                            ext2_filsys journal_dev);
1398 extern errcode_t ext2fs_add_journal_inode(ext2_filsys fs, blk_t num_blocks,
1399                                           int flags);
1400 extern errcode_t ext2fs_add_journal_inode2(ext2_filsys fs, blk_t num_blocks,
1401                                            blk64_t goal, int flags);
1402 extern int ext2fs_default_journal_size(__u64 num_blocks);
1403 extern int ext2fs_journal_sb_start(int blocksize);
1404
1405 /* openfs.c */
1406 extern errcode_t ext2fs_open(const char *name, int flags, int superblock,
1407                              unsigned int block_size, io_manager manager,
1408                              ext2_filsys *ret_fs);
1409 extern errcode_t ext2fs_open2(const char *name, const char *io_options,
1410                               int flags, int superblock,
1411                               unsigned int block_size, io_manager manager,
1412                               ext2_filsys *ret_fs);
1413 /*
1414  * The dgrp_t argument to these two functions is not actually a group number
1415  * but a block number offset within a group table!  Convert with the formula
1416  * (group_number / groups_per_block).
1417  */
1418 extern blk64_t ext2fs_descriptor_block_loc2(ext2_filsys fs,
1419                                         blk64_t group_block, dgrp_t i);
1420 extern blk_t ext2fs_descriptor_block_loc(ext2_filsys fs, blk_t group_block,
1421                                          dgrp_t i);
1422 errcode_t ext2fs_get_data_io(ext2_filsys fs, io_channel *old_io);
1423 errcode_t ext2fs_set_data_io(ext2_filsys fs, io_channel new_io);
1424 errcode_t ext2fs_rewrite_to_io(ext2_filsys fs, io_channel new_io);
1425
1426 /* get_pathname.c */
1427 extern errcode_t ext2fs_get_pathname(ext2_filsys fs, ext2_ino_t dir, ext2_ino_t ino,
1428                                char **name);
1429
1430 /* link.c */
1431 errcode_t ext2fs_link(ext2_filsys fs, ext2_ino_t dir, const char *name,
1432                       ext2_ino_t ino, int flags);
1433 errcode_t ext2fs_unlink(ext2_filsys fs, ext2_ino_t dir, const char *name,
1434                         ext2_ino_t ino, int flags);
1435
1436 /* symlink.c */
1437 errcode_t ext2fs_symlink(ext2_filsys fs, ext2_ino_t parent, ext2_ino_t ino,
1438                          const char *name, char *target);
1439
1440 /* mmp.c */
1441 errcode_t ext2fs_mmp_read(ext2_filsys fs, blk64_t mmp_blk, void *buf);
1442 errcode_t ext2fs_mmp_write(ext2_filsys fs, blk64_t mmp_blk, void *buf);
1443 errcode_t ext2fs_mmp_clear(ext2_filsys fs);
1444 errcode_t ext2fs_mmp_init(ext2_filsys fs);
1445 errcode_t ext2fs_mmp_start(ext2_filsys fs);
1446 errcode_t ext2fs_mmp_update(ext2_filsys fs);
1447 errcode_t ext2fs_mmp_stop(ext2_filsys fs);
1448 unsigned ext2fs_mmp_new_seq(void);
1449
1450 /* read_bb.c */
1451 extern errcode_t ext2fs_read_bb_inode(ext2_filsys fs,
1452                                       ext2_badblocks_list *bb_list);
1453
1454 /* read_bb_file.c */
1455 extern errcode_t ext2fs_read_bb_FILE2(ext2_filsys fs, FILE *f,
1456                                       ext2_badblocks_list *bb_list,
1457                                       void *priv_data,
1458                                       void (*invalid)(ext2_filsys fs,
1459                                                       blk_t blk,
1460                                                       char *badstr,
1461                                                       void *priv_data));
1462 extern errcode_t ext2fs_read_bb_FILE(ext2_filsys fs, FILE *f,
1463                                      ext2_badblocks_list *bb_list,
1464                                      void (*invalid)(ext2_filsys fs,
1465                                                      blk_t blk));
1466
1467 /* res_gdt.c */
1468 extern errcode_t ext2fs_create_resize_inode(ext2_filsys fs);
1469
1470 /* swapfs.c */
1471 extern void ext2fs_swap_ext_attr(char *to, char *from, int bufsize,
1472                                  int has_header);
1473 extern void ext2fs_swap_ext_attr_header(struct ext2_ext_attr_header *to_header,
1474                                         struct ext2_ext_attr_header *from_hdr);
1475 extern void ext2fs_swap_ext_attr_entry(struct ext2_ext_attr_entry *to_entry,
1476                                        struct ext2_ext_attr_entry *from_entry);
1477 extern void ext2fs_swap_super(struct ext2_super_block * super);
1478 extern void ext2fs_swap_group_desc(struct ext2_group_desc *gdp);
1479 extern void ext2fs_swap_group_desc2(ext2_filsys, struct ext2_group_desc *gdp);
1480 extern void ext2fs_swap_inode_full(ext2_filsys fs, struct ext2_inode_large *t,
1481                                    struct ext2_inode_large *f, int hostorder,
1482                                    int bufsize);
1483 extern void ext2fs_swap_inode(ext2_filsys fs,struct ext2_inode *t,
1484                               struct ext2_inode *f, int hostorder);
1485 extern void ext2fs_swap_mmp(struct mmp_struct *mmp);
1486
1487 /* unix_io.c */
1488 extern int ext2fs_open_file(const char *pathname, int flags, mode_t mode);
1489 extern int ext2fs_stat(const char *path, ext2fs_struct_stat *buf);
1490 extern int ext2fs_fstat(int fd, ext2fs_struct_stat *buf);
1491
1492 /* valid_blk.c */
1493 extern int ext2fs_inode_has_valid_blocks(struct ext2_inode *inode);
1494 extern int ext2fs_inode_has_valid_blocks2(ext2_filsys fs,
1495                                           struct ext2_inode *inode);
1496
1497 /* version.c */
1498 extern int ext2fs_parse_version_string(const char *ver_string);
1499 extern int ext2fs_get_library_version(const char **ver_string,
1500                                       const char **date_string);
1501
1502 /* write_bb_file.c */
1503 extern errcode_t ext2fs_write_bb_FILE(ext2_badblocks_list bb_list,
1504                                       unsigned int flags,
1505                                       FILE *f);
1506
1507
1508 /* inline functions */
1509 #ifdef NO_INLINE_FUNCS
1510 extern errcode_t ext2fs_get_mem(unsigned long size, void *ptr);
1511 extern errcode_t ext2fs_get_memzero(unsigned long size, void *ptr);
1512 extern errcode_t ext2fs_get_array(unsigned long count,
1513                                   unsigned long size, void *ptr);
1514 extern errcode_t ext2fs_get_arrayzero(unsigned long count,
1515                                       unsigned long size, void *ptr);
1516 extern errcode_t ext2fs_free_mem(void *ptr);
1517 extern errcode_t ext2fs_resize_mem(unsigned long old_size,
1518                                    unsigned long size, void *ptr);
1519 extern void ext2fs_mark_super_dirty(ext2_filsys fs);
1520 extern void ext2fs_mark_changed(ext2_filsys fs);
1521 extern int ext2fs_test_changed(ext2_filsys fs);
1522 extern void ext2fs_mark_valid(ext2_filsys fs);
1523 extern void ext2fs_unmark_valid(ext2_filsys fs);
1524 extern int ext2fs_test_valid(ext2_filsys fs);
1525 extern void ext2fs_mark_ib_dirty(ext2_filsys fs);
1526 extern void ext2fs_mark_bb_dirty(ext2_filsys fs);
1527 extern int ext2fs_test_ib_dirty(ext2_filsys fs);
1528 extern int ext2fs_test_bb_dirty(ext2_filsys fs);
1529 extern dgrp_t ext2fs_group_of_blk(ext2_filsys fs, blk_t blk);
1530 extern dgrp_t ext2fs_group_of_ino(ext2_filsys fs, ext2_ino_t ino);
1531 extern blk_t ext2fs_group_first_block(ext2_filsys fs, dgrp_t group);
1532 extern blk_t ext2fs_group_last_block(ext2_filsys fs, dgrp_t group);
1533 extern blk_t ext2fs_inode_data_blocks(ext2_filsys fs,
1534                                       struct ext2_inode *inode);
1535 extern unsigned int ext2fs_div_ceil(unsigned int a, unsigned int b);
1536 extern __u64 ext2fs_div64_ceil(__u64 a, __u64 b);
1537 #endif
1538
1539 /*
1540  * The actual inlined functions definitions themselves...
1541  *
1542  * If NO_INLINE_FUNCS is defined, then we won't try to do inline
1543  * functions at all!
1544  */
1545 #if (defined(INCLUDE_INLINE_FUNCS) || !defined(NO_INLINE_FUNCS))
1546 #ifdef INCLUDE_INLINE_FUNCS
1547 #define _INLINE_ extern
1548 #else
1549 #if (__STDC_VERSION__ >= 199901L)
1550 #define _INLINE_ inline
1551 #else
1552 #ifdef __GNUC__
1553 #define _INLINE_ extern __inline__
1554 #else                           /* For Watcom C */
1555 #define _INLINE_ extern inline
1556 #endif /* __GNUC__ */
1557 #endif /* __STDC_VERSION__ >= 199901L */
1558 #endif
1559
1560 #ifndef EXT2_CUSTOM_MEMORY_ROUTINES
1561 #include <string.h>
1562 /*
1563  *  Allocate memory.  The 'ptr' arg must point to a pointer.
1564  */
1565 _INLINE_ errcode_t ext2fs_get_mem(unsigned long size, void *ptr)
1566 {
1567         void *pp;
1568
1569         pp = malloc(size);
1570         if (!pp)
1571                 return EXT2_ET_NO_MEMORY;
1572         memcpy(ptr, &pp, sizeof (pp));
1573         return 0;
1574 }
1575
1576 _INLINE_ errcode_t ext2fs_get_memzero(unsigned long size, void *ptr)
1577 {
1578         void *pp;
1579
1580         pp = malloc(size);
1581         if (!pp)
1582                 return EXT2_ET_NO_MEMORY;
1583         memset(pp, 0, size);
1584         memcpy(ptr, &pp, sizeof(pp));
1585         return 0;
1586 }
1587
1588 _INLINE_ errcode_t ext2fs_get_array(unsigned long count, unsigned long size, void *ptr)
1589 {
1590         if (count && (-1UL)/count<size)
1591                 return EXT2_ET_NO_MEMORY;
1592         return ext2fs_get_mem(count*size, ptr);
1593 }
1594
1595 _INLINE_ errcode_t ext2fs_get_arrayzero(unsigned long count,
1596                                         unsigned long size, void *ptr)
1597 {
1598         void *pp;
1599
1600         if (count && (-1UL)/count<size)
1601                 return EXT2_ET_NO_MEMORY;
1602         pp = calloc(count, size);
1603         if (!pp)
1604                 return EXT2_ET_NO_MEMORY;
1605         memcpy(ptr, &pp, sizeof(pp));
1606         return 0;
1607 }
1608
1609 /*
1610  * Free memory.  The 'ptr' arg must point to a pointer.
1611  */
1612 _INLINE_ errcode_t ext2fs_free_mem(void *ptr)
1613 {
1614         void *p;
1615
1616         memcpy(&p, ptr, sizeof(p));
1617         free(p);
1618         p = 0;
1619         memcpy(ptr, &p, sizeof(p));
1620         return 0;
1621 }
1622
1623 /*
1624  *  Resize memory.  The 'ptr' arg must point to a pointer.
1625  */
1626 _INLINE_ errcode_t ext2fs_resize_mem(unsigned long EXT2FS_ATTR((unused)) old_size,
1627                                      unsigned long size, void *ptr)
1628 {
1629         void *p;
1630
1631         /* Use "memcpy" for pointer assignments here to avoid problems
1632          * with C99 strict type aliasing rules. */
1633         memcpy(&p, ptr, sizeof(p));
1634         p = realloc(p, size);
1635         if (!p)
1636                 return EXT2_ET_NO_MEMORY;
1637         memcpy(ptr, &p, sizeof(p));
1638         return 0;
1639 }
1640 #endif  /* Custom memory routines */
1641
1642 /*
1643  * Mark a filesystem superblock as dirty
1644  */
1645 _INLINE_ void ext2fs_mark_super_dirty(ext2_filsys fs)
1646 {
1647         fs->flags |= EXT2_FLAG_DIRTY | EXT2_FLAG_CHANGED;
1648 }
1649
1650 /*
1651  * Mark a filesystem as changed
1652  */
1653 _INLINE_ void ext2fs_mark_changed(ext2_filsys fs)
1654 {
1655         fs->flags |= EXT2_FLAG_CHANGED;
1656 }
1657
1658 /*
1659  * Check to see if a filesystem has changed
1660  */
1661 _INLINE_ int ext2fs_test_changed(ext2_filsys fs)
1662 {
1663         return (fs->flags & EXT2_FLAG_CHANGED);
1664 }
1665
1666 /*
1667  * Mark a filesystem as valid
1668  */
1669 _INLINE_ void ext2fs_mark_valid(ext2_filsys fs)
1670 {
1671         fs->flags |= EXT2_FLAG_VALID;
1672 }
1673
1674 /*
1675  * Mark a filesystem as NOT valid
1676  */
1677 _INLINE_ void ext2fs_unmark_valid(ext2_filsys fs)
1678 {
1679         fs->flags &= ~EXT2_FLAG_VALID;
1680 }
1681
1682 /*
1683  * Check to see if a filesystem is valid
1684  */
1685 _INLINE_ int ext2fs_test_valid(ext2_filsys fs)
1686 {
1687         return (fs->flags & EXT2_FLAG_VALID);
1688 }
1689
1690 /*
1691  * Mark the inode bitmap as dirty
1692  */
1693 _INLINE_ void ext2fs_mark_ib_dirty(ext2_filsys fs)
1694 {
1695         fs->flags |= EXT2_FLAG_IB_DIRTY | EXT2_FLAG_CHANGED;
1696 }
1697
1698 /*
1699  * Mark the block bitmap as dirty
1700  */
1701 _INLINE_ void ext2fs_mark_bb_dirty(ext2_filsys fs)
1702 {
1703         fs->flags |= EXT2_FLAG_BB_DIRTY | EXT2_FLAG_CHANGED;
1704 }
1705
1706 /*
1707  * Check to see if a filesystem's inode bitmap is dirty
1708  */
1709 _INLINE_ int ext2fs_test_ib_dirty(ext2_filsys fs)
1710 {
1711         return (fs->flags & EXT2_FLAG_IB_DIRTY);
1712 }
1713
1714 /*
1715  * Check to see if a filesystem's block bitmap is dirty
1716  */
1717 _INLINE_ int ext2fs_test_bb_dirty(ext2_filsys fs)
1718 {
1719         return (fs->flags & EXT2_FLAG_BB_DIRTY);
1720 }
1721
1722 /*
1723  * Return the group # of a block
1724  */
1725 _INLINE_ dgrp_t ext2fs_group_of_blk(ext2_filsys fs, blk_t blk)
1726 {
1727         return ext2fs_group_of_blk2(fs, blk);
1728 }
1729 /*
1730  * Return the group # of an inode number
1731  */
1732 _INLINE_ dgrp_t ext2fs_group_of_ino(ext2_filsys fs, ext2_ino_t ino)
1733 {
1734         return (ino - 1) / fs->super->s_inodes_per_group;
1735 }
1736
1737 /*
1738  * Return the first block (inclusive) in a group
1739  */
1740 _INLINE_ blk_t ext2fs_group_first_block(ext2_filsys fs, dgrp_t group)
1741 {
1742         return (blk_t) ext2fs_group_first_block2(fs, group);
1743 }
1744
1745 /*
1746  * Return the last block (inclusive) in a group
1747  */
1748 _INLINE_ blk_t ext2fs_group_last_block(ext2_filsys fs, dgrp_t group)
1749 {
1750         return (blk_t) ext2fs_group_last_block2(fs, group);
1751 }
1752
1753 _INLINE_ blk_t ext2fs_inode_data_blocks(ext2_filsys fs,
1754                                         struct ext2_inode *inode)
1755 {
1756         return (blk_t) ext2fs_inode_data_blocks2(fs, inode);
1757 }
1758
1759 /*
1760  * This is an efficient, overflow safe way of calculating ceil((1.0 * a) / b)
1761  */
1762 _INLINE_ unsigned int ext2fs_div_ceil(unsigned int a, unsigned int b)
1763 {
1764         if (!a)
1765                 return 0;
1766         return ((a - 1) / b) + 1;
1767 }
1768
1769 _INLINE_ __u64 ext2fs_div64_ceil(__u64 a, __u64 b)
1770 {
1771         if (!a)
1772                 return 0;
1773         return ((a - 1) / b) + 1;
1774 }
1775
1776 #undef _INLINE_
1777 #endif
1778
1779 #ifdef __cplusplus
1780 }
1781 #endif
1782
1783 #endif /* _EXT2FS_EXT2FS_H */