kmem.c 3.1 KB
Newer Older
L
Linus Torvalds 已提交
1
/*
2 3
 * Copyright (c) 2000-2005 Silicon Graphics, Inc.
 * All Rights Reserved.
L
Linus Torvalds 已提交
4
 *
5 6
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License as
L
Linus Torvalds 已提交
7 8
 * published by the Free Software Foundation.
 *
9 10 11 12
 * This program is distributed in the hope that it would be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
L
Linus Torvalds 已提交
13
 *
14 15 16
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write the Free Software Foundation,
 * Inc.,  51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
L
Linus Torvalds 已提交
17 18 19
 */
#include <linux/mm.h>
#include <linux/highmem.h>
20
#include <linux/slab.h>
L
Linus Torvalds 已提交
21 22
#include <linux/swap.h>
#include <linux/blkdev.h>
23
#include <linux/backing-dev.h>
L
Linus Torvalds 已提交
24
#include "kmem.h"
25
#include "xfs_message.h"
L
Linus Torvalds 已提交
26

27 28 29 30 31 32 33 34 35
/*
 * Greedy allocation.  May fail and may return vmalloced memory.
 */
void *
kmem_zalloc_greedy(size_t *size, size_t minsize, size_t maxsize)
{
	void		*ptr;
	size_t		kmsize = maxsize;

36
	while (!(ptr = vzalloc(kmsize))) {
37 38 39 40 41 42 43
		if ((kmsize >>= 1) <= minsize)
			kmsize = minsize;
	}
	if (ptr)
		*size = kmsize;
	return ptr;
}
L
Linus Torvalds 已提交
44 45

void *
46
kmem_alloc(size_t size, xfs_km_flags_t flags)
L
Linus Torvalds 已提交
47
{
A
Al Viro 已提交
48 49 50
	int	retries = 0;
	gfp_t	lflags = kmem_flags_convert(flags);
	void	*ptr;
L
Linus Torvalds 已提交
51 52

	do {
53
		ptr = kmalloc(size, lflags);
L
Linus Torvalds 已提交
54 55 56
		if (ptr || (flags & (KM_MAYFAIL|KM_NOSLEEP)))
			return ptr;
		if (!(++retries % 100))
57 58
			xfs_err(NULL,
		"possible memory allocation deadlock in %s (mode:0x%x)",
59
					__func__, lflags);
60
		congestion_wait(BLK_RW_ASYNC, HZ/50);
L
Linus Torvalds 已提交
61 62 63
	} while (1);
}

64 65 66
void *
kmem_zalloc_large(size_t size, xfs_km_flags_t flags)
{
67
	unsigned noio_flag = 0;
68
	void	*ptr;
69
	gfp_t	lflags;
70 71 72 73

	ptr = kmem_zalloc(size, flags | KM_MAYFAIL);
	if (ptr)
		return ptr;
74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91

	/*
	 * __vmalloc() will allocate data pages and auxillary structures (e.g.
	 * pagetables) with GFP_KERNEL, yet we may be under GFP_NOFS context
	 * here. Hence we need to tell memory reclaim that we are in such a
	 * context via PF_MEMALLOC_NOIO to prevent memory reclaim re-entering
	 * the filesystem here and potentially deadlocking.
	 */
	if ((current->flags & PF_FSTRANS) || (flags & KM_NOFS))
		noio_flag = memalloc_noio_save();

	lflags = kmem_flags_convert(flags);
	ptr = __vmalloc(size, lflags | __GFP_HIGHMEM | __GFP_ZERO, PAGE_KERNEL);

	if ((current->flags & PF_FSTRANS) || (flags & KM_NOFS))
		memalloc_noio_restore(noio_flag);

	return ptr;
92 93
}

L
Linus Torvalds 已提交
94
void *
95
kmem_realloc(const void *ptr, size_t newsize, size_t oldsize,
96
	     xfs_km_flags_t flags)
L
Linus Torvalds 已提交
97 98 99 100 101 102 103 104
{
	void	*new;

	new = kmem_alloc(newsize, flags);
	if (ptr) {
		if (new)
			memcpy(new, ptr,
				((oldsize < newsize) ? oldsize : newsize));
105
		kmem_free(ptr);
L
Linus Torvalds 已提交
106 107 108 109 110
	}
	return new;
}

void *
111
kmem_zone_alloc(kmem_zone_t *zone, xfs_km_flags_t flags)
L
Linus Torvalds 已提交
112
{
A
Al Viro 已提交
113 114 115
	int	retries = 0;
	gfp_t	lflags = kmem_flags_convert(flags);
	void	*ptr;
L
Linus Torvalds 已提交
116 117 118 119 120 121

	do {
		ptr = kmem_cache_alloc(zone, lflags);
		if (ptr || (flags & (KM_MAYFAIL|KM_NOSLEEP)))
			return ptr;
		if (!(++retries % 100))
122 123
			xfs_err(NULL,
		"possible memory allocation deadlock in %s (mode:0x%x)",
124
					__func__, lflags);
125
		congestion_wait(BLK_RW_ASYNC, HZ/50);
L
Linus Torvalds 已提交
126 127
	} while (1);
}