filemap_xip.c 10.1 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
/*
 *	linux/mm/filemap_xip.c
 *
 * Copyright (C) 2005 IBM Corporation
 * Author: Carsten Otte <cotte@de.ibm.com>
 *
 * derived from linux/mm/filemap.c - Copyright (C) Linus Torvalds
 *
 */

#include <linux/fs.h>
#include <linux/pagemap.h>
#include <linux/module.h>
#include <linux/uio.h>
#include <linux/rmap.h>
A
Alexey Dobriyan 已提交
16
#include <linux/sched.h>
17 18
#include <asm/tlbflush.h>

19 20 21 22 23 24 25 26 27
/*
 * We do use our own empty page to avoid interference with other users
 * of ZERO_PAGE(), such as /dev/zero
 */
static struct page *__xip_sparse_page;

static struct page *xip_sparse_page(void)
{
	if (!__xip_sparse_page) {
28 29 30
		struct page *page = alloc_page(GFP_HIGHUSER | __GFP_ZERO);

		if (page) {
31 32 33
			static DEFINE_SPINLOCK(xip_alloc_lock);
			spin_lock(&xip_alloc_lock);
			if (!__xip_sparse_page)
34
				__xip_sparse_page = page;
35
			else
36
				__free_page(page);
37 38 39 40 41 42
			spin_unlock(&xip_alloc_lock);
		}
	}
	return __xip_sparse_page;
}

43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58
/*
 * This is a file read routine for execute in place files, and uses
 * the mapping->a_ops->get_xip_page() function for the actual low-level
 * stuff.
 *
 * Note the struct file* is not used at all.  It may be NULL.
 */
static void
do_xip_mapping_read(struct address_space *mapping,
		    struct file_ra_state *_ra,
		    struct file *filp,
		    loff_t *ppos,
		    read_descriptor_t *desc,
		    read_actor_t actor)
{
	struct inode *inode = mapping->host;
J
Jan Kara 已提交
59 60
	pgoff_t index, end_index;
	unsigned long offset;
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
	loff_t isize;

	BUG_ON(!mapping->a_ops->get_xip_page);

	index = *ppos >> PAGE_CACHE_SHIFT;
	offset = *ppos & ~PAGE_CACHE_MASK;

	isize = i_size_read(inode);
	if (!isize)
		goto out;

	end_index = (isize - 1) >> PAGE_CACHE_SHIFT;
	for (;;) {
		struct page *page;
		unsigned long nr, ret;

		/* nr is the maximum number of bytes to copy from this page */
		nr = PAGE_CACHE_SIZE;
		if (index >= end_index) {
			if (index > end_index)
				goto out;
			nr = ((isize - 1) & ~PAGE_CACHE_MASK) + 1;
			if (nr <= offset) {
				goto out;
			}
		}
		nr = nr - offset;

		page = mapping->a_ops->get_xip_page(mapping,
			index*(PAGE_SIZE/512), 0);
		if (!page)
			goto no_xip_page;
		if (unlikely(IS_ERR(page))) {
			if (PTR_ERR(page) == -ENODATA) {
				/* sparse */
C
Carsten Otte 已提交
96
				page = ZERO_PAGE(0);
97 98 99 100
			} else {
				desc->error = PTR_ERR(page);
				goto out;
			}
C
Carsten Otte 已提交
101
		}
102 103 104 105 106 107 108 109 110

		/* If users can be writing to this page using arbitrary
		 * virtual addresses, take care about potential aliasing
		 * before reading the page on the kernel side.
		 */
		if (mapping_writably_mapped(mapping))
			flush_dcache_page(page);

		/*
C
Carsten Otte 已提交
111
		 * Ok, we have the page, so now we can copy it to user space...
112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140
		 *
		 * The actor routine returns how many bytes were actually used..
		 * NOTE! This may not be the same as how much of a user buffer
		 * we filled up (we may be padding etc), so we can only update
		 * "pos" here (the actor routine has to update the user buffer
		 * pointers and the remaining count).
		 */
		ret = actor(desc, page, offset, nr);
		offset += ret;
		index += offset >> PAGE_CACHE_SHIFT;
		offset &= ~PAGE_CACHE_MASK;

		if (ret == nr && desc->count)
			continue;
		goto out;

no_xip_page:
		/* Did not get the page. Report it */
		desc->error = -EIO;
		goto out;
	}

out:
	*ppos = ((loff_t) index << PAGE_CACHE_SHIFT) + offset;
	if (filp)
		file_accessed(filp);
}

