1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24#include <linux/fs.h>
25#include <linux/sysv_fs.h>
26#include <linux/locks.h>
27#include <linux/smp_lock.h>
28#include <linux/highuid.h>
29#include <asm/byteorder.h>
30
31
32static void sysv_write_super(struct super_block *sb)
33{
34 if (!(sb->s_flags & MS_RDONLY)) {
35
36
37
38 unsigned long time = CURRENT_TIME;
39 unsigned long old_time = fs32_to_cpu(sb, *sb->sv_sb_time);
40 if (sb->sv_type == FSTYPE_SYSV4)
41 if (*sb->sv_sb_state == cpu_to_fs32(sb, 0x7c269d38 - old_time))
42 *sb->sv_sb_state = cpu_to_fs32(sb, 0x7c269d38 - time);
43 *sb->sv_sb_time = cpu_to_fs32(sb, time);
44 mark_buffer_dirty(sb->sv_bh2);
45 }
46 sb->s_dirt = 0;
47}
48
49static void sysv_put_super(struct super_block *sb)
50{
51 if (!(sb->s_flags & MS_RDONLY)) {
52
53 mark_buffer_dirty(sb->sv_bh1);
54 if (sb->sv_bh1 != sb->sv_bh2)
55 mark_buffer_dirty(sb->sv_bh2);
56 }
57
58 brelse(sb->sv_bh1);
59 if (sb->sv_bh1 != sb->sv_bh2)
60 brelse(sb->sv_bh2);
61}
62
63static int sysv_statfs(struct super_block *sb, struct statfs *buf)
64{
65 buf->f_type = sb->s_magic;
66 buf->f_bsize = sb->s_blocksize;
67 buf->f_blocks = sb->sv_ndatazones;
68 buf->f_bavail = buf->f_bfree = sysv_count_free_blocks(sb);
69 buf->f_files = sb->sv_ninodes;
70 buf->f_ffree = sysv_count_free_inodes(sb);
71 buf->f_namelen = SYSV_NAMELEN;
72 return 0;
73}
74
75
76
77
78static inline void read3byte(struct super_block *sb,
79 unsigned char * from, unsigned char * to)
80{
81 if (sb->sv_bytesex == BYTESEX_PDP) {
82 to[0] = from[0];
83 to[1] = 0;
84 to[2] = from[1];
85 to[3] = from[2];
86 } else if (sb->sv_bytesex == BYTESEX_LE) {
87 to[0] = from[0];
88 to[1] = from[1];
89 to[2] = from[2];
90 to[3] = 0;
91 } else {
92 to[0] = 0;
93 to[1] = from[0];
94 to[2] = from[1];
95 to[3] = from[2];
96 }
97}
98
99static inline void write3byte(struct super_block *sb,
100 unsigned char * from, unsigned char * to)
101{
102 if (sb->sv_bytesex == BYTESEX_PDP) {
103 to[0] = from[0];
104 to[1] = from[2];
105 to[2] = from[3];
106 } else if (sb->sv_bytesex == BYTESEX_LE) {
107 to[0] = from[0];
108 to[1] = from[1];
109 to[2] = from[2];
110 } else {
111 to[0] = from[1];
112 to[1] = from[2];
113 to[2] = from[3];
114 }
115}
116
117static struct inode_operations sysv_symlink_inode_operations = {
118 readlink: page_readlink,
119 follow_link: page_follow_link,
120};
121
122void sysv_set_inode(struct inode *inode, dev_t rdev)
123{
124 if (S_ISREG(inode->i_mode)) {
125 inode->i_op = &sysv_file_inode_operations;
126 inode->i_fop = &sysv_file_operations;
127 inode->i_mapping->a_ops = &sysv_aops;
128 } else if (S_ISDIR(inode->i_mode)) {
129 inode->i_op = &sysv_dir_inode_operations;
130 inode->i_fop = &sysv_dir_operations;
131 inode->i_mapping->a_ops = &sysv_aops;
132 } else if (S_ISLNK(inode->i_mode)) {
133 if (inode->i_blocks) {
134 inode->i_op = &sysv_symlink_inode_operations;
135 inode->i_mapping->a_ops = &sysv_aops;
136 } else
137 inode->i_op = &sysv_fast_symlink_inode_operations;
138 } else
139 init_special_inode(inode, inode->i_mode, rdev);
140}
141
142static void sysv_read_inode(struct inode *inode)
143{
144 struct super_block * sb = inode->i_sb;
145 struct buffer_head * bh;
146 struct sysv_inode * raw_inode;
147 unsigned int block, ino;
148 dev_t rdev = 0;
149
150 ino = inode->i_ino;
151 if (!ino || ino > sb->sv_ninodes) {
152 printk("Bad inode number on dev %s"
153 ": %d is out of range\n",
154 kdevname(inode->i_dev), ino);
155 goto bad_inode;
156 }
157 raw_inode = sysv_raw_inode(sb, ino, &bh);
158 if (!raw_inode) {
159 printk("Major problem: unable to read inode from dev %s\n",
160 bdevname(inode->i_dev));
161 goto bad_inode;
162 }
163
164 inode->i_mode = fs16_to_cpu(sb, raw_inode->i_mode);
165 inode->i_uid = (uid_t)fs16_to_cpu(sb, raw_inode->i_uid);
166 inode->i_gid = (gid_t)fs16_to_cpu(sb, raw_inode->i_gid);
167 inode->i_nlink = fs16_to_cpu(sb, raw_inode->i_nlink);
168 inode->i_size = fs32_to_cpu(sb, raw_inode->i_size);
169 inode->i_atime = fs32_to_cpu(sb, raw_inode->i_atime);
170 inode->i_mtime = fs32_to_cpu(sb, raw_inode->i_mtime);
171 inode->i_ctime = fs32_to_cpu(sb, raw_inode->i_ctime);
172 inode->i_blocks = inode->i_blksize = 0;
173 for (block = 0; block < 10+1+1+1; block++)
174 read3byte(sb, &raw_inode->i_a.i_addb[3*block],
175 (unsigned char*)&inode->u.sysv_i.i_data[block]);
176 brelse(bh);
177 if (S_ISCHR(inode->i_mode) || S_ISBLK(inode->i_mode))
178 rdev = (u16)fs32_to_cpu(sb, inode->u.sysv_i.i_data[0]);
179 inode->u.sysv_i.i_dir_start_lookup = 0;
180 sysv_set_inode(inode, rdev);
181 return;
182
183bad_inode:
184 make_bad_inode(inode);
185 return;
186}
187
188static struct buffer_head * sysv_update_inode(struct inode * inode)
189{
190 struct super_block * sb = inode->i_sb;
191 struct buffer_head * bh;
192 struct sysv_inode * raw_inode;
193 unsigned int ino, block;
194
195 ino = inode->i_ino;
196 if (!ino || ino > sb->sv_ninodes) {
197 printk("Bad inode number on dev %s: %d is out of range\n",
198 bdevname(inode->i_dev), ino);
199 return 0;
200 }
201 raw_inode = sysv_raw_inode(sb, ino, &bh);
202 if (!raw_inode) {
203 printk("unable to read i-node block\n");
204 return 0;
205 }
206
207 raw_inode->i_mode = cpu_to_fs16(sb, inode->i_mode);
208 raw_inode->i_uid = cpu_to_fs16(sb, fs_high2lowuid(inode->i_uid));
209 raw_inode->i_gid = cpu_to_fs16(sb, fs_high2lowgid(inode->i_gid));
210 raw_inode->i_nlink = cpu_to_fs16(sb, inode->i_nlink);
211 raw_inode->i_size = cpu_to_fs32(sb, inode->i_size);
212 raw_inode->i_atime = cpu_to_fs32(sb, inode->i_atime);
213 raw_inode->i_mtime = cpu_to_fs32(sb, inode->i_mtime);
214 raw_inode->i_ctime = cpu_to_fs32(sb, inode->i_ctime);
215 if (S_ISCHR(inode->i_mode) || S_ISBLK(inode->i_mode))
216 inode->u.sysv_i.i_data[0] =
217 cpu_to_fs32(sb, kdev_t_to_nr(inode->i_rdev));
218 for (block = 0; block < 10+1+1+1; block++)
219 write3byte(sb, (unsigned char*)&inode->u.sysv_i.i_data[block],
220 &raw_inode->i_a.i_addb[3*block]);
221 mark_buffer_dirty(bh);
222 return bh;
223}
224
225void sysv_write_inode(struct inode * inode, int wait)
226{
227 struct buffer_head *bh;
228 lock_kernel();
229 bh = sysv_update_inode(inode);
230 brelse(bh);
231 unlock_kernel();
232}
233
234int sysv_sync_inode(struct inode * inode)
235{
236 int err = 0;
237 struct buffer_head *bh;
238
239 bh = sysv_update_inode(inode);
240 if (bh && buffer_dirty(bh)) {
241 ll_rw_block(WRITE, 1, &bh);
242 wait_on_buffer(bh);
243 if (buffer_req(bh) && !buffer_uptodate(bh)) {
244 printk ("IO error syncing sysv inode [%s:%08lx]\n",
245 bdevname(inode->i_dev), inode->i_ino);
246 err = -1;
247 }
248 }
249 else if (!bh)
250 err = -1;
251 brelse (bh);
252 return err;
253}
254
255static void sysv_delete_inode(struct inode *inode)
256{
257 lock_kernel();
258 inode->i_size = 0;
259 sysv_truncate(inode);
260 sysv_free_inode(inode);
261 unlock_kernel();
262}
263
264struct super_operations sysv_sops = {
265 read_inode: sysv_read_inode,
266 write_inode: sysv_write_inode,
267 delete_inode: sysv_delete_inode,
268 put_super: sysv_put_super,
269 write_super: sysv_write_super,
270 statfs: sysv_statfs,
271};
272