tzh
2024-08-22 c7d0944258c7d0943aa7b2211498fd612971ce27
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
/*
 * newdir.c --- create a new directory block
 *
 * Copyright (C) 1994, 1995 Theodore Ts'o.
 *
 * %Begin-Header%
 * This file may be redistributed under the terms of the GNU Library
 * General Public License, version 2.
 * %End-Header%
 */
 
#include "config.h"
#include <stdio.h>
#include <string.h>
#if HAVE_UNISTD_H
#include <unistd.h>
#endif
 
#include "ext2_fs.h"
#include "ext2fs.h"
 
#ifndef EXT2_FT_DIR
#define EXT2_FT_DIR        2
#endif
 
/*
 * Create new directory block
 */
errcode_t ext2fs_new_dir_block(ext2_filsys fs, ext2_ino_t dir_ino,
                  ext2_ino_t parent_ino, char **block)
{
   struct ext2_dir_entry     *dir = NULL;
   errcode_t        retval;
   char            *buf;
   int            rec_len;
   int            filetype = 0;
   struct ext2_dir_entry_tail    *t;
   int            csum_size = 0;
 
   EXT2_CHECK_MAGIC(fs, EXT2_ET_MAGIC_EXT2FS_FILSYS);
 
   retval = ext2fs_get_mem(fs->blocksize, &buf);
   if (retval)
       return retval;
   memset(buf, 0, fs->blocksize);
   dir = (struct ext2_dir_entry *) buf;
 
   if (ext2fs_has_feature_metadata_csum(fs->super))
       csum_size = sizeof(struct ext2_dir_entry_tail);
 
   retval = ext2fs_set_rec_len(fs, fs->blocksize - csum_size, dir);
   if (retval) {
       ext2fs_free_mem(&buf);
       return retval;
   }
 
   if (dir_ino) {
       if (ext2fs_has_feature_filetype(fs->super))
           filetype = EXT2_FT_DIR;
       /*
        * Set up entry for '.'
        */
       dir->inode = dir_ino;
       ext2fs_dirent_set_name_len(dir, 1);
       ext2fs_dirent_set_file_type(dir, filetype);
       dir->name[0] = '.';
       rec_len = (fs->blocksize - csum_size) - EXT2_DIR_REC_LEN(1);
       dir->rec_len = EXT2_DIR_REC_LEN(1);
 
       /*
        * Set up entry for '..'
        */
       dir = (struct ext2_dir_entry *) (buf + dir->rec_len);
       retval = ext2fs_set_rec_len(fs, rec_len, dir);
       if (retval) {
           ext2fs_free_mem(&buf);
           return retval;
       }
       dir->inode = parent_ino;
       ext2fs_dirent_set_name_len(dir, 2);
       ext2fs_dirent_set_file_type(dir, filetype);
       dir->name[0] = '.';
       dir->name[1] = '.';
 
   }
 
   if (csum_size) {
       t = EXT2_DIRENT_TAIL(buf, fs->blocksize);
       ext2fs_initialize_dirent_tail(fs, t);
   }
   *block = buf;
   return 0;
}
 
/*
 * Create new directory on inline data
 */
errcode_t ext2fs_new_dir_inline_data(ext2_filsys fs,
                    ext2_ino_t dir_ino EXT2FS_ATTR((unused)),
                    ext2_ino_t parent_ino, __u32 *iblock)
{
   struct ext2_dir_entry     *dir = NULL;
   errcode_t        retval;
   int            rec_len;
 
   EXT2_CHECK_MAGIC(fs, EXT2_ET_MAGIC_EXT2FS_FILSYS);
 
   iblock[0] = ext2fs_cpu_to_le32(parent_ino);
 
   dir = (struct ext2_dir_entry *)((char *)iblock +
                   EXT4_INLINE_DATA_DOTDOT_SIZE);
   dir->inode = 0;
   rec_len = EXT4_MIN_INLINE_DATA_SIZE - EXT4_INLINE_DATA_DOTDOT_SIZE;
   retval = ext2fs_set_rec_len(fs, rec_len, dir);
   if (retval)
       goto errout;
 
#ifdef WORDS_BIGENDIAN
   retval = ext2fs_dirent_swab_out2(fs, (char *)dir, rec_len, 0);
   if (retval)
       goto errout;
#endif
 
errout:
   return retval;
}