ssize_t
141
xip_file_read(struct file *filp, char __user *buf, size_t len, loff_t *ppos)
142
{
143
	read_descriptor_t desc;
144

145 146
	if (!access_ok(VERIFY_WRITE, buf, len))
		return -EFAULT;
147

148 149 150 151
	desc.written = 0;
	desc.arg.buf = buf;
	desc.count = len;
	desc.error = 0;
152

153 154 155 156 157 158 159
	do_xip_mapping_read(filp->f_mapping, &filp->f_ra, filp,
			    ppos, &desc, file_read_actor);

	if (desc.written)
		return desc.written;
	else
		return desc.error;
160
}
161
EXPORT_SYMBOL_GPL(xip_file_read);
162 163 164 165 166 167

/*
 * __xip_unmap is invoked from xip_unmap and
 * xip_write
 *
 * This function walks all vmas of the address_space and unmaps the
168
 * __xip_sparse_page when found at pgoff.
169 170 171 172 173 174 175 176 177 178 179
 */
static void
__xip_unmap (struct address_space * mapping,
		     unsigned long pgoff)
{
	struct vm_area_struct *vma;
	struct mm_struct *mm;
	struct prio_tree_iter iter;
	unsigned long address;
	pte_t *pte;
	pte_t pteval;
H
Hugh Dickins 已提交
180
	spinlock_t *ptl;
H
Hugh Dickins 已提交
181
	struct page *page;
182

183 184 185 186
	page = __xip_sparse_page;
	if (!page)
		return;

187 188 189 190 191 192
	spin_lock(&mapping->i_mmap_lock);
	vma_prio_tree_foreach(vma, &iter, &mapping->i_mmap, pgoff, pgoff) {
		mm = vma->vm_mm;
		address = vma->vm_start +
			((pgoff - vma->vm_pgoff) << PAGE_SHIFT);
		BUG_ON(address < vma->vm_start || address >= vma->vm_end);
H
Hugh Dickins 已提交
193 194
		pte = page_check_address(page, mm, address, &ptl);
		if (pte) {
195
			/* Nuke the page table entry. */
196
			flush_cache_page(vma, address, pte_pfn(*pte));
197
			pteval = ptep_clear_flush(vma, address, pte);
N
Nick Piggin 已提交
198
			page_remove_rmap(page, vma);
N
Nick Piggin 已提交
199
			dec_mm_counter(mm, file_rss);
200
			BUG_ON(pte_dirty(pteval));
H
Hugh Dickins 已提交
201
			pte_unmap_unlock(pte, ptl);
N
Nick Piggin 已提交
202
			page_cache_release(page);
203 204 205 206 207 208
		}
	}
	spin_unlock(&mapping->i_mmap_lock);
}

/*
209
 * xip_fault() is invoked via the vma operations vector for a
210 211
 * mapped memory region to read in file data during a page fault.
 *
212
 * This function is derived from filemap_fault, but used for execute in place
213
 */
