blob: 0d6befed1ae54c39312c0e683925caf0c7b454a0 [file] [log] [blame]
David Teiglandb3b94fa2006-01-16 16:50:04 +00001/*
2 * Copyright (C) Sistina Software, Inc. 1997-2003 All rights reserved.
Steven Whitehouse3a8a9a12006-05-18 15:09:15 -04003 * Copyright (C) 2004-2006 Red Hat, Inc. All rights reserved.
David Teiglandb3b94fa2006-01-16 16:50:04 +00004 *
5 * This copyrighted material is made available to anyone wishing to use,
6 * modify, copy, or redistribute it subject to the terms and conditions
7 * of the GNU General Public License v.2.
8 */
9
10#include <linux/sched.h>
11#include <linux/slab.h>
12#include <linux/spinlock.h>
13#include <linux/completion.h>
14#include <linux/buffer_head.h>
15#include <linux/pagemap.h>
16#include <linux/mm.h>
Steven Whitehouse5c676f62006-02-27 17:23:27 -050017#include <linux/gfs2_ondisk.h>
David Teiglandb3b94fa2006-01-16 16:50:04 +000018
19#include "gfs2.h"
Steven Whitehouse5c676f62006-02-27 17:23:27 -050020#include "lm_interface.h"
21#include "incore.h"
David Teiglandb3b94fa2006-01-16 16:50:04 +000022#include "bmap.h"
23#include "inode.h"
24#include "page.h"
25#include "trans.h"
Steven Whitehouse257f9b42006-01-31 10:00:25 +000026#include "ops_address.h"
Steven Whitehouse5c676f62006-02-27 17:23:27 -050027#include "util.h"
David Teiglandb3b94fa2006-01-16 16:50:04 +000028
29/**
30 * gfs2_pte_inval - Sync and invalidate all PTEs associated with a glock
31 * @gl: the glock
32 *
33 */
34
35void gfs2_pte_inval(struct gfs2_glock *gl)
36{
37 struct gfs2_inode *ip;
38 struct inode *inode;
39
Steven Whitehouse5c676f62006-02-27 17:23:27 -050040 ip = gl->gl_object;
Steven Whitehousefeaa7bb2006-06-14 15:32:57 -040041 inode = &ip->i_inode;
David Teiglandb3b94fa2006-01-16 16:50:04 +000042 if (!ip || !S_ISREG(ip->i_di.di_mode))
43 return;
44
45 if (!test_bit(GIF_PAGED, &ip->i_flags))
46 return;
47
Steven Whitehousefeaa7bb2006-06-14 15:32:57 -040048 unmap_shared_mapping_range(inode->i_mapping, 0, 0);
David Teiglandb3b94fa2006-01-16 16:50:04 +000049
Steven Whitehousefeaa7bb2006-06-14 15:32:57 -040050 if (test_bit(GIF_SW_PAGED, &ip->i_flags))
51 set_bit(GLF_DIRTY, &gl->gl_flags);
David Teiglandb3b94fa2006-01-16 16:50:04 +000052
53 clear_bit(GIF_SW_PAGED, &ip->i_flags);
54}
55
56/**
57 * gfs2_page_inval - Invalidate all pages associated with a glock
58 * @gl: the glock
59 *
60 */
61
62void gfs2_page_inval(struct gfs2_glock *gl)
63{
64 struct gfs2_inode *ip;
65 struct inode *inode;
66
Steven Whitehouse5c676f62006-02-27 17:23:27 -050067 ip = gl->gl_object;
Steven Whitehousefeaa7bb2006-06-14 15:32:57 -040068 inode = &ip->i_inode;
David Teiglandb3b94fa2006-01-16 16:50:04 +000069 if (!ip || !S_ISREG(ip->i_di.di_mode))
70 return;
71
Steven Whitehousefeaa7bb2006-06-14 15:32:57 -040072 truncate_inode_pages(inode->i_mapping, 0);
73 gfs2_assert_withdraw(GFS2_SB(&ip->i_inode), !inode->i_mapping->nrpages);
David Teiglandb3b94fa2006-01-16 16:50:04 +000074 clear_bit(GIF_PAGED, &ip->i_flags);
75}
76
77/**
78 * gfs2_page_sync - Sync the data pages (not metadata) associated with a glock
79 * @gl: the glock
80 * @flags: DIO_START | DIO_WAIT
81 *
82 * Syncs data (not metadata) for a regular file.
83 * No-op for all other types.
84 */
85
86void gfs2_page_sync(struct gfs2_glock *gl, int flags)
87{
88 struct gfs2_inode *ip;
89 struct inode *inode;
Steven Whitehousefeaa7bb2006-06-14 15:32:57 -040090 struct address_space *mapping;
91 int error = 0;
David Teiglandb3b94fa2006-01-16 16:50:04 +000092
Steven Whitehouse5c676f62006-02-27 17:23:27 -050093 ip = gl->gl_object;
Steven Whitehousefeaa7bb2006-06-14 15:32:57 -040094 inode = &ip->i_inode;
David Teiglandb3b94fa2006-01-16 16:50:04 +000095 if (!ip || !S_ISREG(ip->i_di.di_mode))
96 return;
97
Steven Whitehousefeaa7bb2006-06-14 15:32:57 -040098 mapping = inode->i_mapping;
David Teiglandb3b94fa2006-01-16 16:50:04 +000099
Steven Whitehousefeaa7bb2006-06-14 15:32:57 -0400100 if (flags & DIO_START)
101 filemap_fdatawrite(mapping);
102 if (!error && (flags & DIO_WAIT))
103 error = filemap_fdatawait(mapping);
David Teiglandb3b94fa2006-01-16 16:50:04 +0000104
Steven Whitehousefeaa7bb2006-06-14 15:32:57 -0400105 /* Put back any errors cleared by filemap_fdatawait()
106 so they can be caught by someone who can pass them
107 up to user space. */
David Teiglandb3b94fa2006-01-16 16:50:04 +0000108
Steven Whitehousefeaa7bb2006-06-14 15:32:57 -0400109 if (error == -ENOSPC)
110 set_bit(AS_ENOSPC, &mapping->flags);
111 else if (error)
112 set_bit(AS_EIO, &mapping->flags);
David Teiglandb3b94fa2006-01-16 16:50:04 +0000113
David Teiglandb3b94fa2006-01-16 16:50:04 +0000114}
115
116/**
Steven Whitehouse257f9b42006-01-31 10:00:25 +0000117 * gfs2_block_truncate_page - Deal with zeroing out data for truncate
David Teiglandb3b94fa2006-01-16 16:50:04 +0000118 *
Steven Whitehouse257f9b42006-01-31 10:00:25 +0000119 * This is partly borrowed from ext3.
David Teiglandb3b94fa2006-01-16 16:50:04 +0000120 */
Steven Whitehouse257f9b42006-01-31 10:00:25 +0000121int gfs2_block_truncate_page(struct address_space *mapping)
David Teiglandb3b94fa2006-01-16 16:50:04 +0000122{
Steven Whitehouse257f9b42006-01-31 10:00:25 +0000123 struct inode *inode = mapping->host;
Steven Whitehousefeaa7bb2006-06-14 15:32:57 -0400124 struct gfs2_inode *ip = GFS2_I(inode);
125 struct gfs2_sbd *sdp = GFS2_SB(inode);
Steven Whitehouse257f9b42006-01-31 10:00:25 +0000126 loff_t from = inode->i_size;
127 unsigned long index = from >> PAGE_CACHE_SHIFT;
128 unsigned offset = from & (PAGE_CACHE_SIZE-1);
129 unsigned blocksize, iblock, length, pos;
David Teiglandb3b94fa2006-01-16 16:50:04 +0000130 struct buffer_head *bh;
Steven Whitehouse257f9b42006-01-31 10:00:25 +0000131 struct page *page;
David Teiglandb3b94fa2006-01-16 16:50:04 +0000132 void *kaddr;
Steven Whitehouse257f9b42006-01-31 10:00:25 +0000133 int err;
David Teiglandb3b94fa2006-01-16 16:50:04 +0000134
Steven Whitehouse257f9b42006-01-31 10:00:25 +0000135 page = grab_cache_page(mapping, index);
136 if (!page)
137 return 0;
David Teiglandb3b94fa2006-01-16 16:50:04 +0000138
Steven Whitehouse257f9b42006-01-31 10:00:25 +0000139 blocksize = inode->i_sb->s_blocksize;
140 length = blocksize - (offset & (blocksize - 1));
141 iblock = index << (PAGE_CACHE_SHIFT - inode->i_sb->s_blocksize_bits);
David Teiglandb3b94fa2006-01-16 16:50:04 +0000142
143 if (!page_has_buffers(page))
Steven Whitehouse257f9b42006-01-31 10:00:25 +0000144 create_empty_buffers(page, blocksize, 0);
David Teiglandb3b94fa2006-01-16 16:50:04 +0000145
Steven Whitehouse257f9b42006-01-31 10:00:25 +0000146 /* Find the buffer that contains "offset" */
147 bh = page_buffers(page);
148 pos = blocksize;
149 while (offset >= pos) {
150 bh = bh->b_this_page;
151 iblock++;
152 pos += blocksize;
153 }
David Teiglandb3b94fa2006-01-16 16:50:04 +0000154
Steven Whitehouse257f9b42006-01-31 10:00:25 +0000155 err = 0;
David Teiglandb3b94fa2006-01-16 16:50:04 +0000156
Steven Whitehouse257f9b42006-01-31 10:00:25 +0000157 if (!buffer_mapped(bh)) {
158 gfs2_get_block(inode, iblock, bh, 0);
159 /* unmapped? It's a hole - nothing to do */
160 if (!buffer_mapped(bh))
161 goto unlock;
162 }
163
164 /* Ok, it's mapped. Make sure it's up-to-date */
165 if (PageUptodate(page))
166 set_buffer_uptodate(bh);
167
168 if (!buffer_uptodate(bh)) {
169 err = -EIO;
170 ll_rw_block(READ, 1, &bh);
171 wait_on_buffer(bh);
172 /* Uhhuh. Read error. Complain and punt. */
173 if (!buffer_uptodate(bh))
174 goto unlock;
175 }
176
Steven Whitehouse18ec7d52006-02-08 11:50:51 +0000177 if (sdp->sd_args.ar_data == GFS2_DATA_ORDERED || gfs2_is_jdata(ip))
178 gfs2_trans_add_bh(ip->i_gl, bh, 0);
David Teiglandb3b94fa2006-01-16 16:50:04 +0000179
Steven Whitehouse257f9b42006-01-31 10:00:25 +0000180 kaddr = kmap_atomic(page, KM_USER0);
181 memset(kaddr + offset, 0, length);
182 flush_dcache_page(page);
183 kunmap_atomic(kaddr, KM_USER0);
184
185unlock:
David Teiglandb3b94fa2006-01-16 16:50:04 +0000186 unlock_page(page);
187 page_cache_release(page);
Steven Whitehouse257f9b42006-01-31 10:00:25 +0000188 return err;
David Teiglandb3b94fa2006-01-16 16:50:04 +0000189}
190
Steven Whitehouse257f9b42006-01-31 10:00:25 +0000191void gfs2_page_add_databufs(struct gfs2_inode *ip, struct page *page,
David Teiglandb3b94fa2006-01-16 16:50:04 +0000192 unsigned int from, unsigned int to)
193{
194 struct buffer_head *head = page_buffers(page);
195 unsigned int bsize = head->b_size;
196 struct buffer_head *bh;
197 unsigned int start, end;
198
Steven Whitehouse29937ac2006-07-06 17:58:03 -0400199 for (bh = head, start = 0; bh != head || !start;
David Teiglandb3b94fa2006-01-16 16:50:04 +0000200 bh = bh->b_this_page, start = end) {
201 end = start + bsize;
202 if (end <= from || start >= to)
203 continue;
Steven Whitehouse18ec7d52006-02-08 11:50:51 +0000204 gfs2_trans_add_bh(ip->i_gl, bh, 0);
David Teiglandb3b94fa2006-01-16 16:50:04 +0000205 }
206}
207