blob: cb3a104c781854fe3575484a98d338a8e89016b4 [file] [log] [blame]
Theodore Ts'o50e1e101997-04-26 13:58:21 +00001/*
2 * dirblock.c --- directory block routines.
Theodore Ts'oefc6f622008-08-27 23:07:54 -04003 *
Theodore Ts'o21c84b71997-04-29 16:15:03 +00004 * Copyright (C) 1995, 1996 Theodore Ts'o.
5 *
6 * %Begin-Header%
Theodore Ts'o543547a2010-05-17 21:31:56 -04007 * This file may be redistributed under the terms of the GNU Library
8 * General Public License, version 2.
Theodore Ts'o21c84b71997-04-29 16:15:03 +00009 * %End-Header%
Theodore Ts'o50e1e101997-04-26 13:58:21 +000010 */
11
Theodore Ts'od1154eb2011-09-18 17:34:37 -040012#include "config.h"
Theodore Ts'o50e1e101997-04-26 13:58:21 +000013#include <stdio.h>
Theodore Ts'o4cbe8af1997-08-10 23:07:40 +000014#if HAVE_UNISTD_H
Theodore Ts'o50e1e101997-04-26 13:58:21 +000015#include <unistd.h>
Theodore Ts'o4cbe8af1997-08-10 23:07:40 +000016#endif
Theodore Ts'o21c84b71997-04-29 16:15:03 +000017#include <string.h>
Theodore Ts'o50e1e101997-04-26 13:58:21 +000018#include <time.h>
Theodore Ts'o50e1e101997-04-26 13:58:21 +000019
Theodore Ts'ob5abe6f1998-01-19 14:47:53 +000020#include "ext2_fs.h"
Theodore Ts'o50e1e101997-04-26 13:58:21 +000021#include "ext2fs.h"
22
Jose R. Santos57e24672009-06-01 16:15:40 -040023errcode_t ext2fs_read_dir_block3(ext2_filsys fs, blk64_t block,
Theodore Ts'o54434922003-12-07 01:28:50 -050024 void *buf, int flags EXT2FS_ATTR((unused)))
Theodore Ts'o50e1e101997-04-26 13:58:21 +000025{
26 errcode_t retval;
27 char *p, *end;
28 struct ext2_dir_entry *dirent;
Theodore Ts'o126a2912007-08-11 01:56:48 -040029 unsigned int name_len, rec_len;
Theodore Ts'oefc6f622008-08-27 23:07:54 -040030
Theodore Ts'o50e1e101997-04-26 13:58:21 +000031
Jose R. Santos57e24672009-06-01 16:15:40 -040032 retval = io_channel_read_blk64(fs->io, block, 1, buf);
Theodore Ts'o50e1e101997-04-26 13:58:21 +000033 if (retval)
34 return retval;
Theodore Ts'o126a2912007-08-11 01:56:48 -040035
Theodore Ts'ob5abe6f1998-01-19 14:47:53 +000036 p = (char *) buf;
Theodore Ts'o50e1e101997-04-26 13:58:21 +000037 end = (char *) buf + fs->blocksize;
Theodore Ts'oe446d712001-01-05 22:27:06 +000038 while (p < end-8) {
Theodore Ts'obfcd9a12001-05-05 16:21:48 +000039 dirent = (struct ext2_dir_entry *) p;
Theodore Ts'o126a2912007-08-11 01:56:48 -040040#ifdef WORDS_BIGENDIAN
41 dirent->inode = ext2fs_swab32(dirent->inode);
42 dirent->rec_len = ext2fs_swab16(dirent->rec_len);
43 dirent->name_len = ext2fs_swab16(dirent->name_len);
Theodore Ts'o5df55d72001-06-11 07:00:04 +000044#endif
Theodore Ts'of9190c82002-03-12 01:05:06 -050045 name_len = dirent->name_len;
46#ifdef WORDS_BIGENDIAN
47 if (flags & EXT2_DIRBLOCK_V2_STRUCT)
48 dirent->name_len = ext2fs_swab16(dirent->name_len);
49#endif
Theodore Ts'o8a480352009-06-21 21:07:38 -040050 if ((retval = ext2fs_get_rec_len(fs, dirent, &rec_len)) != 0)
51 return retval;
Theodore Ts'o1cca4d62001-05-05 05:12:14 +000052 if ((rec_len < 8) || (rec_len % 4)) {
53 rec_len = 8;
54 retval = EXT2_ET_DIR_CORRUPTED;
Theodore Ts'o5dd77db2008-08-25 21:08:19 -040055 } else if (((name_len & 0xFF) + 8) > rec_len)
Theodore Ts'o1cca4d62001-05-05 05:12:14 +000056 retval = EXT2_ET_DIR_CORRUPTED;
57 p += rec_len;
Theodore Ts'o50e1e101997-04-26 13:58:21 +000058 }
Theodore Ts'o1cca4d62001-05-05 05:12:14 +000059 return retval;
Theodore Ts'o50e1e101997-04-26 13:58:21 +000060}
61
Jose R. Santos57e24672009-06-01 16:15:40 -040062errcode_t ext2fs_read_dir_block2(ext2_filsys fs, blk_t block,
63 void *buf, int flags EXT2FS_ATTR((unused)))
64{
65 return ext2fs_read_dir_block3(fs, block, buf, flags);
66}
67
Theodore Ts'of9190c82002-03-12 01:05:06 -050068errcode_t ext2fs_read_dir_block(ext2_filsys fs, blk_t block,
69 void *buf)
Theodore Ts'o50e1e101997-04-26 13:58:21 +000070{
Jose R. Santos57e24672009-06-01 16:15:40 -040071 return ext2fs_read_dir_block3(fs, block, buf, 0);
Theodore Ts'of9190c82002-03-12 01:05:06 -050072}
73
74
Jose R. Santos57e24672009-06-01 16:15:40 -040075errcode_t ext2fs_write_dir_block3(ext2_filsys fs, blk64_t block,
Theodore Ts'o54434922003-12-07 01:28:50 -050076 void *inbuf, int flags EXT2FS_ATTR((unused)))
Theodore Ts'of9190c82002-03-12 01:05:06 -050077{
Theodore Ts'o126a2912007-08-11 01:56:48 -040078#ifdef WORDS_BIGENDIAN
Theodore Ts'o50e1e101997-04-26 13:58:21 +000079 errcode_t retval;
Theodore Ts'of9190c82002-03-12 01:05:06 -050080 char *p, *end;
Theodore Ts'o50e1e101997-04-26 13:58:21 +000081 char *buf = 0;
Theodore Ts'o8a480352009-06-21 21:07:38 -040082 unsigned int rec_len;
Theodore Ts'o50e1e101997-04-26 13:58:21 +000083 struct ext2_dir_entry *dirent;
84
Theodore Ts'oc4e3d3f2003-08-01 09:41:07 -040085 retval = ext2fs_get_mem(fs->blocksize, &buf);
Theodore Ts'of9190c82002-03-12 01:05:06 -050086 if (retval)
87 return retval;
88 memcpy(buf, inbuf, fs->blocksize);
89 p = buf;
90 end = buf + fs->blocksize;
91 while (p < end) {
92 dirent = (struct ext2_dir_entry *) p;
Theodore Ts'o8a480352009-06-21 21:07:38 -040093 if ((retval = ext2fs_get_rec_len(fs, dirent, &rec_len)) != 0)
94 return retval;
Theodore Ts'o5dd77db2008-08-25 21:08:19 -040095 if ((rec_len < 8) ||
96 (rec_len % 4)) {
Theodore Ts'oc4e3d3f2003-08-01 09:41:07 -040097 ext2fs_free_mem(&buf);
Theodore Ts'of9190c82002-03-12 01:05:06 -050098 return (EXT2_ET_DIR_CORRUPTED);
99 }
Theodore Ts'o5dd77db2008-08-25 21:08:19 -0400100 p += rec_len;
Theodore Ts'o126a2912007-08-11 01:56:48 -0400101 dirent->inode = ext2fs_swab32(dirent->inode);
102 dirent->rec_len = ext2fs_swab16(dirent->rec_len);
103 dirent->name_len = ext2fs_swab16(dirent->name_len);
104
Theodore Ts'of9190c82002-03-12 01:05:06 -0500105 if (flags & EXT2_DIRBLOCK_V2_STRUCT)
106 dirent->name_len = ext2fs_swab16(dirent->name_len);
Theodore Ts'of9190c82002-03-12 01:05:06 -0500107 }
Jose R. Santos57e24672009-06-01 16:15:40 -0400108 retval = io_channel_write_blk64(fs->io, block, 1, buf);
Theodore Ts'oc4e3d3f2003-08-01 09:41:07 -0400109 ext2fs_free_mem(&buf);
Theodore Ts'o50e1e101997-04-26 13:58:21 +0000110 return retval;
Theodore Ts'of9190c82002-03-12 01:05:06 -0500111#else
Valerie Aurora Henson24a117a2009-09-07 21:14:24 -0400112 return io_channel_write_blk64(fs->io, block, 1, (char *) inbuf);
Theodore Ts'of9190c82002-03-12 01:05:06 -0500113#endif
Theodore Ts'o50e1e101997-04-26 13:58:21 +0000114}
115
Jose R. Santos57e24672009-06-01 16:15:40 -0400116errcode_t ext2fs_write_dir_block2(ext2_filsys fs, blk_t block,
117 void *inbuf, int flags EXT2FS_ATTR((unused)))
118{
119 return ext2fs_write_dir_block3(fs, block, inbuf, flags);
120}
Theodore Ts'o50e1e101997-04-26 13:58:21 +0000121
Theodore Ts'of9190c82002-03-12 01:05:06 -0500122errcode_t ext2fs_write_dir_block(ext2_filsys fs, blk_t block,
123 void *inbuf)
124{
Jose R. Santos57e24672009-06-01 16:15:40 -0400125 return ext2fs_write_dir_block3(fs, block, inbuf, 0);
Theodore Ts'of9190c82002-03-12 01:05:06 -0500126}
127