blob: c6eb9004380a165320750a0b5ff4ef2ee3dc2c3f [file] [log] [blame]
/*
* Copyright 2000-2004 by Hans Reiser, licensing governed by
* reiserfsprogs/README
*/
#include "includes.h"
int leaf_count_ih(const char *buf, int blocksize)
{
const struct item_head *ih;
int prev_location;
int nr;
/* look at the table of item head */
prev_location = blocksize;
ih = (struct item_head *)(buf + BLKH_SIZE);
nr = 0;
while (1) {
if (get_ih_location(ih) + get_ih_item_len(ih) != prev_location)
break;
if (get_ih_location(ih) < IH_SIZE * (nr + 1) + BLKH_SIZE)
break;
if (get_ih_item_len(ih) > MAX_ITEM_LEN(blocksize))
break;
prev_location = get_ih_location(ih);
ih++;
nr++;
}
return nr;
}
int leaf_free_space_estimate(const char *buf, int blocksize)
{
const struct block_head *blkh;
const struct item_head *ih;
int nr;
blkh = (struct block_head *)buf;
nr = get_blkh_nr_items(blkh);
ih = (struct item_head *)(buf + BLKH_SIZE) + nr - 1;
return (nr ? get_ih_location(ih) : blocksize) - BLKH_SIZE -
IH_SIZE * nr;
}
static int leaf_blkh_correct(const char *buf, int blocksize)
{
struct block_head *blkh;
unsigned int nr;
blkh = (struct block_head *)buf;
if (!is_leaf_block_head(buf))
return 0;
nr = get_blkh_nr_items(blkh);
if (nr > ((blocksize - BLKH_SIZE) / (IH_SIZE + MIN_ITEM_LEN)))
/* item number is too big or too small */
return 0;
return leaf_free_space_estimate(buf,
blocksize) == get_blkh_free_space(blkh);
}
int is_a_leaf(const char *buf, int blocksize)
{
const struct block_head *blkh;
int counted;
blkh = (struct block_head *)buf;
if (!is_leaf_block_head(buf))
return 0;
counted = leaf_count_ih(buf, blocksize);
/* if leaf block header is ok, check item count also. */
if (leaf_blkh_correct(buf, blocksize))
return counted >=
get_blkh_nr_items(blkh) ? THE_LEAF : HAS_IH_ARRAY;
/* leaf block header is corrupted, it is ih_array if some items were detected. */
return counted ? HAS_IH_ARRAY : 0;
}
int leaf_item_number_estimate(const struct buffer_head *bh)
{
const struct block_head *blkh;
int nr;
nr = leaf_count_ih(bh->b_data, bh->b_size);
blkh = (const struct block_head *)bh->b_data;
return nr >= get_blkh_nr_items(blkh) ? get_blkh_nr_items(blkh) : nr;
}
#if 0
/* this only checks block header and item head array (ih_location-s
and ih_item_len-s). Item internals are not checked */
int does_node_look_like_a_leaf(char *buf, int blocksize)
{
struct block_head *blkh;
struct item_head *ih;
int used_space;
int prev_location;
int i;
int nr;
blkh = (struct block_head *)buf;
if (!is_leaf_block_head(buf))
return 0;
nr = get_blkh_nr_items(blkh);
if (nr < 0 || nr > ((blocksize - BLKH_SIZE) / (IH_SIZE + MIN_ITEM_LEN)))
/* item number is too big or too small */
return 0;
ih = (struct item_head *)(buf + BLKH_SIZE) + nr - 1;
used_space =
BLKH_SIZE + IH_SIZE * nr + (blocksize -
(nr ? get_ih_location(ih) : blocksize));
if (used_space != blocksize - get_blkh_free_space(blkh))
/* free space does not match to calculated amount of use space */
return 0;
// FIXME: it is_leaf will hit performance too much - we may have
// return 1 here
/* check tables of item heads */
ih = (struct item_head *)(buf + BLKH_SIZE);
prev_location = blocksize;
for (i = 0; i < nr; i++, ih++) {
/* items of zero length are allowed - they may exist for short time
during balancing */
if (get_ih_location(ih) > blocksize
|| get_ih_location(ih) < IH_SIZE * nr)
return 0;
if ( /*ih_item_len (ih) < 1 || */ get_ih_item_len(ih) >
MAX_ITEM_LEN(blocksize))
return 0;
if (prev_location - get_ih_location(ih) != get_ih_item_len(ih))
return 0;
prev_location = get_ih_location(ih);
}
// one may imagine much more checks
return 1;
}
/* check ih_item_len and ih_location. Should be useful when block head is
corrupted */
static int does_node_have_ih_array(char *buf, int blocksize)
{
struct item_head *ih;
int prev_location;
int nr;
/* look at the table of item head */
prev_location = blocksize;
ih = (struct item_head *)(buf + BLKH_SIZE);
nr = 0;
while (1) {
if (get_ih_location(ih) + get_ih_item_len(ih) != prev_location)
break;
prev_location = get_ih_location(ih);
ih++;
nr++;
}
if (nr < 2)
return 0;
return nr;
}
#endif
/* returns 1 if buf looks like an internal node, 0 otherwise */
static int is_correct_internal(const char *buf, int blocksize)
{
const struct block_head *blkh;
unsigned int nr;
int used_space;
blkh = (struct block_head *)buf;
if (!is_internal_block_head(buf))
return 0;
nr = get_blkh_nr_items(blkh);
if (nr > (blocksize - BLKH_SIZE - DC_SIZE) / (KEY_SIZE + DC_SIZE))
/* for internal which is not root we might check min number of keys */
return 0;
used_space = BLKH_SIZE + KEY_SIZE * nr + DC_SIZE * (nr + 1);
if (used_space != blocksize - get_blkh_free_space(blkh))
return 0;
// one may imagine much more checks
return 1;
}
// make sure that bh contains formatted node of reiserfs tree of
// 'level'-th level
int is_tree_node(struct buffer_head *bh, int level)
{
if (B_LEVEL(bh) != level)
return 0;
if (is_leaf_node(bh))
return is_a_leaf(bh->b_data, bh->b_size);
return is_correct_internal(bh->b_data, bh->b_size);
}
static int is_desc_block(const char *buf, unsigned long buf_size)
{
const struct reiserfs_journal_desc *desc =
(struct reiserfs_journal_desc *)buf;
if (!memcmp(buf + buf_size - 12, JOURNAL_DESC_MAGIC, 8)
&& le32_to_cpu(desc->j2_len) > 0)
return 1;
return 0;
}
int is_reiserfs_3_5_magic_string(struct reiserfs_super_block *rs)
{
return (!strncmp(rs->s_v1.s_magic, REISERFS_3_5_SUPER_MAGIC_STRING,
strlen(REISERFS_3_5_SUPER_MAGIC_STRING)));
}
int is_reiserfs_3_6_magic_string(struct reiserfs_super_block *rs)
{
return (!strncmp(rs->s_v1.s_magic, REISERFS_3_6_SUPER_MAGIC_STRING,
strlen(REISERFS_3_6_SUPER_MAGIC_STRING)));
}
int is_reiserfs_jr_magic_string(struct reiserfs_super_block *rs)
{
return (!strncmp(rs->s_v1.s_magic, REISERFS_JR_SUPER_MAGIC_STRING,
strlen(REISERFS_JR_SUPER_MAGIC_STRING)));
}
int is_any_reiserfs_magic_string(struct reiserfs_super_block *rs)
{
if (is_reiserfs_3_5_magic_string(rs) ||
is_reiserfs_3_6_magic_string(rs) || is_reiserfs_jr_magic_string(rs))
return 1;
return 0;
}
int get_reiserfs_format(struct reiserfs_super_block *sb)
{
/* after conversion to 3.6 format we change magic correctly,
but do not change sb_format. When we create non-standard journal
field format in sb get adjusted correctly. Thereby, for standard
journal we should rely on magic and for non-standard - on format */
if (is_reiserfs_3_5_magic_string(sb) ||
(is_reiserfs_jr_magic_string(sb) &&
get_sb_version(sb) == REISERFS_FORMAT_3_5))
return REISERFS_FORMAT_3_5;
if (is_reiserfs_3_6_magic_string(sb) ||
(is_reiserfs_jr_magic_string(sb) &&
get_sb_version(sb) == REISERFS_FORMAT_3_6))
return REISERFS_FORMAT_3_6;
return REISERFS_FORMAT_UNKNOWN;
}
int reiserfs_super_block_size(struct reiserfs_super_block *sb)
{
switch (get_reiserfs_format(sb)) {
case REISERFS_FORMAT_3_5:
return SB_SIZE_V1;
case REISERFS_FORMAT_3_6:
return SB_SIZE;
}
reiserfs_panic("Unknown format found");
return 0;
}
/* this one had signature in different place of the super_block
structure */
int is_prejournaled_reiserfs(struct reiserfs_super_block *rs)
{
return (!strncmp((char *)rs + REISERFS_SUPER_MAGIC_STRING_OFFSET_NJ,
REISERFS_3_5_SUPER_MAGIC_STRING,
strlen(REISERFS_3_5_SUPER_MAGIC_STRING)));
}
int does_look_like_super_block(struct reiserfs_super_block *sb)
{
if (!is_any_reiserfs_magic_string(sb))
return 0;
if (!is_blocksize_correct(get_sb_block_size(sb)))
return 0;
return 1;
}
/* returns code of reiserfs metadata block (leaf, internal, super
block, journal descriptor), unformatted */
int who_is_this(const char *buf, int blocksize)
{
int res;
/* super block? */
if (does_look_like_super_block((void *)buf))
return THE_SUPER;
if ((res = is_a_leaf(buf, blocksize)))
/* if block head and item head array seem matching (node level, free
space, item number, item locations and length), then it is THE_LEAF,
otherwise, it is HAS_IH_ARRAY */
return res;
if (is_correct_internal(buf, blocksize))
return THE_INTERNAL;
/* journal descriptor block? */
if (is_desc_block(buf, blocksize))
return THE_JDESC;
/* contents of buf does not look like reiserfs metadata. Bitmaps
are possible here */
return THE_UNKNOWN;
}
char *which_block(int code)
{
static char *leaf = "leaf";
static char *broken_leaf = "broken leaf";
static char *internal = "internal";
static char *other = "unknown";
switch (code) {
case THE_LEAF:
return leaf;
case HAS_IH_ARRAY:
return broken_leaf;
case THE_INTERNAL:
return internal;
}
return other;
}
/** */
int block_of_journal(reiserfs_filsys_t fs, unsigned long block)
{
if (!is_reiserfs_jr_magic_string(fs->fs_ondisk_sb)) {
/* standard journal */
if (block >= get_journal_start_must(fs) &&
block <=
get_journal_start_must(fs) +
get_jp_journal_size(sb_jp(fs->fs_ondisk_sb)))
return 1;
return 0;
}
if (get_sb_reserved_for_journal(fs->fs_ondisk_sb))
/* there is space reserved for the journal on the host device */
if (block >= get_journal_start_must(fs) &&
block <
get_journal_start_must(fs) +
get_sb_reserved_for_journal(fs->fs_ondisk_sb))
return 1;
return 0;
}
int block_of_bitmap(reiserfs_filsys_t fs, unsigned long block)
{
if (spread_bitmaps(fs)) {
if (!(block % (fs->fs_blocksize * 8)))
/* bitmap block */
return 1;
return (block ==
(REISERFS_DISK_OFFSET_IN_BYTES / fs->fs_blocksize + 1));
} else {
/* bitmap in */
return (block > 2ul
&& block < 3ul + reiserfs_fs_bmap_nr(fs)) ? 1 : 0;
}
return 0;
}
/* check whether 'block' can be pointed to by an indirect item */
int not_data_block(reiserfs_filsys_t fs, unsigned long block)
{
if (block_of_bitmap(fs, block))
/* it is one of bitmap blocks */
return 1;
if (block_of_journal(fs, block))
/* block of journal area */
return 1;
if (block <= fs->fs_super_bh->b_blocknr)
/* either super block or a block from skipped area at the
beginning of filesystem */
return 1;
return 0;
}
/* check whether 'block' can be logged */
int not_journalable(reiserfs_filsys_t fs, unsigned long block)
{
/* we should not update SB with journal copy during fsck */
if (block < fs->fs_super_bh->b_blocknr)
return 1;
if (block_of_journal(fs, block))
return 1;
if (block >= get_sb_block_count(fs->fs_ondisk_sb))
return 1;
return 0;
}
// in reiserfs version 0 (undistributed bitmap)
// FIXME: what if number of bitmaps is 15?
unsigned int get_journal_old_start_must(reiserfs_filsys_t fs)
{
return (REISERFS_OLD_DISK_OFFSET_IN_BYTES / fs->fs_blocksize) + 1 +
get_sb_bmap_nr(fs->fs_ondisk_sb);
}
unsigned int get_journal_new_start_must(reiserfs_filsys_t fs)
{
return (REISERFS_DISK_OFFSET_IN_BYTES / fs->fs_blocksize) + 2;
}
unsigned int get_journal_start_must(reiserfs_filsys_t fs)
{
if (is_old_sb_location(fs->fs_super_bh->b_blocknr, fs->fs_blocksize))
return get_journal_old_start_must(fs);
return get_journal_new_start_must(fs);
}
__u32 get_bytes_number(struct item_head * ih, int blocksize)
{
switch (get_type(&ih->ih_key)) {
case TYPE_DIRECT:
return get_ih_item_len(ih);
case TYPE_INDIRECT:
return I_UNFM_NUM(ih) * blocksize; // - get_ih_free_space (ih);
case TYPE_STAT_DATA:
case TYPE_DIRENTRY:
return 0;
}
reiserfs_warning(stderr,
"get_bytes_number: called for wrong type of item %h",
ih);
return 0;
}
int check_item_f(reiserfs_filsys_t fs, struct item_head *ih, char *item);
/* ih_key, ih_location and ih_item_len seem correct, check other fields */
static int does_ih_look_correct(struct item_head *ih)
{
int ih_key_format;
int key_key_format;
/* key format from item_head */
ih_key_format = get_ih_key_format(ih);
if (ih_key_format != KEY_FORMAT_1 && ih_key_format != KEY_FORMAT_2)
return 0;
/* key format calculated on key */
key_key_format = key_format(&ih->ih_key);
if (is_stat_data_ih(ih)) {
/* for stat data we can not find key format from a key itself, so look at
the item length */
if (get_ih_item_len(ih) == SD_SIZE)
key_key_format = KEY_FORMAT_2;
else if (get_ih_item_len(ih) == SD_V1_SIZE)
key_key_format = KEY_FORMAT_1;
else
return 0;
}
if (key_key_format != ih_key_format)
return 0;
/* we do not check ih_format.fsck_need as fsck might change it. So,
debugreiserfs -p will have to dump it */
return 1;
}
/* check item length, ih_free_space for pure 3.5 format, unformatted node
pointers */
static int is_bad_indirect(reiserfs_filsys_t fs, struct item_head *ih,
const char *item, check_unfm_func_t check_unfm_func)
{
unsigned int i;
__le32 *ind = (__le32 *) item;
if (get_ih_item_len(ih) % UNFM_P_SIZE)
return 1;
for (i = 0; i < I_UNFM_NUM(ih); i++) {
if (!ind[i])
continue;
if (check_unfm_func && check_unfm_func(fs, d32_get(ind, i)))
return 1;
}
if (fs->fs_format == REISERFS_FORMAT_3_5) {
/* check ih_free_space for 3.5 format only */
if (get_ih_free_space(ih) > fs->fs_blocksize - 1)
return 1;
}
return 0;
}
static const struct {
hashf_t func;
char *name;
} hashes[] = { {
NULL, "not set"}, {
keyed_hash, "\"tea\""}, {
yura_hash, "\"rupasov\""}, {
r5_hash, "\"r5\""}};
#define HASH_AMOUNT (sizeof (hashes) / sizeof (hashes [0]))
int known_hashes(void)
{
return HASH_AMOUNT;
}
#define good_name(hashfn,name,namelen,deh_offset) \
(hash_value (hashfn, name, namelen) == GET_HASH_VALUE (deh_offset))
/* this also sets hash function */
int is_properly_hashed(reiserfs_filsys_t fs,
const char *name, int namelen, __u32 offset)
{
unsigned int i;
if (namelen == 1 && name[0] == '.') {
if (offset == DOT_OFFSET)
return 1;
return 0;
}
if (namelen == 2 && name[0] == '.' && name[1] == '.') {
if (offset == DOT_DOT_OFFSET)
return 1;
return 0;
}
if (hash_func_is_unknown(fs)) {
/* try to find what hash function the name is sorted with */
for (i = 1; i < HASH_AMOUNT; i++) {
if (good_name(hashes[i].func, name, namelen, offset)) {
if (!hash_func_is_unknown(fs)) {
/* two or more hash functions give the same value for this
name */
fprintf(stderr,
"Detecting hash code: could not detect hash with name \"%.*s\"\n",
namelen, name);
reiserfs_hash(fs) = NULL;
return 1;
}
/* set hash function */
reiserfs_hash(fs) = hashes[i].func;
}
}
if (hash_func_is_unknown(fs)) {
return 0;
}
}
if (good_name(reiserfs_hash(fs), name, namelen, offset))
return 1;
return 0;
}
int find_hash_in_use(const char *name, int namelen, __u32 offset,
unsigned int code_to_try_first)
{
unsigned int i;
if (!namelen || !name[0])
return UNSET_HASH;
if (code_to_try_first) {
if (good_name
(hashes[code_to_try_first].func, name, namelen, offset))
return code_to_try_first;
}
for (i = 1; i < HASH_AMOUNT; i++) {
if (i == code_to_try_first)
continue;
if (good_name(hashes[i].func, name, namelen, offset))
return i;
}
/* not matching hash found */
return UNSET_HASH;
}
char *code2name(unsigned int code)
{
if (code >= HASH_AMOUNT || code < 0)
return NULL;
return hashes[code].name;
}
int func2code(hashf_t func)
{
unsigned int i;
for (i = 0; i < HASH_AMOUNT; i++)
if (func == hashes[i].func)
return i;
reiserfs_panic("func2code: no hashes matches this function\n");
return 0;
}
hashf_t code2func(unsigned int code)
{
if (code >= HASH_AMOUNT) {
reiserfs_warning(stderr, "code2func: wrong hash code %d.\n"
"Using default %s hash function\n", code,
code2name(DEFAULT_HASH));
code = DEFAULT_HASH;
}
return hashes[code].func;
}
hashf_t name2func(const char *hash)
{
unsigned int i;
for (i = 0; i < HASH_AMOUNT; i++)
if (!strcmp(hash, hashes[i].name))
return hashes[i].func;
return NULL;
}
int dir_entry_bad_location(struct reiserfs_de_head *deh, struct item_head *ih,
int first)
{
if (get_deh_location(deh) < DEH_SIZE * get_ih_entry_count(ih))
return 1;
if (get_deh_location(deh) >= get_ih_item_len(ih))
return 1;
if (!first && get_deh_location(deh) >= get_deh_location(deh - 1))
return 1;
return 0;
}
/* the only corruption which is not considered fatal - is hash mismatching. If
bad_dir is set - directory item having such names is considered bad */
static int is_bad_directory(reiserfs_filsys_t fs, struct item_head *ih,
const char *item, int bad_dir)
{
int i;
int namelen;
struct reiserfs_de_head *deh = (struct reiserfs_de_head *)item;
__u32 prev_offset = 0;
__u16 prev_location = get_ih_item_len(ih);
for (i = 0; i < get_ih_entry_count(ih); i++, deh++) {
if (get_deh_location(deh) >= prev_location)
return 1;
prev_location = get_deh_location(deh);
namelen = name_in_entry_length(ih, deh, i);
if (namelen > (int)REISERFS_MAX_NAME_LEN(fs->fs_blocksize))
return 1;
if (get_deh_offset(deh) <= prev_offset)
return 1;
prev_offset = get_deh_offset(deh);
/* check hash value */
if (!is_properly_hashed
(fs, item + prev_location, namelen, prev_offset)) {
if (bad_dir)
/* make is_bad_leaf to not insert whole leaf. Node will be
marked not-insertable and put into tree item by item in
pass 2 */
return 1;
}
}
return 0;
}
/* used by debugreisrefs -p only yet */
int is_it_bad_item(reiserfs_filsys_t fs, struct item_head *ih, const char *item,
check_unfm_func_t check_unfm, int bad_dir)
{
int retval;
/*
if (!does_key_look_correct (fs, &ih->ih_key))
return 1;
if (!does_ih_look_correct (ih))
return 1;
*/
if (!does_ih_look_correct(ih))
return 1;
if (is_stat_data_ih(ih) || is_direct_ih(ih))
return 0;
if (is_direntry_ih(ih)) {
retval = is_bad_directory(fs, ih, item, bad_dir);
/*
if (retval)
reiserfs_warning (stderr, "is_bad_directory %H\n", ih); */
return retval;
}
if (is_indirect_ih(ih)) {
retval = is_bad_indirect(fs, ih, item, check_unfm);
/*
if (retval)
reiserfs_warning (stderr, "is_bad_indirect %H\n", ih); */
return retval;
}
return 1;
}
/* prepare new or old stat data for the new directory */
void make_dir_stat_data(int blocksize, int key_format,
__u32 dirid, __u32 objectid,
struct item_head *ih, void *sd)
{
memset(ih, 0, IH_SIZE);
set_key_dirid(&ih->ih_key, dirid);
set_key_objectid(&ih->ih_key, objectid);
set_key_offset_v1(&ih->ih_key, SD_OFFSET);
set_key_uniqueness(&ih->ih_key, 0);
set_ih_key_format(ih, key_format);
set_ih_free_space(ih, MAX_US_INT);
if (key_format == KEY_FORMAT_2) {
struct stat_data *sd_v2 = (struct stat_data *)sd;
set_ih_item_len(ih, SD_SIZE);
set_sd_v2_mode(sd_v2, S_IFDIR + 0755);
set_sd_v2_nlink(sd_v2, 2);
set_sd_v2_uid(sd_v2, 0);
set_sd_v2_gid(sd_v2, 0);
set_sd_v2_size(sd_v2, EMPTY_DIR_SIZE);
set_sd_v2_atime(sd_v2, time(NULL));
sd_v2->sd_ctime = sd_v2->sd_mtime = sd_v2->sd_atime; /* all le */
set_sd_v2_rdev(sd_v2, 0);
set_sd_v2_blocks(sd_v2, dir_size2st_blocks(EMPTY_DIR_SIZE));
} else {
struct stat_data_v1 *sd_v1 = (struct stat_data_v1 *)sd;
set_ih_item_len(ih, SD_V1_SIZE);
set_sd_v1_mode(sd_v1, S_IFDIR + 0755);
set_sd_v1_nlink(sd_v1, 2);
set_sd_v1_uid(sd_v1, 0);
set_sd_v1_gid(sd_v1, 0);
set_sd_v1_size(sd_v1, EMPTY_DIR_SIZE_V1);
set_sd_v1_atime(sd_v1, time(NULL));
sd_v1->sd_ctime = sd_v1->sd_mtime = sd_v1->sd_atime; /* all le */
set_sd_v1_blocks(sd_v1, dir_size2st_blocks(EMPTY_DIR_SIZE_V1));
set_sd_v1_first_direct_byte(sd_v1, NO_BYTES_IN_DIRECT_ITEM);
}
}
static void _empty_dir_item(int format, char *body, __u32 dirid, __u32 objid,
__u32 par_dirid, __u32 par_objid)
{
struct reiserfs_de_head *deh;
__u16 state;
memset(body, 0,
(format == KEY_FORMAT_2 ? EMPTY_DIR_SIZE : EMPTY_DIR_SIZE_V1));
deh = (struct reiserfs_de_head *)body;
/* direntry header of "." */
set_deh_offset(deh, DOT_OFFSET);
set_deh_dirid(deh, dirid);
set_deh_objectid(deh, objid);
state = (1 << DEH_Visible2);
set_deh_state(deh, state);
/* direntry header of ".." */
set_deh_offset(deh + 1, DOT_DOT_OFFSET);
/* key of ".." for the root directory */
set_deh_dirid(deh + 1, par_dirid);
set_deh_objectid(deh + 1, par_objid);
set_deh_state(deh + 1, state);
if (format == KEY_FORMAT_2) {
set_deh_location(deh, EMPTY_DIR_SIZE - ROUND_UP(strlen(".")));
set_deh_location(deh + 1,
get_deh_location(deh) -
ROUND_UP(strlen("..")));
} else {
set_deh_location(deh, EMPTY_DIR_SIZE_V1 - strlen("."));
set_deh_location(deh + 1, get_deh_location(deh) - strlen(".."));
}
/* copy ".." and "." */
memcpy(body + get_deh_location(deh), ".", 1);
memcpy(body + get_deh_location(deh + 1), "..", 2);
}
void make_empty_dir_item_v1(char *body, __u32 dirid, __u32 objid,
__u32 par_dirid, __u32 par_objid)
{
_empty_dir_item(KEY_FORMAT_1, body, dirid, objid, par_dirid, par_objid);
}
void make_empty_dir_item(char *body, __u32 dirid, __u32 objid,
__u32 par_dirid, __u32 par_objid)
{
_empty_dir_item(KEY_FORMAT_2, body, dirid, objid, par_dirid, par_objid);
}
/* for every item call common action and an action corresponding to
item type */
void for_every_item(struct buffer_head *bh, item_head_action_t action,
item_action_t *actions)
{
int i;
struct item_head *ih;
item_action_t iaction;
ih = item_head(bh, 0);
for (i = 0; i < get_blkh_nr_items(B_BLK_HEAD(bh)); i++, ih++) {
if (action)
action(ih);
iaction = actions[get_type(&ih->ih_key)];
if (iaction)
iaction(bh, ih);
}
}
static inline int is_key_format_1(int type)
{
return ((type == 0 || type == 15) ? 1 : 0);
}
/* old keys (on i386) have k_offset_v2.k_type == 15 (direct and
indirect) or == 0 (dir items and stat data) */
/* */
int key_format(const struct reiserfs_key *key)
{
int type;
type = get_key_type_v2(key);
if (is_key_format_1(type))
return KEY_FORMAT_1;
return KEY_FORMAT_2;
}
unsigned long long get_offset(const struct reiserfs_key *key)
{
if (key_format(key) == KEY_FORMAT_1)
return get_key_offset_v1(key);
return get_key_offset_v2(key);
}
int uniqueness2type(__u32 uniqueness)
{
switch (uniqueness) {
case V1_SD_UNIQUENESS:
return TYPE_STAT_DATA;
case V1_INDIRECT_UNIQUENESS:
return TYPE_INDIRECT;
case V1_DIRECT_UNIQUENESS:
return TYPE_DIRECT;
case V1_DIRENTRY_UNIQUENESS:
return TYPE_DIRENTRY;
}
return TYPE_UNKNOWN;
}
__u32 type2uniqueness(int type)
{
switch (type) {
case TYPE_STAT_DATA:
return V1_SD_UNIQUENESS;
case TYPE_INDIRECT:
return V1_INDIRECT_UNIQUENESS;
case TYPE_DIRECT:
return V1_DIRECT_UNIQUENESS;
case TYPE_DIRENTRY:
return V1_DIRENTRY_UNIQUENESS;
}
return V1_UNKNOWN_UNIQUENESS;
}
int get_type(const struct reiserfs_key *key)
{
int type_v2 = get_key_type_v2(key);
if (is_key_format_1(type_v2))
return uniqueness2type(get_key_uniqueness(key));
return type_v2;
}
char *key_of_what(const struct reiserfs_key *key)
{
switch (get_type(key)) {
case TYPE_STAT_DATA:
return "SD";
case TYPE_INDIRECT:
return "IND";
case TYPE_DIRECT:
return "DRCT";
case TYPE_DIRENTRY:
return "DIR";
default:
return "???";
}
}
int type_unknown(const struct reiserfs_key *key)
{
int type = get_type(key);
switch (type) {
case TYPE_STAT_DATA:
case TYPE_INDIRECT:
case TYPE_DIRECT:
case TYPE_DIRENTRY:
return 0;
default:
break;
}
return 1;
}
// this sets key format as well as type of item key belongs to
//
void set_type(int format, struct reiserfs_key *key, int type)
{
if (format == KEY_FORMAT_1)
set_key_uniqueness(key, type2uniqueness(type));
else
set_key_type_v2(key, type);
}
//
void set_offset(int format, struct reiserfs_key *key, loff_t offset)
{
if (format == KEY_FORMAT_1)
set_key_offset_v1(key, offset);
else
set_key_offset_v2(key, offset);
}
void set_type_and_offset(int format, struct reiserfs_key *key, loff_t offset,
int type)
{
set_type(format, key, type);
set_offset(format, key, offset);
}
/* length of the directory entry in directory item. This define calculates
length of i-th directory entry using directory entry locations from dir
entry head. When it calculates length of 0-th directory entry, it uses
length of whole item in place of entry location of the non-existent
following entry in the calculation. See picture above.*/
// NOTE: this is not name length. This is length of whole entry
int entry_length(const struct item_head *ih, const struct reiserfs_de_head *deh,
int pos_in_item)
{
if (pos_in_item)
return (get_deh_location(deh - 1) - get_deh_location(deh));
return (get_ih_item_len(ih) - get_deh_location(deh));
}
char *name_in_entry(const struct reiserfs_de_head *deh, int pos_in_item)
{
return ((char *)(deh - pos_in_item) + get_deh_location(deh));
}
int name_in_entry_length(const struct item_head *ih,
const struct reiserfs_de_head *deh, int pos_in_item)
{
int len, i;
char *name;
len = entry_length(ih, deh, pos_in_item);
name = name_in_entry(deh, pos_in_item);
// name might be padded with 0s
i = 0;
while (name[i] && i < len)
i++;
return i;
}
int name_length(const char *name, int key_format)
{
if (key_format == KEY_FORMAT_2)
return ROUND_UP(strlen(name));
else if (key_format == KEY_FORMAT_1)
return strlen(name);
return -1;
}
/* key format is stored in 12 bits starting from 0-th of item_head's ih2_format*/
__u16 get_ih_key_format(const struct item_head * ih)
{
get_bit_field_XX(16, &ih->ih_format, 0, 12);
}
__u16 get_ih_flags(const struct item_head *ih)
{
get_bit_field_XX(16, &ih->ih_format, 12, 4);
}
void set_ih_key_format(struct item_head *ih, __u16 val)
{
set_bit_field_XX(16, &ih->ih_format, val, 0, 12);
}
void set_ih_flags(struct item_head *ih, __u16 val)
{
set_bit_field_XX(16, &ih->ih_format, val, 12, 4);
}
/* access to fields of stat data (both v1 and v2) */
void get_set_sd_field(int field, struct item_head *ih, void *sd,
void *value, int set)
{
if (get_ih_key_format(ih) == KEY_FORMAT_1) {
struct stat_data_v1 *sd_v1 = sd;
switch (field) {
case GET_SD_MODE:
if (set)
sd_v1->sd_mode = cpu_to_le16(*(__u16 *) value);
else
*(__u16 *) value = le16_to_cpu(sd_v1->sd_mode);
break;
case GET_SD_SIZE:
/* value must point to 64 bit int */
if (set)
sd_v1->sd_size = cpu_to_le32(*(__u64 *) value);
else
*(__u64 *) value = le32_to_cpu(sd_v1->sd_size);
break;
case GET_SD_BLOCKS:
if (set)
sd_v1->u.sd_blocks =
cpu_to_le32(*(__u32 *) value);
else
*(__u32 *) value =
le32_to_cpu(sd_v1->u.sd_blocks);
break;
case GET_SD_NLINK:
/* value must point to 32 bit int */
if (set)
sd_v1->sd_nlink = cpu_to_le16(*(__u32 *) value);
else
*(__u32 *) value = le16_to_cpu(sd_v1->sd_nlink);
break;
case GET_SD_FIRST_DIRECT_BYTE:
if (set)
sd_v1->sd_first_direct_byte =
cpu_to_le32(*(__u32 *) value);
else
*(__u32 *) value =
le32_to_cpu(sd_v1->sd_first_direct_byte);
break;
default:
reiserfs_panic
("get_set_sd_field: unknown field of old stat data");
}
} else {
struct stat_data *sd_v2 = sd;
switch (field) {
case GET_SD_MODE:
if (set)
sd_v2->sd_mode = cpu_to_le16(*(__u16 *) value);
else
*(__u16 *) value = le16_to_cpu(sd_v2->sd_mode);
break;
case GET_SD_SIZE:
if (set)
sd_v2->sd_size = cpu_to_le64(*(__u64 *) value);
else
*(__u64 *) value = le64_to_cpu(sd_v2->sd_size);
break;
case GET_SD_BLOCKS:
if (set)
sd_v2->sd_blocks =
cpu_to_le32(*(__u32 *) value);
else
*(__u32 *) value =
le32_to_cpu(sd_v2->sd_blocks);
break;
case GET_SD_NLINK:
if (set)
sd_v2->sd_nlink = cpu_to_le32(*(__u32 *) value);
else
*(__u32 *) value = le32_to_cpu(sd_v2->sd_nlink);
break;
case GET_SD_FIRST_DIRECT_BYTE:
default:
reiserfs_panic
("get_set_sd_field: unknown field of new stat data");
}
}
}
static int comp_ids(const void *p1, const void *p2)
{
__u32 id1 = le32_to_cpu(*(__le32 *) p1);
__u32 id2 = le32_to_cpu(*(__le32 *) p2);
if (id1 < id2)
return -1;
if (id1 > id2)
return 1;
return 0;
}
/* functions to manipulate with super block's objectid map */
int is_objectid_used(reiserfs_filsys_t fs, __u32 objectid)
{
__le32 *objectid_map;
__u32 count = get_sb_oid_cursize(fs->fs_ondisk_sb);
int ret;
__u32 pos;
__le32 le_id = cpu_to_le32(objectid);
objectid_map =
(__le32 *) ((char *)fs->fs_ondisk_sb +
reiserfs_super_block_size(fs->fs_ondisk_sb));
ret =
reiserfs_bin_search(&le_id, objectid_map, count, sizeof(__u32),
&pos, comp_ids);
/* if the position returned is odd, the oid is in use */
if (ret == POSITION_NOT_FOUND)
return (pos & 1);
/* if the position returned is even, the oid is in use */
return !(pos & 1);
}
void mark_objectid_used(reiserfs_filsys_t fs, __u32 objectid)
{
int i;
__le32 *objectid_map;
int cursize;
if (is_objectid_used(fs, objectid)) {
return;
}
objectid_map =
(__le32 *) ((char *)fs->fs_ondisk_sb +
reiserfs_super_block_size(fs->fs_ondisk_sb));
cursize = get_sb_oid_cursize(fs->fs_ondisk_sb);
for (i = 0; i < cursize; i += 2) {
if (objectid >= le32_to_cpu(objectid_map[i]) &&
objectid < le32_to_cpu(objectid_map[i + 1]))
/* it is used */
return;
if (objectid + 1 == le32_to_cpu(objectid_map[i])) {
/* size of objectid map does not change */
objectid_map[i] = cpu_to_le32(objectid);
return;
}
if (objectid == le32_to_cpu(objectid_map[i + 1])) {
/* size of objectid map is decreased */
objectid_map[i + 1] =
cpu_to_le32(le32_to_cpu(objectid_map[i + 1]) + 1);
if (i + 2 < cursize) {
if (objectid_map[i + 1] == objectid_map[i + 2]) {
memmove(objectid_map + i + 1,
objectid_map + i + 1 + 2,
(cursize -
(i + 2 + 2 -
1)) * sizeof(__u32));
set_sb_oid_cursize(fs->fs_ondisk_sb,
cursize - 2);
}
}
return;
}
if (objectid < le32_to_cpu(objectid_map[i])) {
/* size of objectid map must be increased */
if (cursize == get_sb_oid_maxsize(fs->fs_ondisk_sb)) {
/* here all objectids between objectid and objectid_map[i] get
used */
objectid_map[i] = cpu_to_le32(objectid);
return;
} else {
memmove(objectid_map + i + 2, objectid_map + i,
(cursize - i) * sizeof(__u32));
set_sb_oid_cursize(fs->fs_ondisk_sb,
cursize + 2);
}
objectid_map[i] = cpu_to_le32(objectid);
objectid_map[i + 1] = cpu_to_le32(objectid + 1);
return;
}
}
/* append to current objectid map, if we have space */
if (i < get_sb_oid_maxsize(fs->fs_ondisk_sb)) {
objectid_map[i] = cpu_to_le32(objectid);
objectid_map[i + 1] = cpu_to_le32(objectid + 1);
set_sb_oid_cursize(fs->fs_ondisk_sb, cursize + 2);
} else if (i == get_sb_oid_maxsize(fs->fs_ondisk_sb)) {
objectid_map[i - 1] = cpu_to_le32(objectid + 1);
} else
die("mark_objectid_as_used: objectid map corrupted");
return;
}
int is_blocksize_correct(unsigned int blocksize)
{
return ((((blocksize & -blocksize) == blocksize)
&& (blocksize >= 512) && (blocksize <= 8192)));
}