Add hooks for disabling direct reclaim
The port of XFS to Linux introduced a thread-specific PF_FSTRANS bit that is used to mark contexts which are processing transactions. When set, allocations in this context can dip into kernel memory reserves to avoid deadlocks during writeback. Linux 3.9 provided the additional PF_MEMALLOC_NOIO for disabling __GFP_IO in page allocations, which XFS began using in 3.15. This patch implements hooks for marking transactions via PF_FSTRANS. When an allocation is performed in the context of PF_FSTRANS, any KM_SLEEP allocation is transparently converted to a GFP_NOIO allocation. Additionally, when using a Linux 3.9 or newer kernel, it will set PF_MEMALLOC_NOIO to prevent direct reclaim from entering pageout() on on any KM_PUSHPAGE or KM_NOSLEEP allocation. This effectively allows the spl_vmalloc() helper function to be used safely in a thread which is responsible for IO. Signed-off-by: Brian Behlendorf <behlendorf1@llnl.gov>
This commit is contained in:
parent
c3eabc75b1
commit
c2fa09454e
|
@ -25,6 +25,7 @@
|
||||||
#ifndef _SPL_KMEM_H
|
#ifndef _SPL_KMEM_H
|
||||||
#define _SPL_KMEM_H
|
#define _SPL_KMEM_H
|
||||||
|
|
||||||
|
#include <sys/debug.h>
|
||||||
#include <linux/slab.h>
|
#include <linux/slab.h>
|
||||||
#include <linux/sched.h>
|
#include <linux/sched.h>
|
||||||
|
|
||||||
|
@ -72,6 +73,39 @@ kmem_flags_convert(int flags)
|
||||||
return (lflags);
|
return (lflags);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
struct task_struct *fstrans_thread;
|
||||||
|
unsigned int saved_flags;
|
||||||
|
} fstrans_cookie_t;
|
||||||
|
|
||||||
|
static inline fstrans_cookie_t
|
||||||
|
spl_fstrans_mark(void)
|
||||||
|
{
|
||||||
|
fstrans_cookie_t cookie;
|
||||||
|
|
||||||
|
cookie.fstrans_thread = current;
|
||||||
|
cookie.saved_flags = current->flags & PF_FSTRANS;
|
||||||
|
current->flags |= PF_FSTRANS;
|
||||||
|
|
||||||
|
return (cookie);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void
|
||||||
|
spl_fstrans_unmark(fstrans_cookie_t cookie)
|
||||||
|
{
|
||||||
|
ASSERT3P(cookie.fstrans_thread, ==, current);
|
||||||
|
ASSERT(current->flags & PF_FSTRANS);
|
||||||
|
|
||||||
|
current->flags &= ~(PF_FSTRANS);
|
||||||
|
current->flags |= cookie.saved_flags;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline int
|
||||||
|
spl_fstrans_check(void)
|
||||||
|
{
|
||||||
|
return (current->flags & PF_FSTRANS);
|
||||||
|
}
|
||||||
|
|
||||||
#ifdef HAVE_ATOMIC64_T
|
#ifdef HAVE_ATOMIC64_T
|
||||||
#define kmem_alloc_used_add(size) atomic64_add(size, &kmem_alloc_used)
|
#define kmem_alloc_used_add(size) atomic64_add(size, &kmem_alloc_used)
|
||||||
#define kmem_alloc_used_sub(size) atomic64_sub(size, &kmem_alloc_used)
|
#define kmem_alloc_used_sub(size) atomic64_sub(size, &kmem_alloc_used)
|
||||||
|
|
|
@ -36,6 +36,7 @@ extern vmem_t *zio_alloc_arena;
|
||||||
extern vmem_t *zio_arena;
|
extern vmem_t *zio_arena;
|
||||||
|
|
||||||
extern size_t vmem_size(vmem_t *vmp, int typemask);
|
extern size_t vmem_size(vmem_t *vmp, int typemask);
|
||||||
|
extern void *spl_vmalloc(unsigned long size, gfp_t lflags, pgprot_t prot);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Memory allocation interfaces
|
* Memory allocation interfaces
|
||||||
|
|
|
@ -153,7 +153,7 @@ kv_alloc(spl_kmem_cache_t *skc, int size, int flags)
|
||||||
if (skc->skc_flags & KMC_KMEM)
|
if (skc->skc_flags & KMC_KMEM)
|
||||||
ptr = (void *)__get_free_pages(lflags, get_order(size));
|
ptr = (void *)__get_free_pages(lflags, get_order(size));
|
||||||
else
|
else
|
||||||
ptr = __vmalloc(size, lflags | __GFP_HIGHMEM, PAGE_KERNEL);
|
ptr = spl_vmalloc(size, lflags | __GFP_HIGHMEM, PAGE_KERNEL);
|
||||||
|
|
||||||
/* Resulting allocated memory will be page aligned */
|
/* Resulting allocated memory will be page aligned */
|
||||||
ASSERT(IS_P2ALIGNED(ptr, PAGE_SIZE));
|
ASSERT(IS_P2ALIGNED(ptr, PAGE_SIZE));
|
||||||
|
@ -1098,7 +1098,9 @@ spl_cache_grow_work(void *data)
|
||||||
sks = spl_slab_alloc(skc, ska->ska_flags);
|
sks = spl_slab_alloc(skc, ska->ska_flags);
|
||||||
memalloc_noio_restore(noio_flag);
|
memalloc_noio_restore(noio_flag);
|
||||||
#else
|
#else
|
||||||
|
fstrans_cookie_t cookie = spl_fstrans_mark();
|
||||||
sks = spl_slab_alloc(skc, ska->ska_flags);
|
sks = spl_slab_alloc(skc, ska->ska_flags);
|
||||||
|
spl_fstrans_unmark(cookie);
|
||||||
#endif
|
#endif
|
||||||
spin_lock(&skc->skc_lock);
|
spin_lock(&skc->skc_lock);
|
||||||
if (sks) {
|
if (sks) {
|
||||||
|
|
|
@ -184,7 +184,7 @@ spl_kmem_alloc_impl(size_t size, int flags, int node)
|
||||||
*/
|
*/
|
||||||
if (unlikely(size > spl_kmem_alloc_max)) {
|
if (unlikely(size > spl_kmem_alloc_max)) {
|
||||||
if (flags & KM_VMEM) {
|
if (flags & KM_VMEM) {
|
||||||
ptr = __vmalloc(size, lflags, PAGE_KERNEL);
|
ptr = spl_vmalloc(size, lflags, PAGE_KERNEL);
|
||||||
} else {
|
} else {
|
||||||
return (NULL);
|
return (NULL);
|
||||||
}
|
}
|
||||||
|
|
|
@ -97,6 +97,31 @@ spl_vmem_free(const void *buf, size_t size)
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(spl_vmem_free);
|
EXPORT_SYMBOL(spl_vmem_free);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Public vmalloc() interface designed to be safe to be called during I/O.
|
||||||
|
*/
|
||||||
|
void *
|
||||||
|
spl_vmalloc(unsigned long size, gfp_t lflags, pgprot_t prot)
|
||||||
|
{
|
||||||
|
#if defined(PF_MEMALLOC_NOIO)
|
||||||
|
void *ptr;
|
||||||
|
unsigned noio_flag = 0;
|
||||||
|
|
||||||
|
if (spl_fstrans_check())
|
||||||
|
noio_flag = memalloc_noio_save();
|
||||||
|
|
||||||
|
ptr = __vmalloc(size, lflags, prot);
|
||||||
|
|
||||||
|
if (spl_fstrans_check())
|
||||||
|
memalloc_noio_restore(noio_flag);
|
||||||
|
|
||||||
|
return (ptr);
|
||||||
|
#else
|
||||||
|
return (__vmalloc(size, lflags, prot));
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(spl_vmalloc);
|
||||||
|
|
||||||
int
|
int
|
||||||
spl_vmem_init(void)
|
spl_vmem_init(void)
|
||||||
{
|
{
|
||||||
|
|
Loading…
Reference in New Issue