Hash :
b4913eb1
Author :
Thomas de Grivel
Date :
2025-07-22T23:18:41
wip crc32c
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443
/* ext4fs
* Copyright 2025 kmx.io <contact@kmx.io>
*
* Permission is hereby granted to use this software granted the above
* copyright notice and this permission paragraph are included in all
* copies and substantial portions of this software.
*
* THIS SOFTWARE IS PROVIDED "AS-IS" WITHOUT ANY GUARANTEE OF
* PURPOSE AND PERFORMANCE. IN NO EVENT WHATSOEVER SHALL THE
* AUTHOR BE CONSIDERED LIABLE FOR THE USE AND PERFORMANCE OF
* THIS SOFTWARE.
*/
#ifndef EXT4FS_H
#define EXT4FS_H
#include <stddef.h>
#include <stdint.h>
#include <time.h>
typedef struct {
uint32_t value;
const char *name;
} s_value_name;
#define EXT4FS_FUNCTION_MAX 32
#define EXT4FS_REV_DYNAMIC 1
#define EXT4FS_REV_MINOR 0
#define EXT4FS_LAST_MOUNTED_MAX 64
#define EXT4FS_MAGIC 0xEF53
#define EXT4FS_MOUNT_OPTS_MAX 64
#define EXT4FS_SUPER_BLOCK_OFFSET 1024
#define EXT4FS_SUPER_BLOCK_SIZE 1024
#define EXT4FS_VOLUME_NAME_MAX 16
#define EXT4FS_CHECKSUM_TYPE_NONE 0x0000
#define EXT4FS_CHECKSUM_TYPE_CRC32C 0x0001
extern const s_value_name ext4fs_checksum_type_names[];
#define EXT4FS_ENCODING_NONE 0x0000 // No encoding: legacy behavior
#define EXT4FS_ENCODING_UTF8 0x0001 // UTF-8 encoding, Unicode 12.1.0
extern const s_value_name ext4fs_encoding_names[];
#define EXT4FS_ENCODING_FLAG_NONE 0x0000
#define EXT4FS_ENCODING_FLAG_STRICT_MODE 0x0001 /* Reject invalid encoding */
extern const s_value_name ext4fs_encoding_flag_names[];
#define EXT4FS_ERRORS_CONTINUE 1 // Log and keep going
#define EXT4FS_ERRORS_RO 2 // Remount read-only
#define EXT4FS_ERRORS_PANIC 3 // Kernel panic
extern const char *ext4fs_error_policy_names[];
#define EXT4FS_FEATURE_COMPAT_DIR_PREALLOC 0x0001
#define EXT4FS_FEATURE_COMPAT_IMAGIC_INODES 0x0002
#define EXT4FS_FEATURE_COMPAT_HAS_JOURNAL 0x0004
#define EXT4FS_FEATURE_COMPAT_EXT_ATTR 0x0008
#define EXT4FS_FEATURE_COMPAT_RESIZE_INODE 0x0010
#define EXT4FS_FEATURE_COMPAT_DIR_INDEX 0x0020
extern const s_value_name ext4fs_feature_compat_names[];
#define EXT4FS_FEATURE_INCOMPAT_COMPRESSION 0x00001 // Not used
#define EXT4FS_FEATURE_INCOMPAT_FILETYPE 0x00002 // ftype in dirent
#define EXT4FS_FEATURE_INCOMPAT_RECOVER 0x00004 // needs journal recovery
#define EXT4FS_FEATURE_INCOMPAT_JOURNAL_DEV 0x00008 // External journal device
#define EXT4FS_FEATURE_INCOMPAT_META_BG 0x00010 // Meta block groups
#define EXT4FS_FEATURE_INCOMPAT_EXTENTS 0x00040 // Extents instead of block map
#define EXT4FS_FEATURE_INCOMPAT_64BIT 0x00080 // 64-bit block numbers
#define EXT4FS_FEATURE_INCOMPAT_MMP 0x00100 // Multiple mount protection
#define EXT4FS_FEATURE_INCOMPAT_FLEX_BG 0x00200 // Flexible block groups
#define EXT4FS_FEATURE_INCOMPAT_EA_INODE 0x00400 // In-inode xattrs
#define EXT4FS_FEATURE_INCOMPAT_DIRDATA 0x01000 // Inline dir data
#define EXT4FS_FEATURE_INCOMPAT_CSUM_SEED 0x02000 // Metadata checksum seed
#define EXT4FS_FEATURE_INCOMPAT_LARGEDIR 0x04000 // Large dir support
#define EXT4FS_FEATURE_INCOMPAT_INLINE_DATA 0x08000 // Inline file data
#define EXT4FS_FEATURE_INCOMPAT_ENCRYPT 0x10000 // File encryption
extern const s_value_name ext4fs_feature_incompat_names[];
#define EXT4FS_FEATURE_RO_COMPAT_SPARSE_SUPER 0x0001
#define EXT4FS_FEATURE_RO_COMPAT_LARGE_FILE 0x0002
#define EXT4FS_FEATURE_RO_COMPAT_BTREE_DIR 0x0004 // Not used in ext4
#define EXT4FS_FEATURE_RO_COMPAT_HUGE_FILE 0x0008
#define EXT4FS_FEATURE_RO_COMPAT_GDT_CSUM 0x0010
#define EXT4FS_FEATURE_RO_COMPAT_DIR_NLINK 0x0020
#define EXT4FS_FEATURE_RO_COMPAT_EXTRA_ISIZE 0x0040
#define EXT4FS_FEATURE_RO_COMPAT_HAS_SNAPSHOT 0x0080 // Unused
#define EXT4FS_FEATURE_RO_COMPAT_QUOTA 0x0100
#define EXT4FS_FEATURE_RO_COMPAT_BIGALLOC 0x0200
#define EXT4FS_FEATURE_RO_COMPAT_METADATA_CSUM 0x0400
#define EXT4FS_FEATURE_RO_COMPAT_REPLICA 0x0800 // Unused
#define EXT4FS_FEATURE_RO_COMPAT_READONLY 0x1000
#define EXT4FS_FEATURE_RO_COMPAT_PROJECT 0x2000
extern const s_value_name ext4fs_feature_ro_compat_names[];
#define EXT4FS_FLAG_SIGNED_HASH 0x0001 // Legacy hash uses signed ints
#define EXT4FS_FLAG_UNSIGNED_HASH 0x0002 // Default: unsigned directory hash
#define EXT4FS_FLAG_TEST_FILESYS 0x0004 // Marked as a test filesystem
#define EXT4FS_FLAG_64BIT 0x0008 // Uses 64-bit block numbers
#define EXT4FS_FLAG_MOUNT_OPT_CHECK 0x0010 // Kernel must check mount opts
extern const s_value_name ext4fs_flag_names[];
#define EXT4FS_MOUNT_READONLY 0x0001
#define EXT4FS_MOUNT_NO_ATIME 0x0002
#define EXT4FS_MOUNT_DIRSYNC 0x0004
#define EXT4FS_MOUNT_DATA_JOURNAL 0x0008
#define EXT4FS_MOUNT_DATA_ORDERED 0x0010
#define EXT4FS_MOUNT_DATA_WRITEBACK 0x0020
#define EXT4FS_MOUNT_ERRORS_CONTINUE 0x0040
#define EXT4FS_MOUNT_ERRORS_REMOUNT_RO 0x0080
#define EXT4FS_MOUNT_ERRORS_PANIC 0x0100
#define EXT4FS_MOUNT_DISCARD 0x0200
#define EXT4FS_MOUNT_NO_BUFFER_HEADS 0x0400
#define EXT4FS_MOUNT_SKIP_JOURNAL 0x0800
#define EXT4FS_MOUNT_NOAUTO_DELAYED_ALLOC 0x1000
extern const s_value_name ext4fs_mount_names[];
#define EXT4FS_OS_LINUX 0
#define EXT4FS_OS_HURD 1
#define EXT4FS_OS_MASIX 2
#define EXT4FS_OS_FREEBSD 3
#define EXT4FS_OS_LITES 4
#define EXT4FS_OS_OPENBSD 5
extern const char *ext4fs_os_str[];
#define EXT4FS_STATE_VALID 0x0001 // Clean unmount
#define EXT4FS_STATE_ERROR 0x0002 // Errors detected (fsck needed)
extern const s_value_name ext4fs_state_names[];
#define EXT4FS_BGD_FLAG_INODE_UNINIT 0x0001
#define EXT4FS_BGD_FLAG_BLOCK_UNINIT 0x0002
#define EXT4FS_BGD_FLAG_INODE_ZEROED 0x0004
#define EXT4FS_BGD_FLAG_DIRTY 0x0008
#define EXT4FS_BGD_FLAG_BLOCK_ZEROED 0x0010
#define EXT4FS_BGD_FLAG_READ_ONLY 0x0020
extern const s_value_name ext4fs_bgd_flag_names[];
struct ext4fs_super_block {
uint32_t sb_inodes_count;
uint32_t sb_blocks_count_lo;
uint32_t sb_reserved_blocks_count_lo;
uint32_t sb_free_blocks_count_lo;
// 0x10
uint32_t sb_free_inodes_count;
uint32_t sb_first_data_block;
uint32_t sb_log_block_size; // log2(block size) - 10
uint32_t sb_log_cluster_size; // log2(cluster size) - 10
// 0x20
uint32_t sb_blocks_per_group;
uint32_t sb_clusters_per_group;
uint32_t sb_inodes_per_group;
uint32_t sb_mount_time_lo;
// 0x30
uint32_t sb_write_time_lo;
uint16_t sb_mount_count;
int16_t sb_max_mount_count_before_fsck;
uint16_t sb_magic;
uint16_t sb_state; // EXT4FS_STATE_*
uint16_t sb_errors; // EXT4FS_ERRORS_*
uint16_t sb_revision_level_minor;
// 0x40
uint32_t sb_check_time_lo;
uint32_t sb_check_interval;
uint32_t sb_creator_os; // EXT4FS_OS_*
uint32_t sb_revision_level;
// 0x50
uint16_t sb_default_reserved_uid;
uint16_t sb_default_reserved_gid;
uint32_t sb_first_non_reserved_inode;
uint16_t sb_inode_size;
uint16_t sb_block_group_id;
uint32_t sb_feature_compat;
// 0x60
uint32_t sb_feature_incompat;
uint32_t sb_feature_ro_compat;
uint8_t sb_uuid[16];
char sb_volume_name[EXT4FS_VOLUME_NAME_MAX];
char sb_last_mounted[EXT4FS_LAST_MOUNTED_MAX];
uint32_t sb_algorithm_usage_bitmap;
uint8_t sb_preallocate_blocks;
uint8_t sb_preallocate_dir_blocks;
uint16_t sb_reserved_bgdt_blocks;
// 0xD0
uint8_t sb_journal_uuid[16]; // UUID of journal superblock
// 0xE0
uint32_t sb_journal_inode_number;
uint32_t sb_journal_device_number;
uint32_t sb_last_orphan;
uint32_t sb_hash_seed[4];
uint8_t sb_default_hash_version;
uint8_t sb_journal_backup_type;
uint16_t sb_block_group_descriptor_size;
// 0x100
uint32_t sb_default_mount_opts;
uint32_t sb_first_meta_block_group;
uint32_t sb_newfs_time_lo;
uint32_t sb_jnl_blocks[17]; // Backup of journal inode
// 0x150
uint32_t sb_blocks_count_hi;
uint32_t sb_reserved_blocks_count_hi;
uint32_t sb_free_blocks_count_hi;
uint16_t sb_inode_size_extra_min;
uint16_t sb_inode_size_extra_want;
// 0x160
uint32_t sb_flags;
uint16_t sb_raid_stride_block_count;
uint16_t sb_mmp_interval;
uint64_t sb_mmp_block;
// 0x170
uint32_t sb_raid_stripe_width_block_count;
uint8_t sb_log_groups_per_flex;
uint8_t sb_checksum_type;
uint16_t sb_reserved_176;
uint64_t sb_kilobytes_written;
// 0x180
uint32_t sb_ext3_snapshot_inode;
uint32_t sb_ext3_snapshot_id;
uint64_t sb_ext3_snapshot_reserved_blocks_count;
// 0x190
uint32_t sb_ext3_snapshot_list;
uint32_t sb_error_count;
uint32_t sb_first_error_time_lo;
uint32_t sb_first_error_inode;
// 0x1A0
uint64_t sb_first_error_block;
char sb_first_error_function[EXT4FS_FUNCTION_MAX];
uint32_t sb_first_error_line;
uint32_t sb_last_error_time_lo;
// 0x1D0
uint32_t sb_last_error_inode;
uint32_t sb_last_error_line;
uint64_t sb_last_error_block;
// 0x1E0
char sb_last_error_function[EXT4FS_FUNCTION_MAX];
// 0x200
char sb_mount_opts[EXT4FS_MOUNT_OPTS_MAX];
// 0x240
uint32_t sb_user_quota_inode;
uint32_t sb_group_quota_inode;
uint32_t sb_overhead_clusters;
uint32_t sb_backup_block_groups[2];
uint8_t sb_encrypt_algos[4];
uint8_t sb_encrypt_pw_salt[16];
uint32_t sb_lost_and_found_inode;
uint32_t sb_project_quota_inode;
// 0x270
uint32_t sb_checksum_seed;
uint8_t sb_write_time_hi;
uint8_t sb_mount_time_hi;
uint8_t sb_newfs_time_hi;
uint8_t sb_check_time_hi;
uint8_t sb_first_error_time_hi;
uint8_t sb_last_error_time_hi;
uint8_t sb_first_error_code;
uint8_t sb_last_error_code;
uint16_t sb_encoding;
uint16_t sb_encoding_flags;
// 0x280
uint16_t sb_orphan_file_inode;
uint16_t sb_reserved_284;
uint32_t sb_reserved_288[94];
uint32_t sb_checksum;
} __attribute__((packed));
struct ext4fs_block_group_descriptor {
uint32_t bgd_block_bitmap_block_lo;
uint32_t bgd_inode_bitmap_block_lo;
uint32_t bgd_inode_table_block_lo;
uint16_t bgd_free_blocks_count_lo;
uint16_t bgd_free_inodes_count_lo;
// 0x10
uint16_t bgd_used_dirs_count_lo;
uint16_t bgd_flags;
uint32_t bgd_exclude_bitmap_block_lo;
uint16_t bgd_block_bitmap_checksum_lo;
uint16_t bgd_inode_bitmap_checksum_lo;
uint16_t bgd_inode_table_unused_lo;
uint16_t bgd_checksum;
// 0x20
uint32_t bgd_block_bitmap_block_hi;
uint32_t bgd_inode_bitmap_block_hi;
uint32_t bgd_inode_table_block_hi;
uint16_t bgd_free_blocks_count_hi;
uint16_t bgd_free_inodes_count_hi;
// 0x30
uint16_t bgd_used_dirs_count_hi;
uint16_t bgd_inode_table_unused_hi;
uint32_t bgd_exclude_bitmap_block_hi;
uint16_t bgd_block_bitmap_checksum_hi;
uint16_t bgd_inode_bitmap_checksum_hi;
uint32_t bgd_reserved_3c;
// 0x40
} __attribute__((packed));
int
ext4fs_bgd_block_bitmap_block
(const struct ext4fs_super_block *sb,
const struct ext4fs_block_group_descriptor *bgd,
uint64_t *dest);
int
ext4fs_bgd_block_bitmap_checksum
(const struct ext4fs_super_block *sb,
const struct ext4fs_block_group_descriptor *bgd,
uint32_t *dest);
int
ext4fs_bgd_checksum_compute
(const struct ext4fs_super_block *sb,
const struct ext4fs_block_group_descriptor *bgd,
uint32_t block_group_id, uint16_t *dest);
int
ext4fs_bgd_exclude_bitmap_block
(const struct ext4fs_super_block *sb,
const struct ext4fs_block_group_descriptor *bgd,
uint64_t *dest);
int
ext4fs_bgd_free_blocks_count
(const struct ext4fs_super_block *sb,
const struct ext4fs_block_group_descriptor *bgd,
uint32_t *dest);
int
ext4fs_bgd_free_inodes_count
(const struct ext4fs_super_block *sb,
const struct ext4fs_block_group_descriptor *bgd,
uint32_t *dest);
int
ext4fs_bgd_inode_bitmap_block
(const struct ext4fs_super_block *sb,
const struct ext4fs_block_group_descriptor *bgd,
uint64_t *dest);
int
ext4fs_bgd_inode_bitmap_checksum
(const struct ext4fs_super_block *sb,
const struct ext4fs_block_group_descriptor *bgd,
uint32_t *dest);
int
ext4fs_bgd_inode_table_block
(const struct ext4fs_super_block *sb,
const struct ext4fs_block_group_descriptor *bgd,
uint64_t *dest);
int
ext4fs_bgd_inode_table_unused
(const struct ext4fs_super_block *sb,
const struct ext4fs_block_group_descriptor *bgd,
uint32_t *dest);
int
ext4fs_bgd_used_dirs_count
(const struct ext4fs_super_block *sb,
const struct ext4fs_block_group_descriptor *bgd,
uint32_t *dest);
#ifdef OpenBSD
struct disklabel *
ext4fs_disklabel_get
(struct disklabel *dl, int fd);
#endif // OpenBSD
struct ext4fs_block_group_descriptor *
ext4fs_block_group_descriptor_read
(struct ext4fs_block_group_descriptor *bgd, int fd,
const struct ext4fs_super_block *sb);
int ext4fs_inspect (const char *dev, int fd);
int
ext4fs_inspect_block_group_descriptor
(const struct ext4fs_super_block *sb,
const struct ext4fs_block_group_descriptor *bgd,
uint32_t block_group_id);
int
ext4fs_inspect_block_group_descriptor_hex
(const struct ext4fs_super_block *sb,
const struct ext4fs_block_group_descriptor *bgd);
void ext4fs_inspect_creator_os (uint16_t creator_os);
void ext4fs_inspect_errors (uint16_t errors);
int ext4fs_inspect_flags_names (uint32_t flags,
const s_value_name *names);
int ext4fs_inspect_super_block (const struct ext4fs_super_block *sb);
int ext4fs_sb_last_error_time (const struct ext4fs_super_block *sb,
uint64_t *dest);
int ext4fs_sb_block_size (const struct ext4fs_super_block *sb,
uint32_t *dest);
int ext4fs_sb_blocks_count (const struct ext4fs_super_block *sb,
uint64_t *dest);
int ext4fs_sb_check_time (const struct ext4fs_super_block *sb,
uint64_t *dest);
int ext4fs_sb_first_error_time (const struct ext4fs_super_block *sb,
uint64_t *dest);
int ext4fs_sb_free_blocks_count (const struct ext4fs_super_block *sb,
uint64_t *dest);
int ext4fs_sb_mount_time (const struct ext4fs_super_block *sb,
uint64_t *dest);
int ext4fs_sb_newfs_time (const struct ext4fs_super_block *sb,
uint64_t *dest);
int ext4fs_sb_reserved_blocks_count (const struct ext4fs_super_block *sb,
uint64_t *dest);
int ext4fs_sb_write_time (const struct ext4fs_super_block *sb,
uint64_t *dest);
int ext4fs_size (const char *dev, int fd, uint64_t *dest);
struct ext4fs_super_block *
ext4fs_super_block_read (struct ext4fs_super_block *sb,
int fd);
int ext4fs_time_to_str (time_t time, char *str, size_t size);
#endif /* EXT4FS_H */