N
Nick Piggin 已提交
214
static int xip_file_fault(struct vm_area_struct *area, struct vm_fault *vmf)
215 216 217 218 219
{
	struct file *file = area->vm_file;
	struct address_space *mapping = file->f_mapping;
	struct inode *inode = mapping->host;
	struct page *page;
220
	pgoff_t size;
221

222
	/* XXX: are VM_FAULT_ codes OK? */
223 224

	size = (i_size_read(inode) + PAGE_CACHE_SIZE - 1) >> PAGE_CACHE_SHIFT;
N
Nick Piggin 已提交
225 226
	if (vmf->pgoff >= size)
		return VM_FAULT_SIGBUS;
227

228
	page = mapping->a_ops->get_xip_page(mapping,
N
Nick Piggin 已提交
229
					vmf->pgoff*(PAGE_SIZE/512), 0);
230
	if (!IS_ERR(page))
N
Nick Piggin 已提交
231
		goto out;
N
Nick Piggin 已提交
232 233
	if (PTR_ERR(page) != -ENODATA)
		return VM_FAULT_OOM;
234 235 236 237 238 239

	/* sparse block */
	if ((area->vm_flags & (VM_WRITE | VM_MAYWRITE)) &&
	    (area->vm_flags & (VM_SHARED| VM_MAYSHARE)) &&
	    (!(mapping->host->i_sb->s_flags & MS_RDONLY))) {
		/* maybe shared writable, allocate new block */
240
		page = mapping->a_ops->get_xip_page(mapping,
N
Nick Piggin 已提交
241 242 243
					vmf->pgoff*(PAGE_SIZE/512), 1);
		if (IS_ERR(page))
			return VM_FAULT_SIGBUS;
244
		/* unmap page at pgoff from all other vmas */
N
Nick Piggin 已提交
245
		__xip_unmap(mapping, vmf->pgoff);
246
	} else {
247 248
		/* not shared and writable, use xip_sparse_page() */
		page = xip_sparse_page();
N
Nick Piggin 已提交
249 250
		if (!page)
			return VM_FAULT_OOM;
251 252
	}

N
Nick Piggin 已提交
253 254
out:
	page_cache_get(page);
N
Nick Piggin 已提交
255
	vmf->page = page;
N
Nick Piggin 已提交
256
	return 0;
257 258 259
}

static struct vm_operations_struct xip_file_vm_ops = {
260
	.fault	= xip_file_fault,
261 262 263 264 265 266 267 268
};

int xip_file_mmap(struct file * file, struct vm_area_struct * vma)
{
	BUG_ON(!file->f_mapping->a_ops->get_xip_page);

	file_accessed(file);
	vma->vm_ops = &xip_file_vm_ops;
269
	vma->vm_flags |= VM_CAN_NONLINEAR;
270 271 272 273 274
	return 0;
}
EXPORT_SYMBOL_GPL(xip_file_mmap);

static ssize_t
275 276
__xip_file_write(struct file *filp, const char __user *buf,
		  size_t count, loff_t pos, loff_t *ppos)
277
{
278
	struct address_space * mapping = filp->f_mapping;
279
	const struct address_space_operations *a_ops = mapping->a_ops;
280 281 282 283 284 285 286 287 288 289 290 291
	struct inode 	*inode = mapping->host;
	long		status = 0;
	struct page	*page;
	size_t		bytes;
	ssize_t		written = 0;

	BUG_ON(!mapping->a_ops->get_xip_page);

	do {
		unsigned long index;
		unsigned long offset;
		size_t copied;
N
Nick Piggin 已提交
292
		char *kaddr;
293 294 295 296 297 298 299 300

		offset = (pos & (PAGE_CACHE_SIZE -1)); /* Within page */
		index = pos >> PAGE_CACHE_SHIFT;
		bytes = PAGE_CACHE_SIZE - offset;
		if (bytes > count)
			bytes = count;

		page = a_ops->get_xip_page(mapping,
301
					   index*(PAGE_SIZE/512), 0);
302 303 304
		if (IS_ERR(page) && (PTR_ERR(page) == -ENODATA)) {
			/* we allocate a new page unmap it */
			page = a_ops->get_xip_page(mapping,
305
						   index*(PAGE_SIZE/512), 1);
306
			if (!IS_ERR(page))
307 308
				/* unmap page at pgoff from all other vmas */
				__xip_unmap(mapping, index);
309 310 311 312 313 314 315
		}

		if (IS_ERR(page)) {
			status = PTR_ERR(page);
			break;
		}

N
Nick Piggin 已提交
316 317 318
		fault_in_pages_readable(buf, bytes);
		kaddr = kmap_atomic(page, KM_USER0);
		copied = bytes -
N
Nick Piggin 已提交
319
			__copy_from_user_inatomic_nocache(kaddr + offset, buf, bytes);
N
Nick Piggin 已提交
320
		kunmap_atomic(kaddr, KM_USER0);
321
		flush_dcache_page(page);
N
Nick Piggin 已提交
322

323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341
		if (likely(copied > 0)) {
			status = copied;

			if (status >= 0) {
				written += status;
				count -= status;
				pos += status;
				buf += status;
			}
		}
		if (unlikely(copied != bytes))
			if (status >= 0)
				status = -EFAULT;
		if (status < 0)
			break;
	} while (count);
	*ppos = pos;
	/*
	 * No need to use i_size_read() here, the i_size
342
	 * cannot change under us because we hold i_mutex.
343 344 345 346 347 348 349 350 351
	 */
	if (pos > inode->i_size) {
		i_size_write(inode, pos);
		mark_inode_dirty(inode);
	}

	return written ? written : status;
}

