ftp://ftp.kernel.org/pub/linux/kernel/v2.6/linux-2.6.6.tar.bz2
[linux-2.6.git] / fs / befs / io.c
1 /*
2  * linux/fs/befs/io.c
3  *
4  * Copyright (C) 2001 Will Dyson <will_dyson@pobox.com
5  *
6  * Based on portions of file.c and inode.c 
7  * by Makoto Kato (m_kato@ga2.so-net.ne.jp)
8  *
9  * Many thanks to Dominic Giampaolo, author of Practical File System
10  * Design with the Be File System, for such a helpful book.
11  *
12  */
13
14 #include <linux/buffer_head.h>
15
16 #include "befs.h"
17 #include "io.h"
18
19 /*
20  * Converts befs notion of disk addr to a disk offset and uses
21  * linux kernel function sb_bread() to get the buffer containing
22  * the offset. -Will Dyson
23  *
24  */
25
26 struct buffer_head *
27 befs_bread_iaddr(struct super_block *sb, befs_inode_addr iaddr)
28 {
29         struct buffer_head *bh = NULL;
30         befs_blocknr_t block = 0;
31         vfs_blocknr_t vfs_block = 0;
32         befs_sb_info *befs_sb = BEFS_SB(sb);
33
34         befs_debug(sb, "---> Enter befs_read_iaddr() "
35                    "[%u, %hu, %hu]",
36                    iaddr.allocation_group, iaddr.start, iaddr.len);
37
38         if (iaddr.allocation_group > befs_sb->num_ags) {
39                 befs_error(sb, "BEFS: Invalid allocation group %u, max is %u",
40                            iaddr.allocation_group, befs_sb->num_ags);
41                 goto error;
42         }
43
44         block = iaddr2blockno(sb, &iaddr);
45         vfs_block = (vfs_blocknr_t) block;
46
47         if (vfs_block != block) {
48                 befs_error(sb, "Error converting to host blocknr_t. %Lu "
49                            "is larger than the host can use", block);
50                 goto error;
51         }
52
53         befs_debug(sb, "befs_read_iaddr: offset = %lu", block);
54
55         bh = sb_bread(sb, vfs_block);
56
57         if (bh == NULL) {
58                 befs_error(sb, "Failed to read block %lu", block);
59                 goto error;
60         }
61
62         befs_debug(sb, "<--- befs_read_iaddr()");
63         return bh;
64
65       error:
66         befs_debug(sb, "<--- befs_read_iaddr() ERROR");
67         return NULL;
68 }
69
70 struct buffer_head *
71 befs_bread(struct super_block *sb, befs_blocknr_t block)
72 {
73         struct buffer_head *bh = NULL;
74         vfs_blocknr_t vfs_block = (vfs_blocknr_t) block;
75
76         befs_debug(sb, "---> Enter befs_read() %Lu", block);
77
78         if (vfs_block != block) {
79                 befs_error(sb, "Error converting to host blocknr_t. %Lu "
80                            "is larger than the host can use", block);
81                 goto error;
82         }
83
84         bh = sb_bread(sb, vfs_block);
85
86         if (bh == NULL) {
87                 befs_error(sb, "Failed to read block %lu", vfs_block);
88                 goto error;
89         }
90
91         befs_debug(sb, "<--- befs_read()");
92
93         return bh;
94
95       error:
96         befs_debug(sb, "<--- befs_read() ERROR");
97         return NULL;
98 }