/* * Copyright (C) 2011 Andrea Mazzoleni * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #ifndef __ELEM_H #define __ELEM_H #include "util.h" #include "support.h" #include "tommyds/tommyhash.h" #include "tommyds/tommylist.h" #include "tommyds/tommytree.h" #include "tommyds/tommyhashdyn.h" #include "tommyds/tommyarray.h" #include "tommyds/tommyarrayblkof.h" /****************************************************************************/ /* snapraid */ /** * Number of measures of the operation progress. */ #define PROGRESS_MAX 100 /** * Max UUID length. */ #define UUID_MAX 128 /** * Invalid position. */ #define POS_NULL ((block_off_t)-1) /** * Content file specification. */ struct snapraid_content { char content[PATH_MAX]; /**< Path of the content file. */ uint64_t device; /**< Device identifier. */ void* context; /**< Context used for multithread operations. */ tommy_node node; /**< Next node in the list. */ }; /** * Filter for paths. */ struct snapraid_filter { char pattern[PATH_MAX]; /**< Filter pattern. */ int is_disk; /**< If the pattern is a disk one. */ int is_path; /**< If the pattern is only for the complete path. */ int is_dir; /**< If the pattern is only for dir. */ int direction; /**< If it's an inclusion (=1) or an exclusion (=-1). */ tommy_node node; /**< Next node in the list. */ }; /** * Block pointer used to represent unused blocks. */ #define BLOCK_NULL 0 /** * This block is an empty one. * Note that an empty block is represent with ::BLOCK_NULL. */ #define BLOCK_STATE_EMPTY 0 /** * The block has both the hash and the parity computed. * This is the normal state of a saved block. * * The block hash field IS set. * The parity for this disk is updated. */ #define BLOCK_STATE_BLK 1 /** * The block is new and not yet hashed. * This happens when a new block overwrite a just removed block, or an empty space. * * The block hash field MAY be set and it represents the hash of the OLD data. * The hash may be also INVALID or ZERO. * * If the OLD block was empty, the hash is set to the special ZERO value. * If the OLD block was lost, the hash is set to the special INVALID value. * * The parity for this disk is not updated, but it contains the old data referenced by the hash. * * If the state is read from an incomplete sync, we don't really know if the hash is referring at the * data used to compute the parity, because the sync process was interrupted at an unknown point, * and the parity may or may not be updated. * * For this reason we clear all such hashes when reading the state from an incomplete sync before * starting a new sync, because sync is affected by such hashes, as sync updates the parity, only * if the new data read for CHG blocks has a mismatching hash. * Clearing is done setting the ::clear_past_hash flag before reading the state. * No clearing is done in other commands, as check and fix are instead able to work with unsynced * hashes, and scrub ignores CHG/DELETED blocks. */ #define BLOCK_STATE_CHG 2 /** * The block is new and hashed. * This happens when a new block overwrite a just removed block, or an empty space. * * Note that when the file copy heuristic is enabled, the REP blocks may be set * using this heuristic, meaning that the hash may be wrong. * * For this reason, when the ::force_nocopy flag is enabled in sync, we convert all the REP blocks * to CHG, invalidating the stored hash. * Clearing is done setting the ::clear_past_hash flag before reading the state. * No clearing is done in other commands, as they don't stop the process like in sync * when there is a false silent error. * * The block hash field IS set, and it represents the hash of the new data. * The parity for this disk is not updated. */ #define BLOCK_STATE_REP 3 /** * This block is a deleted one. * This happens when a file is deleted. * * The block hash field IS set, and it represents the hash of the previous data, * but only if it's different by all 0. * The parity for this disk is not updated, but it contains the old data referenced by the hash. * * If the state is read from an incomplete sync, we don't really know if the hash is referring at the * data used to compute the parity, because the sync process was interrupted at an unknown point, * and the parity may or may not be updated. * * A now the sync process is not affected by DELETED hash, so clearing won't be really needed, * but considering that we have to do it for CHG blocks, we do it also for DELETED ones, * clearing all the past hashes. * Clearing is done setting the ::clear_past_hash flag before reading the state. * No clearing is done in other commands, as check and fix are instead able to work with unsynced * hashes, and scrub ignores CHG/DELETED blocks. */ #define BLOCK_STATE_DELETED 4 /** * Block hash size. * * At max HASH_MAX. */ extern int BLOCK_HASH_SIZE; /** * Block of a file. */ struct snapraid_block { unsigned char state; /**< State of the block. */ /** * Hash of the block. * * The effective stored size is BLOCK_HASH_SIZE. */ unsigned char hash[HASH_MAX]; }; /** * If a file is present in the disk. * It's used only in scan to detect present and missing files. */ #define FILE_IS_PRESENT 0x01 /** * If it's an excluded file from the processing. * It's used in both check and fix to mark files to exclude from the processing. */ #define FILE_IS_EXCLUDED 0x02 /** * If a fix was attempted but it failed. * It's used only in fix to mark that some data is unrecoverable. */ #define FILE_IS_DAMAGED 0x04 /** * If a fix was done. * It's used only in fix to mark that some data was recovered. */ #define FILE_IS_FIXED 0x08 /** * If the file was originally missing, and it was created in the fix process. * It's used only in fix to mark files recovered from scratch, * meaning that they don't have any previous content. * This is important because it means that deleting them, you are not going * to lose something that cannot be recovered. * Note that excluded files won't ever get this flag. */ #define FILE_IS_CREATED 0x10 /** * If the file has completed its processing, meaning that it won't be opened anymore. * It's used only in fix to mark when we finish processing one file. * Note that excluded files won't ever get this flag. */ #define FILE_IS_FINISHED 0x20 /** * If the file hash was obtained from a file copy * identified by the same name, size and stamp. */ #define FILE_IS_COPY 0x40 /** * If the file was opened. * It's used in fix to detect if it's the first time a file is opened. */ #define FILE_IS_OPENED 0x80 /** * If the file is modified from the latest sync. * It's used in fix to store if the state of the file before being modified. */ #define FILE_IS_UNSYNCED 0x100 /** * If the file is without inode. * It could happen in file-system where inodes are not persistent, * or when restoring a full disk with "fix". * In such cases we have to clear any stored duplicate inode. * After the scan process completes, no file should have this flag set. */ #define FILE_IS_WITHOUT_INODE 0x200 /** * The file is deleted. * This happens when a file is deleted from the array, * but it's keep inside the parity until the next sync. * * During the file-system check we needs this information, * because deleted files may be present only partially. */ #define FILE_IS_DELETED 0x400 /** * The file is missing. * This happens in fix/check when a file is cannot be opened, * and marking it as such prevents to retry to open it again. */ #define FILE_IS_MISSING 0x800 #define FILE_IS_HARDLINK 0x1000 /**< If it's an hardlink. */ #define FILE_IS_SYMLINK 0x2000 /**< If it's a file symlink. */ #define FILE_IS_SYMDIR 0x4000 /**< If it's a dir symlink for Windows. Not yet supported. */ #define FILE_IS_JUNCTION 0x8000 /**< If it's a junction for Windows. Not yet supported. */ #define FILE_IS_LINK_MASK 0xF000 /**< Mask for link type. */ /** * File. */ struct snapraid_file { int64_t mtime_sec; /**< Modification time. */ uint64_t inode; /**< Inode. */ uint64_t physical; /**< Physical offset of the file. */ data_off_t size; /**< Size of the file. */ struct snapraid_block* blockvec; /**< All the blocks of the file. */ int mtime_nsec; /**< Modification time nanoseconds. In the range 0 <= x < 1,000,000,000, or STAT_NSEC_INVALID if not present. */ block_off_t blockmax; /**< Number of blocks. */ unsigned flag; /**< FILE_IS_* flags. */ char* sub; /**< Sub path of the file. Without the disk dir. The disk is implicit. */ /* nodes for data structures */ tommy_node nodelist; tommy_hashdyn_node nodeset; tommy_hashdyn_node pathset; tommy_hashdyn_node stampset; }; /** * Symbolic Link. */ struct snapraid_link { unsigned flag; /**< FILE_IS_* flags. */ char* sub; /**< Sub path of the file. Without the disk dir. The disk is implicit. */ char* linkto; /**< Link to. */ /* nodes for data structures */ tommy_node nodelist; tommy_hashdyn_node nodeset; }; /** * Dir. */ struct snapraid_dir { unsigned flag; /**< FILE_IS_* flags. */ char* sub; /**< Sub path of the file. Without the disk dir. The disk is implicit. */ /* nodes for data structures */ tommy_node nodelist; tommy_hashdyn_node nodeset; }; /** * Chunk. * * A extent represents a fragment of a file mapped into the parity. */ struct snapraid_extent { struct snapraid_file* file; /**< File containing this extent. */ block_off_t parity_pos; /**< Parity position. */ block_off_t file_pos; /**< Position in the file. */ block_off_t count; /**< Number of sequential blocks in the file and parity. */ tommy_tree_node parity_node; /**< Tree sorted by . */ tommy_tree_node file_node; /**< Tree sorter by . */ }; /** * Disk. */ struct snapraid_disk { char name[PATH_MAX]; /**< Name of the disk. */ char dir[PATH_MAX]; /**< Mount point of the disk. It always terminates with /. */ char smartctl[PATH_MAX]; /**< Custom command for smartctl. Empty means auto. */ char uuid[UUID_MAX]; /**< UUID of the disk. */ uint64_t device; /**< Device identifier. */ block_off_t total_blocks; /**< Number of total blocks. */ block_off_t free_blocks; /**< Number of free blocks at the last sync. */ uint64_t tick; /**< Usage time. */ uint64_t progress_tick[PROGRESS_MAX]; /**< Last ticks of progress. */ unsigned cached_blocks; /**< Number of IO blocks cached. */ struct snapraid_file* progress_file; /**< File in progress. */ /** * First free searching block. * Note that it doesn't necessarily point at the first free block, * but it just tell you that no free block is present before this position. */ block_off_t first_free_block; int has_volatile_inodes; /**< If the underline file-system has not persistent inodes. */ int has_volatile_hardlinks; /**< If the underline file-system has not synchronized metadata for hardlink (NTFS). */ int has_unreliable_physical; /**< If the physical offset of files has duplicates. */ int has_different_uuid; /**< If the disk has a different UUID, meaning that it is not the same file-system. */ int has_unsupported_uuid; /**< If the disk doesn't report UUID, meaning it's not supported. */ int had_empty_uuid; /**< If the disk had an empty UUID, meaning that it's a new disk. */ int mapping_idx; /**< Index in the mapping vector. Used only as buffer when writing the content file. */ int skip_access; /**< If the disk is inaccessible and it should be skipped. */ #if HAVE_THREAD /** * Mutex for protecting the filesystem structure. * * Specifically, this protects ::fs_parity, ::fs_file, and ::fs_last, * meaning that it protects only extents. * * Files, links and dirs are not protected as they are not expected to * change during multithread processing. */ thread_mutex_t fs_mutex; int fs_mutex_enabled; /*< If the lock has to be used. */ /** * Mutex for protecting the scan process. * * It's used during the scan process to protect the stampset to identity copy of files */ thread_mutex_t stamp_mutex; #endif /** * Mapping of extents in the parity. * Sorted by and by */ tommy_tree fs_parity; tommy_tree fs_file; /** * Last extent we accessed. * It's used to optimize access of sequential blocks. */ struct snapraid_extent* fs_last; /** * List of all the snapraid_file for the disk. */ tommy_list filelist; /** * List of all the deleted file for the disk. * * These files are kept allocated, because the blocks are still referenced in * the ::blockarr. */ tommy_list deletedlist; tommy_hashdyn inodeset; /**< Hashtable by inode of all the files. */ tommy_hashdyn pathset; /**< Hashtable by path of all the files. */ tommy_hashdyn stampset; /**< Hashtable by stamp (size and time) of all the files. */ tommy_list linklist; /**< List of all the links. */ tommy_hashdyn linkset; /**< Hashtable by name of all the links. */ tommy_list dirlist; /**< List of all the empty dirs. */ tommy_hashdyn dirset; /**< Hashtable by name of all the empty dirs. */ /* nodes for data structures */ tommy_node node; }; /** * Disk mapping. */ struct snapraid_map { char name[PATH_MAX]; /**< Name of the disk. */ char uuid[UUID_MAX]; /**< UUID of the disk. Empty if unknown. */ block_off_t total_blocks; /**< Number of total blocks. */ block_off_t free_blocks; /**< Number of free blocks at last 'sync'. */ unsigned position; /**< Position of the disk in the parity. */ /* nodes for data structures */ tommy_node node; }; /** * Max number of parity split. */ #define SPLIT_MAX 8 /** * Invalid parity size. * * This value is used to identify new parities, * like when you alter the configuration adding * a new parity level, creating it with 'fix'. * Given that 'fix' doesn't write the content file, * the new size will be written only at the next * 'sync'. */ #define PARITY_SIZE_INVALID -1LL /** * Parity split. */ struct snapraid_split { char path[PATH_MAX]; /**< Path of the parity file. */ char uuid[UUID_MAX]; /**< UUID of the disk. Empty if unknown. */ /** * Size of the parity split. * Only the latest not zero size is allowed to grow. * If the value is unset, it's PARITY_SIZE_INVALID. */ data_off_t size; uint64_t device; /**< Device identifier of the parity. */ }; /** * Parity. */ struct snapraid_parity { struct snapraid_split split_map[SPLIT_MAX]; /**< Parity splits. */ unsigned split_mac; /**< Number of parity splits. */ char smartctl[PATH_MAX]; /**< Custom command for smartctl. Empty means auto. */ block_off_t total_blocks; /**< Number of total blocks. */ block_off_t free_blocks; /**< Number of free blocks at the last sync. */ int is_excluded_by_filter; /**< If the parity is excluded by filters. */ int skip_access; /**< If at least one of the parity disk is inaccessible and it should be skipped. */ uint64_t tick; /**< Usage time. */ uint64_t progress_tick[PROGRESS_MAX]; /**< Last cpu ticks of progress. */ unsigned cached_blocks; /**< Number of IO blocks cached. */ }; /** * Info. */ typedef uint32_t snapraid_info; /** * Allocate a content. */ struct snapraid_content* content_alloc(const char* path, uint64_t dev); /** * Deallocate a content. */ void content_free(struct snapraid_content* content); /** * Allocate a filter pattern for files and directories. */ struct snapraid_filter* filter_alloc_file(int is_include, const char* pattern); /** * Allocate a filter pattern for disks. */ struct snapraid_filter* filter_alloc_disk(int is_include, const char* pattern); /** * Deallocate an exclusion. */ void filter_free(struct snapraid_filter* filter); /** * Filter type description. */ const char* filter_type(struct snapraid_filter* filter, char* out, size_t out_size); /** * Filter hidden files. * Return !=0 if it matches and it should be excluded. */ static inline int filter_hidden(int enable, struct dirent* dd) { if (enable && dirent_hidden(dd)) { return 1; /* filter out */ } return 0; } /** * Filter a path using a list of filters. * For each element of the path all the filters are applied, until the first one that matches. * Return !=0 if it should be excluded. */ int filter_path(tommy_list* filterlist, struct snapraid_filter** reason, const char* disk, const char* sub); /** * Filter a file/link/dir if missing. * This call imply a disk check for the file presence. * Return !=0 if the file is present and it should be excluded. */ int filter_existence(int filter_missing, const char* dir, const char* sub); /** * Filter a file if bad. * Return !=0 if the file is correct and it should be excluded. */ int filter_correctness(int filter_error, tommy_arrayblkof* infoarr, struct snapraid_disk* disk, struct snapraid_file* file); /** * Filter a dir using a list of filters. * For each element of the path all the filters are applied, until the first one that matches. * Thesesdir are always by included by default, to allow to apply rules at the contained files. * Return !=0 if should be excluded. */ int filter_subdir(tommy_list* filterlist, struct snapraid_filter** reason, const char* disk, const char* sub); /** * Filter a dir using a list of filters. * For each element of the path all the filters are applied, until the first one that matches. * Return !=0 if should be excluded. */ int filter_emptydir(tommy_list* filterlist, struct snapraid_filter** reason, const char* disk, const char* sub); /** * Filter a path if it's a content file. * Return !=0 if should be excluded. */ int filter_content(tommy_list* contentlist, const char* path); /** * Check if the specified hash is invalid. * * An invalid hash is represented with all bytes at 0x00. * * If working with reduced hash lengths, this function always return 0. */ static inline int hash_is_invalid(const unsigned char* hash) { int i; /* if the hash is reduced, we cannot grant that it's a specific kind of hash */ if (BLOCK_HASH_SIZE != HASH_MAX) return 0; for (i = 0; i < BLOCK_HASH_SIZE; ++i) if (hash[i] != 0x00) return 0; return 1; } /** * Check if the specified hash represent the zero block. * * A zero hash is represented with all bytes at 0xFF. * * If working with reduced hash lengths, this function always return 0. */ static inline int hash_is_zero(const unsigned char* hash) { int i; /* if the hash is reduced, we cannot grant that it's a specific kind of hash */ if (BLOCK_HASH_SIZE != HASH_MAX) return 0; for (i = 0; i < BLOCK_HASH_SIZE; ++i) if (hash[i] != 0xFF) return 0; return 1; } /** * Check if the specified hash is unequivocally representing the data. * * If working with reduced hash lengths, this function always return 0. */ static inline int hash_is_unique(const unsigned char* hash) { /* if the hash is reduced, we cannot grant that it's a specific kind of hash */ if (BLOCK_HASH_SIZE != HASH_MAX) return 0; return !hash_is_zero(hash) && !hash_is_invalid(hash); } /** * Set the hash to the special INVALID value. */ static inline void hash_invalid_set(unsigned char* hash) { memset(hash, 0x00, BLOCK_HASH_SIZE); } /** * Set the hash to the special ZERO value. */ static inline void hash_zero_set(unsigned char* hash) { memset(hash, 0xFF, BLOCK_HASH_SIZE); } /** * Allocated space for block. */ static inline size_t block_sizeof(void) { return 1 + BLOCK_HASH_SIZE; } /** * Get the state of the block. * * For this function, it's allowed to pass a NULL block * pointer than results in the BLOCK_STATE_EMPTY state. */ static inline unsigned block_state_get(const struct snapraid_block* block) { if (block == BLOCK_NULL) return BLOCK_STATE_EMPTY; return block->state; } /** * Set the state of the block. */ static inline void block_state_set(struct snapraid_block* block, unsigned state) { block->state = state; } /** * Check if the specified block has an updated hash. * * Note that EMPTY / CHG / DELETED return 0. */ static inline int block_has_updated_hash(const struct snapraid_block* block) { unsigned state = block_state_get(block); return state == BLOCK_STATE_BLK || state == BLOCK_STATE_REP; } /** * Check if the specified block has a past hash, * i.e. the hash of the data that it's now overwritten or lost. * * Note that EMPTY / BLK / REP return 0. */ static inline int block_has_past_hash(const struct snapraid_block* block) { unsigned state = block_state_get(block); return state == BLOCK_STATE_CHG || state == BLOCK_STATE_DELETED; } /** * Check if the specified block is part of a file. * * Note that EMPTY / DELETED return 0. */ static inline int block_has_file(const struct snapraid_block* block) { unsigned state = block_state_get(block); return state == BLOCK_STATE_BLK || state == BLOCK_STATE_CHG || state == BLOCK_STATE_REP; } /** * Check if the block has an invalid parity than needs to be updated. * * Note that EMPTY / BLK return 0. */ static inline int block_has_invalid_parity(const struct snapraid_block* block) { unsigned state = block_state_get(block); return state == BLOCK_STATE_DELETED || state == BLOCK_STATE_CHG || state == BLOCK_STATE_REP; } /** * Check if the block is part of a file with valid parity. * * Note that anything different than BLK return 0. */ static inline int block_has_file_and_valid_parity(const struct snapraid_block* block) { unsigned state = block_state_get(block); return state == BLOCK_STATE_BLK; } static inline int file_flag_has(const struct snapraid_file* file, unsigned mask) { return (file->flag & mask) == mask; } static inline void file_flag_set(struct snapraid_file* file, unsigned mask) { file->flag |= mask; } static inline void file_flag_clear(struct snapraid_file* file, unsigned mask) { file->flag &= ~mask; } /** * Allocate a file. */ struct snapraid_file* file_alloc(unsigned block_size, const char* sub, data_off_t size, uint64_t mtime_sec, int mtime_nsec, uint64_t inode, uint64_t physical); /** * Duplicate a file. */ struct snapraid_file* file_dup(struct snapraid_file* copy); /** * Deallocate a file. */ void file_free(struct snapraid_file* file); /** * Rename a file. */ void file_rename(struct snapraid_file* file, const char* sub); /** * Copy a file. */ void file_copy(struct snapraid_file* src_file, struct snapraid_file* dest_file); /** * Return the block at the specified position. * * Note that the block size if a runtime value. */ static inline struct snapraid_block* file_block(struct snapraid_file* file, size_t pos) { unsigned char* ptr = (unsigned char*)file->blockvec; return (struct snapraid_block*)(ptr + pos * block_sizeof()); } /** * Return the name of the file, without the dir. */ const char* file_name(const struct snapraid_file* file); /** * Check if the block is the last in the file. */ int file_block_is_last(struct snapraid_file* file, block_off_t file_pos); /** * Get the size in bytes of the block. * If it's the last block of a file it could be less than block_size. */ unsigned file_block_size(struct snapraid_file* file, block_off_t file_pos, unsigned block_size); /** * Compare a file with an inode. */ int file_inode_compare_to_arg(const void* void_arg, const void* void_data); /** * Compare files by inode. */ int file_inode_compare(const void* void_a, const void* void_b); /** * Compare files by path. */ int file_path_compare(const void* void_a, const void* void_b); /** * Compare files by physical address. */ int file_physical_compare(const void* void_a, const void* void_b); /** * Compute the hash of a file inode. */ static inline tommy_uint32_t file_inode_hash(uint64_t inode) { return (tommy_uint32_t)tommy_inthash_u64(inode); } /** * Compare a file with a path. */ int file_path_compare_to_arg(const void* void_arg, const void* void_data); /** * Compare a file with another file for name, stamp, and both. */ int file_name_compare(const void* void_a, const void* void_b); int file_stamp_compare(const void* void_a, const void* void_b); int file_namestamp_compare(const void* void_a, const void* void_b); int file_pathstamp_compare(const void* void_a, const void* void_b); /** * Compute the hash of a file path. */ static inline tommy_uint32_t file_path_hash(const char* sub) { return tommy_hash_u32(0, sub, strlen(sub)); } /** * Compute the hash of a file stamp. */ static inline tommy_uint32_t file_stamp_hash(data_off_t size, int64_t mtime_sec, int mtime_nsec) { return tommy_inthash_u32((tommy_uint32_t)size ^ tommy_inthash_u32(mtime_sec ^ tommy_inthash_u32(mtime_nsec))); } /** * Allocate a extent. */ struct snapraid_extent* extent_alloc(block_off_t parity_pos, struct snapraid_file* file, block_off_t file_pos, block_off_t count); /** * Deallocate a extent. */ void extent_free(struct snapraid_extent* extent); /** * Compare extent by parity position. */ int extent_parity_compare(const void* void_a, const void* void_b); /** * Compare extent by file and file position. */ int extent_file_compare(const void* void_a, const void* void_b); static inline int link_flag_has(const struct snapraid_link* slink, unsigned mask) { return (slink->flag & mask) == mask; } static inline void link_flag_set(struct snapraid_link* slink, unsigned mask) { slink->flag |= mask; } static inline void link_flag_clear(struct snapraid_link* slink, unsigned mask) { slink->flag &= ~mask; } static inline void link_flag_let(struct snapraid_link* slink, unsigned flag, unsigned mask) { slink->flag &= ~mask; slink->flag |= flag & mask; } static inline unsigned link_flag_get(struct snapraid_link* slink, unsigned mask) { return slink->flag & mask; } /** * Allocate a link. */ struct snapraid_link* link_alloc(const char* name, const char* slink, unsigned link_flag); /** * Deallocate a link. */ void link_free(struct snapraid_link* slink); /** * Compare a link with a name. */ int link_name_compare_to_arg(const void* void_arg, const void* void_data); /** * Compare links by path. */ int link_alpha_compare(const void* void_a, const void* void_b); /** * Compute the hash of a link name. */ static inline tommy_uint32_t link_name_hash(const char* name) { return tommy_hash_u32(0, name, strlen(name)); } static inline int dir_flag_has(const struct snapraid_dir* dir, unsigned mask) { return (dir->flag & mask) == mask; } static inline void dir_flag_set(struct snapraid_dir* dir, unsigned mask) { dir->flag |= mask; } static inline void dir_flag_clear(struct snapraid_dir* dir, unsigned mask) { dir->flag &= ~mask; } /** * Allocate a dir. */ struct snapraid_dir* dir_alloc(const char* name); /** * Deallocate a dir. */ void dir_free(struct snapraid_dir* dir); /** * Compare a dir with a name. */ int dir_name_compare(const void* void_arg, const void* void_data); /** * Compute the hash of a dir name. */ static inline tommy_uint32_t dir_name_hash(const char* name) { return tommy_hash_u32(0, name, strlen(name)); } /** * Allocate a disk. */ struct snapraid_disk* disk_alloc(const char* name, const char* dir, uint64_t dev, const char* uuid, int skip_access); /** * Deallocate a disk. */ void disk_free(struct snapraid_disk* disk); /** * Enable multithread support for the disk. */ void disk_start_thread(struct snapraid_disk* disk); /** * Get the size of the disk in blocks. */ block_off_t fs_size(struct snapraid_disk* disk); /** * Check if a disk is totally empty and can be discarded from the content file. * A disk is empty if it doesn't contain any file, symlink, hardlink or dir * and without any DELETED block. * The blockmax is used to limit the search of DELETED block up to blockmax. */ int fs_is_empty(struct snapraid_disk* disk, block_off_t blockmax); /** * Check the file-system for errors. * Return 0 if it's OK. */ int fs_check(struct snapraid_disk* disk); /** * Allocate a parity position for the specified file position. * * After this call you can use the par2file/par2block operations * to query the relation. * * \note This function is NOT thread-safe as it uses the disk cache. + */ void fs_allocate(struct snapraid_disk* disk, block_off_t parity_pos, struct snapraid_file* file, block_off_t file_pos); /** * Deallocate the parity position. * * After this call the par2file/par2block operations * won't find anymore the parity association. * * \note This function is NOT thread-safe as it uses the disk cache. */ void fs_deallocate(struct snapraid_disk* disk, block_off_t pos); /** * Get the block from the file position. */ struct snapraid_block* fs_file2block_get(struct snapraid_file* file, block_off_t file_pos); /** * Get the file position from the parity position. * Return 0 if no file is using it. */ struct snapraid_file* fs_par2file_find(struct snapraid_disk* disk, block_off_t parity_pos, block_off_t* file_pos); /** * Get the file position from the parity position. */ static inline struct snapraid_file* fs_par2file_get(struct snapraid_disk* disk, block_off_t parity_pos, block_off_t* file_pos) { struct snapraid_file* ret; ret = fs_par2file_find(disk, parity_pos, file_pos); if (ret == 0) { /* LCOV_EXCL_START */ log_fatal("Internal inconsistency when deresolving parity to file at position '%u' in disk '%s'\n", parity_pos, disk->name); os_abort(); /* LCOV_EXCL_STOP */ } return ret; } /** * Get the parity position from the file position. * Return POS_NULL if no parity is allocated. */ block_off_t fs_file2par_find(struct snapraid_disk* disk, struct snapraid_file* file, block_off_t file_pos); /** * Get the parity position from the file position. */ static inline block_off_t fs_file2par_get(struct snapraid_disk* disk, struct snapraid_file* file, block_off_t file_pos) { block_off_t ret; ret = fs_file2par_find(disk, file, file_pos); if (ret == POS_NULL) { /* LCOV_EXCL_START */ log_fatal("Internal inconsistency when resolving file '%s' at position '%u/%u' in disk '%s'\n", file->sub, file_pos, file->blockmax, disk->name); os_abort(); /* LCOV_EXCL_STOP */ } return ret; } /** * Get the block from the parity position. * Return BLOCK_NULL==0 if the block is over the end of the disk or not used. */ struct snapraid_block* fs_par2block_find(struct snapraid_disk* disk, block_off_t parity_pos); /** * Get the block from the parity position. */ static inline struct snapraid_block* fs_par2block_get(struct snapraid_disk* disk, block_off_t parity_pos) { struct snapraid_block* ret; ret = fs_par2block_find(disk, parity_pos); if (ret == BLOCK_NULL) { /* LCOV_EXCL_START */ log_fatal("Internal inconsistency when deresolving parity to block at position '%u' in disk '%s'\n", parity_pos, disk->name); os_abort(); /* LCOV_EXCL_STOP */ } return ret; } /** * Allocate a disk mapping. * Uses uuid="" if not available. */ struct snapraid_map* map_alloc(const char* name, unsigned position, block_off_t total_blocks, block_off_t free_blocks, const char* uuid); /** * Deallocate a disk mapping. */ void map_free(struct snapraid_map* map); /** * Mask used to store additional information in the info bits. * * These bits reduce the granularity of the time in the memory representation. */ #define INFO_MASK 0x7 /** * Make an info. */ static inline snapraid_info info_make(time_t last_access, int error, int rehash, int justsynced) { /* clear the lowest bits as reserved for other information */ snapraid_info info = last_access & ~INFO_MASK; if (error != 0) info |= 0x1; if (rehash != 0) info |= 0x2; if (justsynced != 0) info |= 0x4; return info; } /** * Extract the time information. * This is the last time when the block was know to be correct. * The "scrubbed" info tells if the time is referring at the latest sync or scrub. */ static inline time_t info_get_time(snapraid_info info) { return info & ~INFO_MASK; } /** * Extract the error information. * Report if the block address had some problem. */ static inline int info_get_bad(snapraid_info info) { return (info & 0x1) != 0; } /** * Extract the rehash information. * Report if the block address is using the old hash and needs to be rehashed. */ static inline int info_get_rehash(snapraid_info info) { return (info & 0x2) != 0; } /** * Extract the scrubbed information. * Report if the block address was never scrubbed. */ static inline int info_get_justsynced(snapraid_info info) { return (info & 0x4) != 0; } /** * Mark the block address as with error. */ static inline snapraid_info info_set_bad(snapraid_info info) { return info | 0x1; } /** * Mark the block address as with rehash. */ static inline snapraid_info info_set_rehash(snapraid_info info) { return info | 0x2; } /** * Set the info at the specified position. * The position is allocated if not yet done. */ static inline void info_set(tommy_arrayblkof* array, block_off_t pos, snapraid_info info) { tommy_arrayblkof_grow(array, pos + 1); memcpy(tommy_arrayblkof_ref(array, pos), &info, sizeof(snapraid_info)); } /** * Get the info at the specified position. * For not allocated position, 0 is returned. */ static inline snapraid_info info_get(tommy_arrayblkof* array, block_off_t pos) { snapraid_info info; if (pos >= tommy_arrayblkof_size(array)) return 0; memcpy(&info, tommy_arrayblkof_ref(array, pos), sizeof(snapraid_info)); return info; } /** * Compare times */ int time_compare(const void* void_a, const void* void_b); /****************************************************************************/ /* format */ #define FMT_FILE 0 /**< Print only the file. */ #define FMT_DISK 1 /**< Print the disk name and the file. */ #define FMT_PATH 2 /**< Print the full path. */ extern int FMT_MODE; /** * Format a file path for poll reference */ const char* fmt_poll(const struct snapraid_disk* disk, const char* str, char* buffer); /** * Format a path name for terminal reference */ const char* fmt_term(const struct snapraid_disk* disk, const char* str, char* buffer); #endif