352 353 354
ssize_t
xip_file_write(struct file *filp, const char __user *buf, size_t len,
	       loff_t *ppos)
355
{
356 357 358 359 360
	struct address_space *mapping = filp->f_mapping;
	struct inode *inode = mapping->host;
	size_t count;
	loff_t pos;
	ssize_t ret;
361

362
	mutex_lock(&inode->i_mutex);
363

364 365 366
	if (!access_ok(VERIFY_READ, buf, len)) {
		ret=-EFAULT;
		goto out_up;
367 368 369
	}

	pos = *ppos;
370
	count = len;
371 372 373

	vfs_check_frozen(inode->i_sb, SB_FREEZE_WRITE);

374 375
	/* We can write back this queue in page reclaim */
	current->backing_dev_info = mapping->backing_dev_info;
376

377 378 379
	ret = generic_write_checks(filp, &pos, &count, S_ISBLK(inode->i_mode));
	if (ret)
		goto out_backing;
380
	if (count == 0)
381
		goto out_backing;
382

383
	ret = remove_suid(filp->f_path.dentry);
384 385
	if (ret)
		goto out_backing;
386

387
	file_update_time(filp);
388

389
	ret = __xip_file_write (filp, buf, count, pos, ppos);
390

391 392 393
 out_backing:
	current->backing_dev_info = NULL;
 out_up:
394
	mutex_unlock(&inode->i_mutex);
395 396
	return ret;
}
397
EXPORT_SYMBOL_GPL(xip_file_write);
398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426

/*
 * truncate a page used for execute in place
 * functionality is analog to block_truncate_page but does use get_xip_page
 * to get the page instead of page cache
 */
int
xip_truncate_page(struct address_space *mapping, loff_t from)
{
	pgoff_t index = from >> PAGE_CACHE_SHIFT;
	unsigned offset = from & (PAGE_CACHE_SIZE-1);
	unsigned blocksize;
	unsigned length;
	struct page *page;

	BUG_ON(!mapping->a_ops->get_xip_page);

	blocksize = 1 << mapping->host->i_blkbits;
	length = offset & (blocksize - 1);

	/* Block boundary? Nothing to do */
	if (!length)
		return 0;

	length = blocksize - length;

	page = mapping->a_ops->get_xip_page(mapping,
					    index*(PAGE_SIZE/512), 0);
	if (!page)
427
		return -ENOMEM;
428
	if (unlikely(IS_ERR(page))) {
429
		if (PTR_ERR(page) == -ENODATA)
430 431
			/* Hole? No need to truncate */
			return 0;
432 433
		else
			return PTR_ERR(page);
C
Carsten Otte 已提交
434
	}
435
	zero_user(page, offset, length);
436
	return 0;
437 438
}
EXPORT_SYMBOL_GPL(xip_truncate_page);