2
0
mirror of https://github.com/edk2-porting/linux-next.git synced 2025-01-18 10:34:24 +08:00

Merge branch 'akpm' (patches from Andrew)

Merge updates from Andrew Morton:

 - a few misc things

 - a few Y2038 fixes

 - ntfs fixes

 - arch/sh tweaks

 - ocfs2 updates

 - most of MM

* emailed patches from Andrew Morton <akpm@linux-foundation.org>: (111 commits)
  mm/hmm.c: remove unused variables align_start and align_end
  fs/userfaultfd.c: remove redundant pointer uwq
  mm, vmacache: hash addresses based on pmd
  mm/list_lru: introduce list_lru_shrink_walk_irq()
  mm/list_lru.c: pass struct list_lru_node* as an argument to __list_lru_walk_one()
  mm/list_lru.c: move locking from __list_lru_walk_one() to its caller
  mm/list_lru.c: use list_lru_walk_one() in list_lru_walk_node()
  mm, swap: make CONFIG_THP_SWAP depend on CONFIG_SWAP
  mm/sparse: delete old sparse_init and enable new one
  mm/sparse: add new sparse_init_nid() and sparse_init()
  mm/sparse: move buffer init/fini to the common place
  mm/sparse: use the new sparse buffer functions in non-vmemmap
  mm/sparse: abstract sparse buffer allocations
  mm/hugetlb.c: don't zero 1GiB bootmem pages
  mm, page_alloc: double zone's batchsize
  mm/oom_kill.c: document oom_lock
  mm/hugetlb: remove gigantic page support for HIGHMEM
  mm, oom: remove sleep from under oom_lock
  kernel/dma: remove unsupported gfp_mask parameter from dma_alloc_from_contiguous()
  mm/cma: remove unsupported gfp_mask parameter from cma_alloc()
  ...
This commit is contained in:
Linus Torvalds 2018-08-17 16:49:31 -07:00
commit 6ada4e2826
152 changed files with 2134 additions and 1177 deletions

View File

@ -2571,6 +2571,11 @@ S: Helstorfer Str. 7
S: D-30625 Hannover S: D-30625 Hannover
S: Germany S: Germany
N: Ron Minnich
E: rminnich@sandia.gov
E: rminnich@gmail.com
D: 9p filesystem development
N: Corey Minyard N: Corey Minyard
E: minyard@wf-rch.cirr.com E: minyard@wf-rch.cirr.com
E: minyard@mvista.com E: minyard@mvista.com

View File

@ -65,6 +65,11 @@ workload one should:
are not reclaimable, he or she can filter them out using are not reclaimable, he or she can filter them out using
``/proc/kpageflags``. ``/proc/kpageflags``.
The page-types tool in the tools/vm directory can be used to assist in this.
If the tool is run initially with the appropriate option, it will mark all the
queried pages as idle. Subsequent runs of the tool can then show which pages have
their idle flag cleared in the interim.
See :ref:`Documentation/admin-guide/mm/pagemap.rst <pagemap>` for more See :ref:`Documentation/admin-guide/mm/pagemap.rst <pagemap>` for more
information about ``/proc/pid/pagemap``, ``/proc/kpageflags``, and information about ``/proc/pid/pagemap``, ``/proc/kpageflags``, and
``/proc/kpagecgroup``. ``/proc/kpagecgroup``.

View File

@ -44,6 +44,9 @@ There are four components to pagemap:
* ``/proc/kpagecount``. This file contains a 64-bit count of the number of * ``/proc/kpagecount``. This file contains a 64-bit count of the number of
times each page is mapped, indexed by PFN. times each page is mapped, indexed by PFN.
The page-types tool in the tools/vm directory can be used to query the
number of times a page is mapped.
* ``/proc/kpageflags``. This file contains a 64-bit set of flags for each * ``/proc/kpageflags``. This file contains a 64-bit set of flags for each
page, indexed by PFN. page, indexed by PFN.

View File

@ -66,23 +66,39 @@ kernel 3.10. Current versions require the following update
The iterator interface The iterator interface
Modules implementing a virtual file with seq_file must implement a simple Modules implementing a virtual file with seq_file must implement an
iterator object that allows stepping through the data of interest. iterator object that allows stepping through the data of interest
Iterators must be able to move to a specific position - like the file they during a "session" (roughly one read() system call). If the iterator
implement - but the interpretation of that position is up to the iterator is able to move to a specific position - like the file they implement,
itself. A seq_file implementation that is formatting firewall rules, for though with freedom to map the position number to a sequence location
example, could interpret position N as the Nth rule in the chain. in whatever way is convenient - the iterator need only exist
Positioning can thus be done in whatever way makes the most sense for the transiently during a session. If the iterator cannot easily find a
generator of the data, which need not be aware of how a position translates numerical position but works well with a first/next interface, the
to an offset in the virtual file. The one obvious exception is that a iterator can be stored in the private data area and continue from one
position of zero should indicate the beginning of the file. session to the next.
A seq_file implementation that is formatting firewall rules from a
table, for example, could provide a simple iterator that interprets
position N as the Nth rule in the chain. A seq_file implementation
that presents the content of a, potentially volatile, linked list
might record a pointer into that list, providing that can be done
without risk of the current location being removed.
Positioning can thus be done in whatever way makes the most sense for
the generator of the data, which need not be aware of how a position
translates to an offset in the virtual file. The one obvious exception
is that a position of zero should indicate the beginning of the file.
The /proc/sequence iterator just uses the count of the next number it The /proc/sequence iterator just uses the count of the next number it
will output as its position. will output as its position.
Four functions must be implemented to make the iterator work. The first, Four functions must be implemented to make the iterator work. The
called start() takes a position as an argument and returns an iterator first, called start(), starts a session and takes a position as an
which will start reading at that position. For our simple sequence example, argument, returning an iterator which will start reading at that
position. The pos passed to start() will always be either zero, or
the most recent pos used in the previous session.
For our simple sequence example,
the start() function looks like: the start() function looks like:
static void *ct_seq_start(struct seq_file *s, loff_t *pos) static void *ct_seq_start(struct seq_file *s, loff_t *pos)
@ -101,11 +117,12 @@ implementations; in most cases the start() function should check for a
"past end of file" condition and return NULL if need be. "past end of file" condition and return NULL if need be.
For more complicated applications, the private field of the seq_file For more complicated applications, the private field of the seq_file
structure can be used. There is also a special value which can be returned structure can be used to hold state from session to session. There is
by the start() function called SEQ_START_TOKEN; it can be used if you wish also a special value which can be returned by the start() function
to instruct your show() function (described below) to print a header at the called SEQ_START_TOKEN; it can be used if you wish to instruct your
top of the output. SEQ_START_TOKEN should only be used if the offset is show() function (described below) to print a header at the top of the
zero, however. output. SEQ_START_TOKEN should only be used if the offset is zero,
however.
The next function to implement is called, amazingly, next(); its job is to The next function to implement is called, amazingly, next(); its job is to
move the iterator forward to the next position in the sequence. The move the iterator forward to the next position in the sequence. The
@ -121,9 +138,13 @@ complete. Here's the example version:
return spos; return spos;
} }
The stop() function is called when iteration is complete; its job, of The stop() function closes a session; its job, of course, is to clean
course, is to clean up. If dynamic memory is allocated for the iterator, up. If dynamic memory is allocated for the iterator, stop() is the
stop() is the place to free it. place to free it; if a lock was taken by start(), stop() must release
that lock. The value that *pos was set to by the last next() call
before stop() is remembered, and used for the first start() call of
the next session unless lseek() has been called on the file; in that
case next start() will be asked to start at position zero.
static void ct_seq_stop(struct seq_file *s, void *v) static void ct_seq_stop(struct seq_file *s, void *v)
{ {

View File

@ -199,12 +199,13 @@ F: drivers/net/ethernet/8390/
9P FILE SYSTEM 9P FILE SYSTEM
M: Eric Van Hensbergen <ericvh@gmail.com> M: Eric Van Hensbergen <ericvh@gmail.com>
M: Ron Minnich <rminnich@sandia.gov>
M: Latchesar Ionkov <lucho@ionkov.net> M: Latchesar Ionkov <lucho@ionkov.net>
M: Dominique Martinet <asmadeus@codewreck.org>
L: v9fs-developer@lists.sourceforge.net L: v9fs-developer@lists.sourceforge.net
W: http://swik.net/v9fs W: http://swik.net/v9fs
Q: http://patchwork.kernel.org/project/v9fs-devel/list/ Q: http://patchwork.kernel.org/project/v9fs-devel/list/
T: git git://git.kernel.org/pub/scm/linux/kernel/git/ericvh/v9fs.git T: git git://git.kernel.org/pub/scm/linux/kernel/git/ericvh/v9fs.git
T: git git://github.com/martinetd/linux.git
S: Maintained S: Maintained
F: Documentation/filesystems/9p.txt F: Documentation/filesystems/9p.txt
F: fs/9p/ F: fs/9p/

View File

@ -87,7 +87,8 @@ do_page_fault(unsigned long address, unsigned long mmcsr,
struct vm_area_struct * vma; struct vm_area_struct * vma;
struct mm_struct *mm = current->mm; struct mm_struct *mm = current->mm;
const struct exception_table_entry *fixup; const struct exception_table_entry *fixup;
int fault, si_code = SEGV_MAPERR; int si_code = SEGV_MAPERR;
vm_fault_t fault;
unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE; unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE;
/* As of EV6, a load into $31/$f31 is a prefetch, and never faults /* As of EV6, a load into $31/$f31 is a prefetch, and never faults

View File

@ -15,6 +15,7 @@
#include <linux/uaccess.h> #include <linux/uaccess.h>
#include <linux/kdebug.h> #include <linux/kdebug.h>
#include <linux/perf_event.h> #include <linux/perf_event.h>
#include <linux/mm_types.h>
#include <asm/pgalloc.h> #include <asm/pgalloc.h>
#include <asm/mmu.h> #include <asm/mmu.h>
@ -66,7 +67,8 @@ void do_page_fault(unsigned long address, struct pt_regs *regs)
struct task_struct *tsk = current; struct task_struct *tsk = current;
struct mm_struct *mm = tsk->mm; struct mm_struct *mm = tsk->mm;
siginfo_t info; siginfo_t info;
int fault, ret; int ret;
vm_fault_t fault;
int write = regs->ecr_cause & ECR_C_PROTV_STORE; /* ST/EX */ int write = regs->ecr_cause & ECR_C_PROTV_STORE; /* ST/EX */
unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE; unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE;

View File

@ -594,7 +594,7 @@ static void *__alloc_from_contiguous(struct device *dev, size_t size,
struct page *page; struct page *page;
void *ptr = NULL; void *ptr = NULL;
page = dma_alloc_from_contiguous(dev, count, order, gfp); page = dma_alloc_from_contiguous(dev, count, order, gfp & __GFP_NOWARN);
if (!page) if (!page)
return NULL; return NULL;
@ -1299,7 +1299,8 @@ static struct page **__iommu_alloc_buffer(struct device *dev, size_t size,
unsigned long order = get_order(size); unsigned long order = get_order(size);
struct page *page; struct page *page;
page = dma_alloc_from_contiguous(dev, count, order, gfp); page = dma_alloc_from_contiguous(dev, count, order,
gfp & __GFP_NOWARN);
if (!page) if (!page)
goto error; goto error;

View File

@ -224,12 +224,12 @@ static inline bool access_error(unsigned int fsr, struct vm_area_struct *vma)
return vma->vm_flags & mask ? false : true; return vma->vm_flags & mask ? false : true;
} }
static int __kprobes static vm_fault_t __kprobes
__do_page_fault(struct mm_struct *mm, unsigned long addr, unsigned int fsr, __do_page_fault(struct mm_struct *mm, unsigned long addr, unsigned int fsr,
unsigned int flags, struct task_struct *tsk) unsigned int flags, struct task_struct *tsk)
{ {
struct vm_area_struct *vma; struct vm_area_struct *vma;
int fault; vm_fault_t fault;
vma = find_vma(mm, addr); vma = find_vma(mm, addr);
fault = VM_FAULT_BADMAP; fault = VM_FAULT_BADMAP;
@ -264,7 +264,8 @@ do_page_fault(unsigned long addr, unsigned int fsr, struct pt_regs *regs)
{ {
struct task_struct *tsk; struct task_struct *tsk;
struct mm_struct *mm; struct mm_struct *mm;
int fault, sig, code; int sig, code;
vm_fault_t fault;
unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE; unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE;
if (notify_page_fault(regs, fsr)) if (notify_page_fault(regs, fsr))

View File

@ -355,7 +355,7 @@ static int __init atomic_pool_init(void)
if (dev_get_cma_area(NULL)) if (dev_get_cma_area(NULL))
page = dma_alloc_from_contiguous(NULL, nr_pages, page = dma_alloc_from_contiguous(NULL, nr_pages,
pool_size_order, GFP_KERNEL); pool_size_order, false);
else else
page = alloc_pages(GFP_DMA32, pool_size_order); page = alloc_pages(GFP_DMA32, pool_size_order);
@ -573,7 +573,7 @@ static void *__iommu_alloc_attrs(struct device *dev, size_t size,
struct page *page; struct page *page;
page = dma_alloc_from_contiguous(dev, size >> PAGE_SHIFT, page = dma_alloc_from_contiguous(dev, size >> PAGE_SHIFT,
get_order(size), gfp); get_order(size), gfp & __GFP_NOWARN);
if (!page) if (!page)
return NULL; return NULL;

View File

@ -379,12 +379,12 @@ static void do_bad_area(unsigned long addr, unsigned int esr, struct pt_regs *re
#define VM_FAULT_BADMAP 0x010000 #define VM_FAULT_BADMAP 0x010000
#define VM_FAULT_BADACCESS 0x020000 #define VM_FAULT_BADACCESS 0x020000
static int __do_page_fault(struct mm_struct *mm, unsigned long addr, static vm_fault_t __do_page_fault(struct mm_struct *mm, unsigned long addr,
unsigned int mm_flags, unsigned long vm_flags, unsigned int mm_flags, unsigned long vm_flags,
struct task_struct *tsk) struct task_struct *tsk)
{ {
struct vm_area_struct *vma; struct vm_area_struct *vma;
int fault; vm_fault_t fault;
vma = find_vma(mm, addr); vma = find_vma(mm, addr);
fault = VM_FAULT_BADMAP; fault = VM_FAULT_BADMAP;
@ -427,7 +427,7 @@ static int __kprobes do_page_fault(unsigned long addr, unsigned int esr,
struct task_struct *tsk; struct task_struct *tsk;
struct mm_struct *mm; struct mm_struct *mm;
struct siginfo si; struct siginfo si;
int fault, major = 0; vm_fault_t fault, major = 0;
unsigned long vm_flags = VM_READ | VM_WRITE; unsigned long vm_flags = VM_READ | VM_WRITE;
unsigned int mm_flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE; unsigned int mm_flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE;

View File

@ -52,7 +52,7 @@ void do_page_fault(unsigned long address, long cause, struct pt_regs *regs)
struct mm_struct *mm = current->mm; struct mm_struct *mm = current->mm;
int si_signo; int si_signo;
int si_code = SEGV_MAPERR; int si_code = SEGV_MAPERR;
int fault; vm_fault_t fault;
const struct exception_table_entry *fixup; const struct exception_table_entry *fixup;
unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE; unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE;

View File

@ -86,7 +86,7 @@ ia64_do_page_fault (unsigned long address, unsigned long isr, struct pt_regs *re
struct vm_area_struct *vma, *prev_vma; struct vm_area_struct *vma, *prev_vma;
struct mm_struct *mm = current->mm; struct mm_struct *mm = current->mm;
unsigned long mask; unsigned long mask;
int fault; vm_fault_t fault;
unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE; unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE;
mask = ((((isr >> IA64_ISR_X_BIT) & 1UL) << VM_EXEC_BIT) mask = ((((isr >> IA64_ISR_X_BIT) & 1UL) << VM_EXEC_BIT)

View File

@ -70,7 +70,7 @@ int do_page_fault(struct pt_regs *regs, unsigned long address,
{ {
struct mm_struct *mm = current->mm; struct mm_struct *mm = current->mm;
struct vm_area_struct * vma; struct vm_area_struct * vma;
int fault; vm_fault_t fault;
unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE; unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE;
pr_debug("do page fault:\nregs->sr=%#x, regs->pc=%#lx, address=%#lx, %ld, %p\n", pr_debug("do page fault:\nregs->sr=%#x, regs->pc=%#lx, address=%#lx, %ld, %p\n",
@ -136,7 +136,7 @@ good_area:
*/ */
fault = handle_mm_fault(vma, address, flags); fault = handle_mm_fault(vma, address, flags);
pr_debug("handle_mm_fault returns %d\n", fault); pr_debug("handle_mm_fault returns %x\n", fault);
if ((fault & VM_FAULT_RETRY) && fatal_signal_pending(current)) if ((fault & VM_FAULT_RETRY) && fatal_signal_pending(current))
return 0; return 0;

View File

@ -90,7 +90,7 @@ void do_page_fault(struct pt_regs *regs, unsigned long address,
struct mm_struct *mm = current->mm; struct mm_struct *mm = current->mm;
int code = SEGV_MAPERR; int code = SEGV_MAPERR;
int is_write = error_code & ESR_S; int is_write = error_code & ESR_S;
int fault; vm_fault_t fault;
unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE; unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE;
regs->ear = address; regs->ear = address;

View File

@ -43,7 +43,7 @@ static void __kprobes __do_page_fault(struct pt_regs *regs, unsigned long write,
struct mm_struct *mm = tsk->mm; struct mm_struct *mm = tsk->mm;
const int field = sizeof(unsigned long) * 2; const int field = sizeof(unsigned long) * 2;
int si_code; int si_code;
int fault; vm_fault_t fault;
unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE; unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE;
static DEFINE_RATELIMIT_STATE(ratelimit_state, 5 * HZ, 10); static DEFINE_RATELIMIT_STATE(ratelimit_state, 5 * HZ, 10);

View File

@ -73,7 +73,7 @@ void do_page_fault(unsigned long entry, unsigned long addr,
struct mm_struct *mm; struct mm_struct *mm;
struct vm_area_struct *vma; struct vm_area_struct *vma;
int si_code; int si_code;
int fault; vm_fault_t fault;
unsigned int mask = VM_READ | VM_WRITE | VM_EXEC; unsigned int mask = VM_READ | VM_WRITE | VM_EXEC;
unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE; unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE;

View File

@ -47,7 +47,7 @@ asmlinkage void do_page_fault(struct pt_regs *regs, unsigned long cause,
struct task_struct *tsk = current; struct task_struct *tsk = current;
struct mm_struct *mm = tsk->mm; struct mm_struct *mm = tsk->mm;
int code = SEGV_MAPERR; int code = SEGV_MAPERR;
int fault; vm_fault_t fault;
unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE; unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE;
cause >>= 2; cause >>= 2;

View File

@ -53,7 +53,7 @@ asmlinkage void do_page_fault(struct pt_regs *regs, unsigned long address,
struct mm_struct *mm; struct mm_struct *mm;
struct vm_area_struct *vma; struct vm_area_struct *vma;
int si_code; int si_code;
int fault; vm_fault_t fault;
unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE; unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE;
tsk = current; tsk = current;

View File

@ -262,7 +262,7 @@ void do_page_fault(struct pt_regs *regs, unsigned long code,
struct task_struct *tsk; struct task_struct *tsk;
struct mm_struct *mm; struct mm_struct *mm;
unsigned long acc_type; unsigned long acc_type;
int fault = 0; vm_fault_t fault = 0;
unsigned int flags; unsigned int flags;
if (faulthandler_disabled()) if (faulthandler_disabled())

View File

@ -10,13 +10,15 @@
#ifndef _ASM_POWERPC_COPRO_H #ifndef _ASM_POWERPC_COPRO_H
#define _ASM_POWERPC_COPRO_H #define _ASM_POWERPC_COPRO_H
#include <linux/mm_types.h>
struct copro_slb struct copro_slb
{ {
u64 esid, vsid; u64 esid, vsid;
}; };
int copro_handle_mm_fault(struct mm_struct *mm, unsigned long ea, int copro_handle_mm_fault(struct mm_struct *mm, unsigned long ea,
unsigned long dsisr, unsigned *flt); unsigned long dsisr, vm_fault_t *flt);
int copro_calculate_slb(struct mm_struct *mm, u64 ea, struct copro_slb *slb); int copro_calculate_slb(struct mm_struct *mm, u64 ea, struct copro_slb *slb);

View File

@ -77,7 +77,7 @@ struct page *kvm_alloc_hpt_cma(unsigned long nr_pages)
VM_BUG_ON(order_base_2(nr_pages) < KVM_CMA_CHUNK_ORDER - PAGE_SHIFT); VM_BUG_ON(order_base_2(nr_pages) < KVM_CMA_CHUNK_ORDER - PAGE_SHIFT);
return cma_alloc(kvm_cma, nr_pages, order_base_2(HPT_ALIGN_PAGES), return cma_alloc(kvm_cma, nr_pages, order_base_2(HPT_ALIGN_PAGES),
GFP_KERNEL); false);
} }
EXPORT_SYMBOL_GPL(kvm_alloc_hpt_cma); EXPORT_SYMBOL_GPL(kvm_alloc_hpt_cma);

View File

@ -34,7 +34,7 @@
* to handle fortunately. * to handle fortunately.
*/ */
int copro_handle_mm_fault(struct mm_struct *mm, unsigned long ea, int copro_handle_mm_fault(struct mm_struct *mm, unsigned long ea,
unsigned long dsisr, unsigned *flt) unsigned long dsisr, vm_fault_t *flt)
{ {
struct vm_area_struct *vma; struct vm_area_struct *vma;
unsigned long is_write; unsigned long is_write;

View File

@ -155,7 +155,7 @@ static noinline int bad_access(struct pt_regs *regs, unsigned long address)
} }
static int do_sigbus(struct pt_regs *regs, unsigned long address, static int do_sigbus(struct pt_regs *regs, unsigned long address,
unsigned int fault) vm_fault_t fault)
{ {
siginfo_t info; siginfo_t info;
unsigned int lsb = 0; unsigned int lsb = 0;
@ -186,7 +186,8 @@ static int do_sigbus(struct pt_regs *regs, unsigned long address,
return 0; return 0;
} }
static int mm_fault_error(struct pt_regs *regs, unsigned long addr, int fault) static int mm_fault_error(struct pt_regs *regs, unsigned long addr,
vm_fault_t fault)
{ {
/* /*
* Kernel page fault interrupted by SIGKILL. We have no reason to * Kernel page fault interrupted by SIGKILL. We have no reason to
@ -414,7 +415,7 @@ static int __do_page_fault(struct pt_regs *regs, unsigned long address,
int is_exec = TRAP(regs) == 0x400; int is_exec = TRAP(regs) == 0x400;
int is_user = user_mode(regs); int is_user = user_mode(regs);
int is_write = page_fault_is_write(error_code); int is_write = page_fault_is_write(error_code);
int fault, major = 0; vm_fault_t fault, major = 0;
bool must_retry = false; bool must_retry = false;
if (notify_page_fault(regs)) if (notify_page_fault(regs))

View File

@ -111,7 +111,7 @@ int spufs_handle_class1(struct spu_context *ctx)
{ {
u64 ea, dsisr, access; u64 ea, dsisr, access;
unsigned long flags; unsigned long flags;
unsigned flt = 0; vm_fault_t flt = 0;
int ret; int ret;
/* /*

View File

@ -41,7 +41,8 @@ asmlinkage void do_page_fault(struct pt_regs *regs)
struct mm_struct *mm; struct mm_struct *mm;
unsigned long addr, cause; unsigned long addr, cause;
unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE; unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE;
int fault, code = SEGV_MAPERR; int code = SEGV_MAPERR;
vm_fault_t fault;
cause = regs->scause; cause = regs->scause;
addr = regs->sbadaddr; addr = regs->sbadaddr;

View File

@ -341,7 +341,8 @@ static noinline int signal_return(struct pt_regs *regs)
return -EACCES; return -EACCES;
} }
static noinline void do_fault_error(struct pt_regs *regs, int access, int fault) static noinline void do_fault_error(struct pt_regs *regs, int access,
vm_fault_t fault)
{ {
int si_code; int si_code;
@ -401,7 +402,7 @@ static noinline void do_fault_error(struct pt_regs *regs, int access, int fault)
* 11 Page translation -> Not present (nullification) * 11 Page translation -> Not present (nullification)
* 3b Region third trans. -> Not present (nullification) * 3b Region third trans. -> Not present (nullification)
*/ */
static inline int do_exception(struct pt_regs *regs, int access) static inline vm_fault_t do_exception(struct pt_regs *regs, int access)
{ {
struct gmap *gmap; struct gmap *gmap;
struct task_struct *tsk; struct task_struct *tsk;
@ -411,7 +412,7 @@ static inline int do_exception(struct pt_regs *regs, int access)
unsigned long trans_exc_code; unsigned long trans_exc_code;
unsigned long address; unsigned long address;
unsigned int flags; unsigned int flags;
int fault; vm_fault_t fault;
tsk = current; tsk = current;
/* /*
@ -564,7 +565,8 @@ out:
void do_protection_exception(struct pt_regs *regs) void do_protection_exception(struct pt_regs *regs)
{ {
unsigned long trans_exc_code; unsigned long trans_exc_code;
int access, fault; int access;
vm_fault_t fault;
trans_exc_code = regs->int_parm_long; trans_exc_code = regs->int_parm_long;
/* /*
@ -599,7 +601,8 @@ NOKPROBE_SYMBOL(do_protection_exception);
void do_dat_exception(struct pt_regs *regs) void do_dat_exception(struct pt_regs *regs)
{ {
int access, fault; int access;
vm_fault_t fault;
access = VM_READ | VM_EXEC | VM_WRITE; access = VM_READ | VM_EXEC | VM_WRITE;
fault = do_exception(regs, access); fault = do_exception(regs, access);

View File

@ -56,15 +56,15 @@ const struct of_cpu_method __cpu_method_of_table_sentinel
static void sh_of_smp_probe(void) static void sh_of_smp_probe(void)
{ {
struct device_node *np = 0; struct device_node *np;
const char *method = 0; const char *method = NULL;
const struct of_cpu_method *m = __cpu_method_of_table; const struct of_cpu_method *m = __cpu_method_of_table;
pr_info("SH generic board support: scanning for cpus\n"); pr_info("SH generic board support: scanning for cpus\n");
init_cpu_possible(cpumask_of(0)); init_cpu_possible(cpumask_of(0));
while ((np = of_find_node_by_type(np, "cpu"))) { for_each_node_by_type(np, "cpu") {
const __be32 *cell = of_get_property(np, "reg", NULL); const __be32 *cell = of_get_property(np, "reg", NULL);
u64 id = -1; u64 id = -1;
if (cell) id = of_read_number(cell, of_n_addr_cells(np)); if (cell) id = of_read_number(cell, of_n_addr_cells(np));
@ -80,6 +80,7 @@ static void sh_of_smp_probe(void)
if (!method) { if (!method) {
np = of_find_node_by_name(NULL, "cpus"); np = of_find_node_by_name(NULL, "cpus");
of_property_read_string(np, "enable-method", &method); of_property_read_string(np, "enable-method", &method);
of_node_put(np);
} }
pr_info("CPU enable method: %s\n", method); pr_info("CPU enable method: %s\n", method);

View File

@ -4,6 +4,7 @@
#include <asm/ptrace.h> #include <asm/ptrace.h>
#include <asm/string.h> #include <asm/string.h>
#include <linux/kernel.h>
/* /*
* KEXEC_SOURCE_MEMORY_LIMIT maximum page get_free_page can return. * KEXEC_SOURCE_MEMORY_LIMIT maximum page get_free_page can return.
@ -61,7 +62,7 @@ static inline void crash_setup_regs(struct pt_regs *newregs,
__asm__ __volatile__ ("stc gbr, %0" : "=r" (newregs->gbr)); __asm__ __volatile__ ("stc gbr, %0" : "=r" (newregs->gbr));
__asm__ __volatile__ ("stc sr, %0" : "=r" (newregs->sr)); __asm__ __volatile__ ("stc sr, %0" : "=r" (newregs->sr));
newregs->pc = (unsigned long)current_text_addr(); newregs->pc = _THIS_IP_;
} }
} }
#else #else

View File

@ -599,7 +599,7 @@ struct dwarf_frame *dwarf_unwind_stack(unsigned long pc,
* time this function makes its first function call. * time this function makes its first function call.
*/ */
if (!pc || !prev) if (!pc || !prev)
pc = (unsigned long)current_text_addr(); pc = _THIS_IP_;
#ifdef CONFIG_FUNCTION_GRAPH_TRACER #ifdef CONFIG_FUNCTION_GRAPH_TRACER
/* /*

View File

@ -313,7 +313,7 @@ do_sigbus(struct pt_regs *regs, unsigned long error_code, unsigned long address)
static noinline int static noinline int
mm_fault_error(struct pt_regs *regs, unsigned long error_code, mm_fault_error(struct pt_regs *regs, unsigned long error_code,
unsigned long address, unsigned int fault) unsigned long address, vm_fault_t fault)
{ {
/* /*
* Pagefault was interrupted by SIGKILL. We have no reason to * Pagefault was interrupted by SIGKILL. We have no reason to
@ -396,7 +396,7 @@ asmlinkage void __kprobes do_page_fault(struct pt_regs *regs,
struct task_struct *tsk; struct task_struct *tsk;
struct mm_struct *mm; struct mm_struct *mm;
struct vm_area_struct * vma; struct vm_area_struct * vma;
int fault; vm_fault_t fault;
unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE; unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE;
tsk = current; tsk = current;

View File

@ -166,7 +166,8 @@ asmlinkage void do_sparc_fault(struct pt_regs *regs, int text_fault, int write,
unsigned int fixup; unsigned int fixup;
unsigned long g2; unsigned long g2;
int from_user = !(regs->psr & PSR_PS); int from_user = !(regs->psr & PSR_PS);
int fault, code; int code;
vm_fault_t fault;
unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE; unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE;
if (text_fault) if (text_fault)

View File

@ -278,7 +278,8 @@ asmlinkage void __kprobes do_sparc64_fault(struct pt_regs *regs)
struct mm_struct *mm = current->mm; struct mm_struct *mm = current->mm;
struct vm_area_struct *vma; struct vm_area_struct *vma;
unsigned int insn = 0; unsigned int insn = 0;
int si_code, fault_code, fault; int si_code, fault_code;
vm_fault_t fault;
unsigned long address, mm_rss; unsigned long address, mm_rss;
unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE; unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE;

View File

@ -72,7 +72,7 @@ good_area:
} }
do { do {
int fault; vm_fault_t fault;
fault = handle_mm_fault(vma, address, flags); fault = handle_mm_fault(vma, address, flags);

View File

@ -168,11 +168,11 @@ static inline bool access_error(unsigned int fsr, struct vm_area_struct *vma)
return vma->vm_flags & mask ? false : true; return vma->vm_flags & mask ? false : true;
} }
static int __do_pf(struct mm_struct *mm, unsigned long addr, unsigned int fsr, static vm_fault_t __do_pf(struct mm_struct *mm, unsigned long addr,
unsigned int flags, struct task_struct *tsk) unsigned int fsr, unsigned int flags, struct task_struct *tsk)
{ {
struct vm_area_struct *vma; struct vm_area_struct *vma;
int fault; vm_fault_t fault;
vma = find_vma(mm, addr); vma = find_vma(mm, addr);
fault = VM_FAULT_BADMAP; fault = VM_FAULT_BADMAP;
@ -209,7 +209,8 @@ static int do_pf(unsigned long addr, unsigned int fsr, struct pt_regs *regs)
{ {
struct task_struct *tsk; struct task_struct *tsk;
struct mm_struct *mm; struct mm_struct *mm;
int fault, sig, code; int sig, code;
vm_fault_t fault;
unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE; unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE;
tsk = current; tsk = current;

View File

@ -16,6 +16,7 @@
#include <linux/prefetch.h> /* prefetchw */ #include <linux/prefetch.h> /* prefetchw */
#include <linux/context_tracking.h> /* exception_enter(), ... */ #include <linux/context_tracking.h> /* exception_enter(), ... */
#include <linux/uaccess.h> /* faulthandler_disabled() */ #include <linux/uaccess.h> /* faulthandler_disabled() */
#include <linux/mm_types.h>
#include <asm/cpufeature.h> /* boot_cpu_has, ... */ #include <asm/cpufeature.h> /* boot_cpu_has, ... */
#include <asm/traps.h> /* dotraplinkage, ... */ #include <asm/traps.h> /* dotraplinkage, ... */
@ -999,7 +1000,7 @@ do_sigbus(struct pt_regs *regs, unsigned long error_code, unsigned long address,
static noinline void static noinline void
mm_fault_error(struct pt_regs *regs, unsigned long error_code, mm_fault_error(struct pt_regs *regs, unsigned long error_code,
unsigned long address, u32 *pkey, unsigned int fault) unsigned long address, u32 *pkey, vm_fault_t fault)
{ {
if (fatal_signal_pending(current) && !(error_code & X86_PF_USER)) { if (fatal_signal_pending(current) && !(error_code & X86_PF_USER)) {
no_context(regs, error_code, address, 0, 0); no_context(regs, error_code, address, 0, 0);
@ -1213,7 +1214,7 @@ __do_page_fault(struct pt_regs *regs, unsigned long error_code,
struct vm_area_struct *vma; struct vm_area_struct *vma;
struct task_struct *tsk; struct task_struct *tsk;
struct mm_struct *mm; struct mm_struct *mm;
int fault, major = 0; vm_fault_t fault, major = 0;
unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE; unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE;
u32 pkey; u32 pkey;

View File

@ -137,7 +137,7 @@ static void *xtensa_dma_alloc(struct device *dev, size_t size,
if (gfpflags_allow_blocking(flag)) if (gfpflags_allow_blocking(flag))
page = dma_alloc_from_contiguous(dev, count, get_order(size), page = dma_alloc_from_contiguous(dev, count, get_order(size),
flag); flag & __GFP_NOWARN);
if (!page) if (!page)
page = alloc_pages(flag, get_order(size)); page = alloc_pages(flag, get_order(size));

View File

@ -42,7 +42,7 @@ void do_page_fault(struct pt_regs *regs)
int code; int code;
int is_write, is_exec; int is_write, is_exec;
int fault; vm_fault_t fault;
unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE; unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE;
code = SEGV_MAPERR; code = SEGV_MAPERR;

View File

@ -219,11 +219,6 @@ static ssize_t firmware_loading_show(struct device *dev,
return sprintf(buf, "%d\n", loading); return sprintf(buf, "%d\n", loading);
} }
/* Some architectures don't have PAGE_KERNEL_RO */
#ifndef PAGE_KERNEL_RO
#define PAGE_KERNEL_RO PAGE_KERNEL
#endif
/* one pages buffer should be mapped/unmapped only once */ /* one pages buffer should be mapped/unmapped only once */
static int map_fw_priv_pages(struct fw_priv *fw_priv) static int map_fw_priv_pages(struct fw_priv *fw_priv)
{ {

View File

@ -736,8 +736,6 @@ int hotplug_memory_register(int nid, struct mem_section *section)
mem->section_count++; mem->section_count++;
} }
if (mem->section_count == sections_per_block)
ret = register_mem_sect_under_node(mem, nid, false);
out: out:
mutex_unlock(&mem_sysfs_mutex); mutex_unlock(&mem_sysfs_mutex);
return ret; return ret;

View File

@ -399,18 +399,12 @@ static int __ref get_nid_for_pfn(unsigned long pfn)
} }
/* register memory section under specified node if it spans that node */ /* register memory section under specified node if it spans that node */
int register_mem_sect_under_node(struct memory_block *mem_blk, int nid, int register_mem_sect_under_node(struct memory_block *mem_blk, void *arg)
bool check_nid)
{ {
int ret; int ret, nid = *(int *)arg;
unsigned long pfn, sect_start_pfn, sect_end_pfn; unsigned long pfn, sect_start_pfn, sect_end_pfn;
if (!mem_blk)
return -EFAULT;
mem_blk->nid = nid; mem_blk->nid = nid;
if (!node_online(nid))
return 0;
sect_start_pfn = section_nr_to_pfn(mem_blk->start_section_nr); sect_start_pfn = section_nr_to_pfn(mem_blk->start_section_nr);
sect_end_pfn = section_nr_to_pfn(mem_blk->end_section_nr); sect_end_pfn = section_nr_to_pfn(mem_blk->end_section_nr);
@ -433,7 +427,7 @@ int register_mem_sect_under_node(struct memory_block *mem_blk, int nid,
* case, during hotplug we know that all pages in the memory * case, during hotplug we know that all pages in the memory
* block belong to the same node. * block belong to the same node.
*/ */
if (check_nid) { if (system_state == SYSTEM_BOOTING) {
page_nid = get_nid_for_pfn(pfn); page_nid = get_nid_for_pfn(pfn);
if (page_nid < 0) if (page_nid < 0)
continue; continue;
@ -490,41 +484,10 @@ int unregister_mem_sect_under_nodes(struct memory_block *mem_blk,
return 0; return 0;
} }
int link_mem_sections(int nid, unsigned long start_pfn, unsigned long nr_pages, int link_mem_sections(int nid, unsigned long start_pfn, unsigned long end_pfn)
bool check_nid)
{ {
unsigned long end_pfn = start_pfn + nr_pages; return walk_memory_range(start_pfn, end_pfn, (void *)&nid,
unsigned long pfn; register_mem_sect_under_node);
struct memory_block *mem_blk = NULL;
int err = 0;
for (pfn = start_pfn; pfn < end_pfn; pfn += PAGES_PER_SECTION) {
unsigned long section_nr = pfn_to_section_nr(pfn);
struct mem_section *mem_sect;
int ret;
if (!present_section_nr(section_nr))
continue;
mem_sect = __nr_to_section(section_nr);
/* same memblock ? */
if (mem_blk)
if ((section_nr >= mem_blk->start_section_nr) &&
(section_nr <= mem_blk->end_section_nr))
continue;
mem_blk = find_memory_block_hinted(mem_sect, mem_blk);
ret = register_mem_sect_under_node(mem_blk, nid, check_nid);
if (!err)
err = ret;
/* discard ref obtained in find_memory_block() */
}
if (mem_blk)
kobject_put(&mem_blk->dev.kobj);
return err;
} }
#ifdef CONFIG_HUGETLBFS #ifdef CONFIG_HUGETLBFS

View File

@ -474,7 +474,7 @@ static int dax_mmap(struct file *filp, struct vm_area_struct *vma)
return rc; return rc;
vma->vm_ops = &dax_vm_ops; vma->vm_ops = &dax_vm_ops;
vma->vm_flags |= VM_MIXEDMAP | VM_HUGEPAGE; vma->vm_flags |= VM_HUGEPAGE;
return 0; return 0;
} }

View File

@ -1205,7 +1205,7 @@ static int ioctl_get_cycle_timer2(struct client *client, union ioctl_arg *arg)
{ {
struct fw_cdev_get_cycle_timer2 *a = &arg->get_cycle_timer2; struct fw_cdev_get_cycle_timer2 *a = &arg->get_cycle_timer2;
struct fw_card *card = client->device->card; struct fw_card *card = client->device->card;
struct timespec ts = {0, 0}; struct timespec64 ts = {0, 0};
u32 cycle_time; u32 cycle_time;
int ret = 0; int ret = 0;
@ -1214,9 +1214,9 @@ static int ioctl_get_cycle_timer2(struct client *client, union ioctl_arg *arg)
cycle_time = card->driver->read_csr(card, CSR_CYCLE_TIME); cycle_time = card->driver->read_csr(card, CSR_CYCLE_TIME);
switch (a->clk_id) { switch (a->clk_id) {
case CLOCK_REALTIME: getnstimeofday(&ts); break; case CLOCK_REALTIME: ktime_get_real_ts64(&ts); break;
case CLOCK_MONOTONIC: ktime_get_ts(&ts); break; case CLOCK_MONOTONIC: ktime_get_ts64(&ts); break;
case CLOCK_MONOTONIC_RAW: getrawmonotonic(&ts); break; case CLOCK_MONOTONIC_RAW: ktime_get_raw_ts64(&ts); break;
default: default:
ret = -EINVAL; ret = -EINVAL;
} }

View File

@ -2620,7 +2620,7 @@ static void *alloc_coherent(struct device *dev, size_t size,
return NULL; return NULL;
page = dma_alloc_from_contiguous(dev, size >> PAGE_SHIFT, page = dma_alloc_from_contiguous(dev, size >> PAGE_SHIFT,
get_order(size), flag); get_order(size), flag & __GFP_NOWARN);
if (!page) if (!page)
return NULL; return NULL;
} }

View File

@ -508,7 +508,7 @@ static void do_fault(struct work_struct *work)
{ {
struct fault *fault = container_of(work, struct fault, work); struct fault *fault = container_of(work, struct fault, work);
struct vm_area_struct *vma; struct vm_area_struct *vma;
int ret = VM_FAULT_ERROR; vm_fault_t ret = VM_FAULT_ERROR;
unsigned int flags = 0; unsigned int flags = 0;
struct mm_struct *mm; struct mm_struct *mm;
u64 address; u64 address;

View File

@ -3758,7 +3758,8 @@ static void *intel_alloc_coherent(struct device *dev, size_t size,
if (gfpflags_allow_blocking(flags)) { if (gfpflags_allow_blocking(flags)) {
unsigned int count = size >> PAGE_SHIFT; unsigned int count = size >> PAGE_SHIFT;
page = dma_alloc_from_contiguous(dev, count, order, flags); page = dma_alloc_from_contiguous(dev, count, order,
flags & __GFP_NOWARN);
if (page && iommu_no_mapping(dev) && if (page && iommu_no_mapping(dev) &&
page_to_phys(page) + size > dev->coherent_dma_mask) { page_to_phys(page) + size > dev->coherent_dma_mask) {
dma_release_from_contiguous(dev, page, count); dma_release_from_contiguous(dev, page, count);

View File

@ -24,6 +24,7 @@
#include <linux/pci-ats.h> #include <linux/pci-ats.h>
#include <linux/dmar.h> #include <linux/dmar.h>
#include <linux/interrupt.h> #include <linux/interrupt.h>
#include <linux/mm_types.h>
#include <asm/page.h> #include <asm/page.h>
#define PASID_ENTRY_P BIT_ULL(0) #define PASID_ENTRY_P BIT_ULL(0)
@ -594,7 +595,8 @@ static irqreturn_t prq_event_thread(int irq, void *d)
struct vm_area_struct *vma; struct vm_area_struct *vma;
struct page_req_dsc *req; struct page_req_dsc *req;
struct qi_desc resp; struct qi_desc resp;
int ret, result; int result;
vm_fault_t ret;
u64 address; u64 address;
handled = 1; handled = 1;

View File

@ -134,7 +134,7 @@ static int cxl_handle_segment_miss(struct cxl_context *ctx,
int cxl_handle_mm_fault(struct mm_struct *mm, u64 dsisr, u64 dar) int cxl_handle_mm_fault(struct mm_struct *mm, u64 dsisr, u64 dar)
{ {
unsigned flt = 0; vm_fault_t flt = 0;
int result; int result;
unsigned long access, flags, inv_flags = 0; unsigned long access, flags, inv_flags = 0;

View File

@ -2,6 +2,7 @@
// Copyright 2017 IBM Corp. // Copyright 2017 IBM Corp.
#include <linux/sched/mm.h> #include <linux/sched/mm.h>
#include <linux/mutex.h> #include <linux/mutex.h>
#include <linux/mm_types.h>
#include <linux/mmu_context.h> #include <linux/mmu_context.h>
#include <asm/copro.h> #include <asm/copro.h>
#include <asm/pnv-ocxl.h> #include <asm/pnv-ocxl.h>
@ -126,7 +127,7 @@ static void ack_irq(struct spa *spa, enum xsl_response r)
static void xsl_fault_handler_bh(struct work_struct *fault_work) static void xsl_fault_handler_bh(struct work_struct *fault_work)
{ {
unsigned int flt = 0; vm_fault_t flt = 0;
unsigned long access, flags, inv_flags = 0; unsigned long access, flags, inv_flags = 0;
enum xsl_response r; enum xsl_response r;
struct xsl_fault *fault = container_of(fault_work, struct xsl_fault, struct xsl_fault *fault = container_of(fault_work, struct xsl_fault,

View File

@ -68,7 +68,7 @@ static void vmcp_response_alloc(struct vmcp_session *session)
* anymore the system won't work anyway. * anymore the system won't work anyway.
*/ */
if (order > 2) if (order > 2)
page = cma_alloc(vmcp_cma, nr_pages, 0, GFP_KERNEL); page = cma_alloc(vmcp_cma, nr_pages, 0, false);
if (page) { if (page) {
session->response = (char *)page_to_phys(page); session->response = (char *)page_to_phys(page);
session->cma_alloc = 1; session->cma_alloc = 1;

View File

@ -39,7 +39,7 @@ static int ion_cma_allocate(struct ion_heap *heap, struct ion_buffer *buffer,
if (align > CONFIG_CMA_ALIGNMENT) if (align > CONFIG_CMA_ALIGNMENT)
align = CONFIG_CMA_ALIGNMENT; align = CONFIG_CMA_ALIGNMENT;
pages = cma_alloc(cma_heap->cma, nr_pages, align, GFP_KERNEL); pages = cma_alloc(cma_heap->cma, nr_pages, align, false);
if (!pages) if (!pages)
return -ENOMEM; return -ENOMEM;

View File

@ -3102,7 +3102,7 @@ static inline void __do_contiguous_readpages(struct extent_io_tree *tree,
for (index = 0; index < nr_pages; index++) { for (index = 0; index < nr_pages; index++) {
__do_readpage(tree, pages[index], btrfs_get_extent, em_cached, __do_readpage(tree, pages[index], btrfs_get_extent, em_cached,
bio, 0, bio_flags, 0, prev_em_start); bio, 0, bio_flags, REQ_RAHEAD, prev_em_start);
put_page(pages[index]); put_page(pages[index]);
} }
} }

View File

@ -45,6 +45,7 @@
#include <linux/mpage.h> #include <linux/mpage.h>
#include <linux/bit_spinlock.h> #include <linux/bit_spinlock.h>
#include <linux/pagevec.h> #include <linux/pagevec.h>
#include <linux/sched/mm.h>
#include <trace/events/block.h> #include <trace/events/block.h>
static int fsync_buffers_list(spinlock_t *lock, struct list_head *list); static int fsync_buffers_list(spinlock_t *lock, struct list_head *list);
@ -813,12 +814,16 @@ struct buffer_head *alloc_page_buffers(struct page *page, unsigned long size,
bool retry) bool retry)
{ {
struct buffer_head *bh, *head; struct buffer_head *bh, *head;
gfp_t gfp = GFP_NOFS; gfp_t gfp = GFP_NOFS | __GFP_ACCOUNT;
long offset; long offset;
struct mem_cgroup *memcg;
if (retry) if (retry)
gfp |= __GFP_NOFAIL; gfp |= __GFP_NOFAIL;
memcg = get_mem_cgroup_from_page(page);
memalloc_use_memcg(memcg);
head = NULL; head = NULL;
offset = PAGE_SIZE; offset = PAGE_SIZE;
while ((offset -= size) >= 0) { while ((offset -= size) >= 0) {
@ -835,6 +840,9 @@ struct buffer_head *alloc_page_buffers(struct page *page, unsigned long size,
/* Link the buffer to its page */ /* Link the buffer to its page */
set_bh_page(bh, page, offset); set_bh_page(bh, page, offset);
} }
out:
memalloc_unuse_memcg();
mem_cgroup_put(memcg);
return head; return head;
/* /*
* In case anything failed, we just free everything we got. * In case anything failed, we just free everything we got.
@ -848,7 +856,7 @@ no_grow:
} while (head); } while (head);
} }
return NULL; goto out;
} }
EXPORT_SYMBOL_GPL(alloc_page_buffers); EXPORT_SYMBOL_GPL(alloc_page_buffers);

View File

@ -292,7 +292,8 @@ void take_dentry_name_snapshot(struct name_snapshot *name, struct dentry *dentry
spin_unlock(&dentry->d_lock); spin_unlock(&dentry->d_lock);
name->name = p->name; name->name = p->name;
} else { } else {
memcpy(name->inline_name, dentry->d_iname, DNAME_INLINE_LEN); memcpy(name->inline_name, dentry->d_iname,
dentry->d_name.len + 1);
spin_unlock(&dentry->d_lock); spin_unlock(&dentry->d_lock);
name->name = name->inline_name; name->name = name->inline_name;
} }

View File

@ -126,7 +126,6 @@ static int ext2_file_mmap(struct file *file, struct vm_area_struct *vma)
file_accessed(file); file_accessed(file);
vma->vm_ops = &ext2_dax_vm_ops; vma->vm_ops = &ext2_dax_vm_ops;
vma->vm_flags |= VM_MIXEDMAP;
return 0; return 0;
} }
#else #else

View File

@ -3062,7 +3062,7 @@ static inline void ext4_set_de_type(struct super_block *sb,
/* readpages.c */ /* readpages.c */
extern int ext4_mpage_readpages(struct address_space *mapping, extern int ext4_mpage_readpages(struct address_space *mapping,
struct list_head *pages, struct page *page, struct list_head *pages, struct page *page,
unsigned nr_pages); unsigned nr_pages, bool is_readahead);
/* symlink.c */ /* symlink.c */
extern const struct inode_operations ext4_encrypted_symlink_inode_operations; extern const struct inode_operations ext4_encrypted_symlink_inode_operations;

View File

@ -374,7 +374,7 @@ static int ext4_file_mmap(struct file *file, struct vm_area_struct *vma)
file_accessed(file); file_accessed(file);
if (IS_DAX(file_inode(file))) { if (IS_DAX(file_inode(file))) {
vma->vm_ops = &ext4_dax_vm_ops; vma->vm_ops = &ext4_dax_vm_ops;
vma->vm_flags |= VM_MIXEDMAP | VM_HUGEPAGE; vma->vm_flags |= VM_HUGEPAGE;
} else { } else {
vma->vm_ops = &ext4_file_vm_ops; vma->vm_ops = &ext4_file_vm_ops;
} }

View File

@ -3325,7 +3325,8 @@ static int ext4_readpage(struct file *file, struct page *page)
ret = ext4_readpage_inline(inode, page); ret = ext4_readpage_inline(inode, page);
if (ret == -EAGAIN) if (ret == -EAGAIN)
return ext4_mpage_readpages(page->mapping, NULL, page, 1); return ext4_mpage_readpages(page->mapping, NULL, page, 1,
false);
return ret; return ret;
} }
@ -3340,7 +3341,7 @@ ext4_readpages(struct file *file, struct address_space *mapping,
if (ext4_has_inline_data(inode)) if (ext4_has_inline_data(inode))
return 0; return 0;
return ext4_mpage_readpages(mapping, pages, NULL, nr_pages); return ext4_mpage_readpages(mapping, pages, NULL, nr_pages, true);
} }
static void ext4_invalidatepage(struct page *page, unsigned int offset, static void ext4_invalidatepage(struct page *page, unsigned int offset,

View File

@ -98,7 +98,7 @@ static void mpage_end_io(struct bio *bio)
int ext4_mpage_readpages(struct address_space *mapping, int ext4_mpage_readpages(struct address_space *mapping,
struct list_head *pages, struct page *page, struct list_head *pages, struct page *page,
unsigned nr_pages) unsigned nr_pages, bool is_readahead)
{ {
struct bio *bio = NULL; struct bio *bio = NULL;
sector_t last_block_in_bio = 0; sector_t last_block_in_bio = 0;
@ -259,7 +259,8 @@ int ext4_mpage_readpages(struct address_space *mapping,
bio->bi_iter.bi_sector = blocks[0] << (blkbits - 9); bio->bi_iter.bi_sector = blocks[0] << (blkbits - 9);
bio->bi_end_io = mpage_end_io; bio->bi_end_io = mpage_end_io;
bio->bi_private = ctx; bio->bi_private = ctx;
bio_set_op_attrs(bio, REQ_OP_READ, 0); bio_set_op_attrs(bio, REQ_OP_READ,
is_readahead ? REQ_RAHEAD : 0);
} }
length = first_hole << blkbits; length = first_hole << blkbits;

View File

@ -1421,6 +1421,11 @@ out:
/* /*
* This function was originally taken from fs/mpage.c, and customized for f2fs. * This function was originally taken from fs/mpage.c, and customized for f2fs.
* Major change was from block_size == page_size in f2fs by default. * Major change was from block_size == page_size in f2fs by default.
*
* Note that the aops->readpages() function is ONLY used for read-ahead. If
* this function ever deviates from doing just read-ahead, it should either
* use ->readpage() or do the necessary surgery to decouple ->readpages()
* readom read-ahead.
*/ */
static int f2fs_mpage_readpages(struct address_space *mapping, static int f2fs_mpage_readpages(struct address_space *mapping,
struct list_head *pages, struct page *page, struct list_head *pages, struct page *page,

View File

@ -19,7 +19,7 @@
#define HOSTFS_ATTR_ATIME_SET 128 #define HOSTFS_ATTR_ATIME_SET 128
#define HOSTFS_ATTR_MTIME_SET 256 #define HOSTFS_ATTR_MTIME_SET 256
/* These two are unused by hostfs. */ /* This one is unused by hostfs. */
#define HOSTFS_ATTR_FORCE 512 /* Not a change, but a change it */ #define HOSTFS_ATTR_FORCE 512 /* Not a change, but a change it */
#define HOSTFS_ATTR_ATTR_FLAG 1024 #define HOSTFS_ATTR_ATTR_FLAG 1024

View File

@ -334,16 +334,23 @@ long hpfs_ioctl(struct file *file, unsigned cmd, unsigned long arg);
* local time (HPFS) to GMT (Unix) * local time (HPFS) to GMT (Unix)
*/ */
static inline time_t local_to_gmt(struct super_block *s, time32_t t) static inline time64_t local_to_gmt(struct super_block *s, time32_t t)
{ {
extern struct timezone sys_tz; extern struct timezone sys_tz;
return t + sys_tz.tz_minuteswest * 60 + hpfs_sb(s)->sb_timeshift; return t + sys_tz.tz_minuteswest * 60 + hpfs_sb(s)->sb_timeshift;
} }
static inline time32_t gmt_to_local(struct super_block *s, time_t t) static inline time32_t gmt_to_local(struct super_block *s, time64_t t)
{ {
extern struct timezone sys_tz; extern struct timezone sys_tz;
return t - sys_tz.tz_minuteswest * 60 - hpfs_sb(s)->sb_timeshift; t = t - sys_tz.tz_minuteswest * 60 - hpfs_sb(s)->sb_timeshift;
return clamp_t(time64_t, t, 0, U32_MAX);
}
static inline time32_t local_get_seconds(struct super_block *s)
{
return gmt_to_local(s, ktime_get_real_seconds());
} }
/* /*

View File

@ -11,7 +11,7 @@
static void hpfs_update_directory_times(struct inode *dir) static void hpfs_update_directory_times(struct inode *dir)
{ {
time_t t = get_seconds(); time64_t t = local_to_gmt(dir->i_sb, local_get_seconds(dir->i_sb));
if (t == dir->i_mtime.tv_sec && if (t == dir->i_mtime.tv_sec &&
t == dir->i_ctime.tv_sec) t == dir->i_ctime.tv_sec)
return; return;
@ -50,7 +50,7 @@ static int hpfs_mkdir(struct inode *dir, struct dentry *dentry, umode_t mode)
/*dee.archive = 0;*/ /*dee.archive = 0;*/
dee.hidden = name[0] == '.'; dee.hidden = name[0] == '.';
dee.fnode = cpu_to_le32(fno); dee.fnode = cpu_to_le32(fno);
dee.creation_date = dee.write_date = dee.read_date = cpu_to_le32(gmt_to_local(dir->i_sb, get_seconds())); dee.creation_date = dee.write_date = dee.read_date = cpu_to_le32(local_get_seconds(dir->i_sb));
result = new_inode(dir->i_sb); result = new_inode(dir->i_sb);
if (!result) if (!result)
goto bail2; goto bail2;
@ -91,7 +91,7 @@ static int hpfs_mkdir(struct inode *dir, struct dentry *dentry, umode_t mode)
dnode->root_dnode = 1; dnode->root_dnode = 1;
dnode->up = cpu_to_le32(fno); dnode->up = cpu_to_le32(fno);
de = hpfs_add_de(dir->i_sb, dnode, "\001\001", 2, 0); de = hpfs_add_de(dir->i_sb, dnode, "\001\001", 2, 0);
de->creation_date = de->write_date = de->read_date = cpu_to_le32(gmt_to_local(dir->i_sb, get_seconds())); de->creation_date = de->write_date = de->read_date = cpu_to_le32(local_get_seconds(dir->i_sb));
if (!(mode & 0222)) de->read_only = 1; if (!(mode & 0222)) de->read_only = 1;
de->first = de->directory = 1; de->first = de->directory = 1;
/*de->hidden = de->system = 0;*/ /*de->hidden = de->system = 0;*/
@ -151,7 +151,7 @@ static int hpfs_create(struct inode *dir, struct dentry *dentry, umode_t mode, b
dee.archive = 1; dee.archive = 1;
dee.hidden = name[0] == '.'; dee.hidden = name[0] == '.';
dee.fnode = cpu_to_le32(fno); dee.fnode = cpu_to_le32(fno);
dee.creation_date = dee.write_date = dee.read_date = cpu_to_le32(gmt_to_local(dir->i_sb, get_seconds())); dee.creation_date = dee.write_date = dee.read_date = cpu_to_le32(local_get_seconds(dir->i_sb));
result = new_inode(dir->i_sb); result = new_inode(dir->i_sb);
if (!result) if (!result)
@ -238,7 +238,7 @@ static int hpfs_mknod(struct inode *dir, struct dentry *dentry, umode_t mode, de
dee.archive = 1; dee.archive = 1;
dee.hidden = name[0] == '.'; dee.hidden = name[0] == '.';
dee.fnode = cpu_to_le32(fno); dee.fnode = cpu_to_le32(fno);
dee.creation_date = dee.write_date = dee.read_date = cpu_to_le32(gmt_to_local(dir->i_sb, get_seconds())); dee.creation_date = dee.write_date = dee.read_date = cpu_to_le32(local_get_seconds(dir->i_sb));
result = new_inode(dir->i_sb); result = new_inode(dir->i_sb);
if (!result) if (!result)
@ -314,7 +314,7 @@ static int hpfs_symlink(struct inode *dir, struct dentry *dentry, const char *sy
dee.archive = 1; dee.archive = 1;
dee.hidden = name[0] == '.'; dee.hidden = name[0] == '.';
dee.fnode = cpu_to_le32(fno); dee.fnode = cpu_to_le32(fno);
dee.creation_date = dee.write_date = dee.read_date = cpu_to_le32(gmt_to_local(dir->i_sb, get_seconds())); dee.creation_date = dee.write_date = dee.read_date = cpu_to_le32(local_get_seconds(dir->i_sb));
result = new_inode(dir->i_sb); result = new_inode(dir->i_sb);
if (!result) if (!result)

View File

@ -133,6 +133,17 @@ map_buffer_to_page(struct page *page, struct buffer_head *bh, int page_block)
} while (page_bh != head); } while (page_bh != head);
} }
struct mpage_readpage_args {
struct bio *bio;
struct page *page;
unsigned int nr_pages;
bool is_readahead;
sector_t last_block_in_bio;
struct buffer_head map_bh;
unsigned long first_logical_block;
get_block_t *get_block;
};
/* /*
* This is the worker routine which does all the work of mapping the disk * This is the worker routine which does all the work of mapping the disk
* blocks and constructs largest possible bios, submits them for IO if the * blocks and constructs largest possible bios, submits them for IO if the
@ -142,16 +153,14 @@ map_buffer_to_page(struct page *page, struct buffer_head *bh, int page_block)
* represent the validity of its disk mapping and to decide when to do the next * represent the validity of its disk mapping and to decide when to do the next
* get_block() call. * get_block() call.
*/ */
static struct bio * static struct bio *do_mpage_readpage(struct mpage_readpage_args *args)
do_mpage_readpage(struct bio *bio, struct page *page, unsigned nr_pages,
sector_t *last_block_in_bio, struct buffer_head *map_bh,
unsigned long *first_logical_block, get_block_t get_block,
gfp_t gfp)
{ {
struct page *page = args->page;
struct inode *inode = page->mapping->host; struct inode *inode = page->mapping->host;
const unsigned blkbits = inode->i_blkbits; const unsigned blkbits = inode->i_blkbits;
const unsigned blocks_per_page = PAGE_SIZE >> blkbits; const unsigned blocks_per_page = PAGE_SIZE >> blkbits;
const unsigned blocksize = 1 << blkbits; const unsigned blocksize = 1 << blkbits;
struct buffer_head *map_bh = &args->map_bh;
sector_t block_in_file; sector_t block_in_file;
sector_t last_block; sector_t last_block;
sector_t last_block_in_file; sector_t last_block_in_file;
@ -161,14 +170,24 @@ do_mpage_readpage(struct bio *bio, struct page *page, unsigned nr_pages,
struct block_device *bdev = NULL; struct block_device *bdev = NULL;
int length; int length;
int fully_mapped = 1; int fully_mapped = 1;
int op_flags;
unsigned nblocks; unsigned nblocks;
unsigned relative_block; unsigned relative_block;
gfp_t gfp;
if (args->is_readahead) {
op_flags = REQ_RAHEAD;
gfp = readahead_gfp_mask(page->mapping);
} else {
op_flags = 0;
gfp = mapping_gfp_constraint(page->mapping, GFP_KERNEL);
}
if (page_has_buffers(page)) if (page_has_buffers(page))
goto confused; goto confused;
block_in_file = (sector_t)page->index << (PAGE_SHIFT - blkbits); block_in_file = (sector_t)page->index << (PAGE_SHIFT - blkbits);
last_block = block_in_file + nr_pages * blocks_per_page; last_block = block_in_file + args->nr_pages * blocks_per_page;
last_block_in_file = (i_size_read(inode) + blocksize - 1) >> blkbits; last_block_in_file = (i_size_read(inode) + blocksize - 1) >> blkbits;
if (last_block > last_block_in_file) if (last_block > last_block_in_file)
last_block = last_block_in_file; last_block = last_block_in_file;
@ -178,9 +197,10 @@ do_mpage_readpage(struct bio *bio, struct page *page, unsigned nr_pages,
* Map blocks using the result from the previous get_blocks call first. * Map blocks using the result from the previous get_blocks call first.
*/ */
nblocks = map_bh->b_size >> blkbits; nblocks = map_bh->b_size >> blkbits;
if (buffer_mapped(map_bh) && block_in_file > *first_logical_block && if (buffer_mapped(map_bh) &&
block_in_file < (*first_logical_block + nblocks)) { block_in_file > args->first_logical_block &&
unsigned map_offset = block_in_file - *first_logical_block; block_in_file < (args->first_logical_block + nblocks)) {
unsigned map_offset = block_in_file - args->first_logical_block;
unsigned last = nblocks - map_offset; unsigned last = nblocks - map_offset;
for (relative_block = 0; ; relative_block++) { for (relative_block = 0; ; relative_block++) {
@ -208,9 +228,9 @@ do_mpage_readpage(struct bio *bio, struct page *page, unsigned nr_pages,
if (block_in_file < last_block) { if (block_in_file < last_block) {
map_bh->b_size = (last_block-block_in_file) << blkbits; map_bh->b_size = (last_block-block_in_file) << blkbits;
if (get_block(inode, block_in_file, map_bh, 0)) if (args->get_block(inode, block_in_file, map_bh, 0))
goto confused; goto confused;
*first_logical_block = block_in_file; args->first_logical_block = block_in_file;
} }
if (!buffer_mapped(map_bh)) { if (!buffer_mapped(map_bh)) {
@ -273,43 +293,45 @@ do_mpage_readpage(struct bio *bio, struct page *page, unsigned nr_pages,
/* /*
* This page will go to BIO. Do we need to send this BIO off first? * This page will go to BIO. Do we need to send this BIO off first?
*/ */
if (bio && (*last_block_in_bio != blocks[0] - 1)) if (args->bio && (args->last_block_in_bio != blocks[0] - 1))
bio = mpage_bio_submit(REQ_OP_READ, 0, bio); args->bio = mpage_bio_submit(REQ_OP_READ, op_flags, args->bio);
alloc_new: alloc_new:
if (bio == NULL) { if (args->bio == NULL) {
if (first_hole == blocks_per_page) { if (first_hole == blocks_per_page) {
if (!bdev_read_page(bdev, blocks[0] << (blkbits - 9), if (!bdev_read_page(bdev, blocks[0] << (blkbits - 9),
page)) page))
goto out; goto out;
} }
bio = mpage_alloc(bdev, blocks[0] << (blkbits - 9), args->bio = mpage_alloc(bdev, blocks[0] << (blkbits - 9),
min_t(int, nr_pages, BIO_MAX_PAGES), gfp); min_t(int, args->nr_pages,
if (bio == NULL) BIO_MAX_PAGES),
gfp);
if (args->bio == NULL)
goto confused; goto confused;
} }
length = first_hole << blkbits; length = first_hole << blkbits;
if (bio_add_page(bio, page, length, 0) < length) { if (bio_add_page(args->bio, page, length, 0) < length) {
bio = mpage_bio_submit(REQ_OP_READ, 0, bio); args->bio = mpage_bio_submit(REQ_OP_READ, op_flags, args->bio);
goto alloc_new; goto alloc_new;
} }
relative_block = block_in_file - *first_logical_block; relative_block = block_in_file - args->first_logical_block;
nblocks = map_bh->b_size >> blkbits; nblocks = map_bh->b_size >> blkbits;
if ((buffer_boundary(map_bh) && relative_block == nblocks) || if ((buffer_boundary(map_bh) && relative_block == nblocks) ||
(first_hole != blocks_per_page)) (first_hole != blocks_per_page))
bio = mpage_bio_submit(REQ_OP_READ, 0, bio); args->bio = mpage_bio_submit(REQ_OP_READ, op_flags, args->bio);
else else
*last_block_in_bio = blocks[blocks_per_page - 1]; args->last_block_in_bio = blocks[blocks_per_page - 1];
out: out:
return bio; return args->bio;
confused: confused:
if (bio) if (args->bio)
bio = mpage_bio_submit(REQ_OP_READ, 0, bio); args->bio = mpage_bio_submit(REQ_OP_READ, op_flags, args->bio);
if (!PageUptodate(page)) if (!PageUptodate(page))
block_read_full_page(page, get_block); block_read_full_page(page, args->get_block);
else else
unlock_page(page); unlock_page(page);
goto out; goto out;
@ -363,15 +385,12 @@ int
mpage_readpages(struct address_space *mapping, struct list_head *pages, mpage_readpages(struct address_space *mapping, struct list_head *pages,
unsigned nr_pages, get_block_t get_block) unsigned nr_pages, get_block_t get_block)
{ {
struct bio *bio = NULL; struct mpage_readpage_args args = {
.get_block = get_block,
.is_readahead = true,
};
unsigned page_idx; unsigned page_idx;
sector_t last_block_in_bio = 0;
struct buffer_head map_bh;
unsigned long first_logical_block = 0;
gfp_t gfp = readahead_gfp_mask(mapping);
map_bh.b_state = 0;
map_bh.b_size = 0;
for (page_idx = 0; page_idx < nr_pages; page_idx++) { for (page_idx = 0; page_idx < nr_pages; page_idx++) {
struct page *page = lru_to_page(pages); struct page *page = lru_to_page(pages);
@ -379,18 +398,16 @@ mpage_readpages(struct address_space *mapping, struct list_head *pages,
list_del(&page->lru); list_del(&page->lru);
if (!add_to_page_cache_lru(page, mapping, if (!add_to_page_cache_lru(page, mapping,
page->index, page->index,
gfp)) { readahead_gfp_mask(mapping))) {
bio = do_mpage_readpage(bio, page, args.page = page;
nr_pages - page_idx, args.nr_pages = nr_pages - page_idx;
&last_block_in_bio, &map_bh, args.bio = do_mpage_readpage(&args);
&first_logical_block,
get_block, gfp);
} }
put_page(page); put_page(page);
} }
BUG_ON(!list_empty(pages)); BUG_ON(!list_empty(pages));
if (bio) if (args.bio)
mpage_bio_submit(REQ_OP_READ, 0, bio); mpage_bio_submit(REQ_OP_READ, REQ_RAHEAD, args.bio);
return 0; return 0;
} }
EXPORT_SYMBOL(mpage_readpages); EXPORT_SYMBOL(mpage_readpages);
@ -400,18 +417,15 @@ EXPORT_SYMBOL(mpage_readpages);
*/ */
int mpage_readpage(struct page *page, get_block_t get_block) int mpage_readpage(struct page *page, get_block_t get_block)
{ {
struct bio *bio = NULL; struct mpage_readpage_args args = {
sector_t last_block_in_bio = 0; .page = page,
struct buffer_head map_bh; .nr_pages = 1,
unsigned long first_logical_block = 0; .get_block = get_block,
gfp_t gfp = mapping_gfp_constraint(page->mapping, GFP_KERNEL); };
map_bh.b_state = 0; args.bio = do_mpage_readpage(&args);
map_bh.b_size = 0; if (args.bio)
bio = do_mpage_readpage(bio, page, 1, &last_block_in_bio, mpage_bio_submit(REQ_OP_READ, 0, args.bio);
&map_bh, &first_logical_block, get_block, gfp);
if (bio)
mpage_bio_submit(REQ_OP_READ, 0, bio);
return 0; return 0;
} }
EXPORT_SYMBOL(mpage_readpage); EXPORT_SYMBOL(mpage_readpage);

View File

@ -384,8 +384,9 @@ out_err:
static int __init dnotify_init(void) static int __init dnotify_init(void)
{ {
dnotify_struct_cache = KMEM_CACHE(dnotify_struct, SLAB_PANIC); dnotify_struct_cache = KMEM_CACHE(dnotify_struct,
dnotify_mark_cache = KMEM_CACHE(dnotify_mark, SLAB_PANIC); SLAB_PANIC|SLAB_ACCOUNT);
dnotify_mark_cache = KMEM_CACHE(dnotify_mark, SLAB_PANIC|SLAB_ACCOUNT);
dnotify_group = fsnotify_alloc_group(&dnotify_fsnotify_ops); dnotify_group = fsnotify_alloc_group(&dnotify_fsnotify_ops);
if (IS_ERR(dnotify_group)) if (IS_ERR(dnotify_group))

View File

@ -11,6 +11,7 @@
#include <linux/types.h> #include <linux/types.h>
#include <linux/wait.h> #include <linux/wait.h>
#include <linux/audit.h> #include <linux/audit.h>
#include <linux/sched/mm.h>
#include "fanotify.h" #include "fanotify.h"
@ -140,8 +141,8 @@ struct fanotify_event_info *fanotify_alloc_event(struct fsnotify_group *group,
struct inode *inode, u32 mask, struct inode *inode, u32 mask,
const struct path *path) const struct path *path)
{ {
struct fanotify_event_info *event; struct fanotify_event_info *event = NULL;
gfp_t gfp = GFP_KERNEL; gfp_t gfp = GFP_KERNEL_ACCOUNT;
/* /*
* For queues with unlimited length lost events are not expected and * For queues with unlimited length lost events are not expected and
@ -151,19 +152,22 @@ struct fanotify_event_info *fanotify_alloc_event(struct fsnotify_group *group,
if (group->max_events == UINT_MAX) if (group->max_events == UINT_MAX)
gfp |= __GFP_NOFAIL; gfp |= __GFP_NOFAIL;
/* Whoever is interested in the event, pays for the allocation. */
memalloc_use_memcg(group->memcg);
if (fanotify_is_perm_event(mask)) { if (fanotify_is_perm_event(mask)) {
struct fanotify_perm_event_info *pevent; struct fanotify_perm_event_info *pevent;
pevent = kmem_cache_alloc(fanotify_perm_event_cachep, gfp); pevent = kmem_cache_alloc(fanotify_perm_event_cachep, gfp);
if (!pevent) if (!pevent)
return NULL; goto out;
event = &pevent->fae; event = &pevent->fae;
pevent->response = 0; pevent->response = 0;
goto init; goto init;
} }
event = kmem_cache_alloc(fanotify_event_cachep, gfp); event = kmem_cache_alloc(fanotify_event_cachep, gfp);
if (!event) if (!event)
return NULL; goto out;
init: __maybe_unused init: __maybe_unused
fsnotify_init_event(&event->fse, inode, mask); fsnotify_init_event(&event->fse, inode, mask);
event->tgid = get_pid(task_tgid(current)); event->tgid = get_pid(task_tgid(current));
@ -174,6 +178,8 @@ init: __maybe_unused
event->path.mnt = NULL; event->path.mnt = NULL;
event->path.dentry = NULL; event->path.dentry = NULL;
} }
out:
memalloc_unuse_memcg();
return event; return event;
} }

View File

@ -16,6 +16,7 @@
#include <linux/uaccess.h> #include <linux/uaccess.h>
#include <linux/compat.h> #include <linux/compat.h>
#include <linux/sched/signal.h> #include <linux/sched/signal.h>
#include <linux/memcontrol.h>
#include <asm/ioctls.h> #include <asm/ioctls.h>
@ -731,6 +732,7 @@ SYSCALL_DEFINE2(fanotify_init, unsigned int, flags, unsigned int, event_f_flags)
group->fanotify_data.user = user; group->fanotify_data.user = user;
atomic_inc(&user->fanotify_listeners); atomic_inc(&user->fanotify_listeners);
group->memcg = get_mem_cgroup_from_mm(current->mm);
oevent = fanotify_alloc_event(group, NULL, FS_Q_OVERFLOW, NULL); oevent = fanotify_alloc_event(group, NULL, FS_Q_OVERFLOW, NULL);
if (unlikely(!oevent)) { if (unlikely(!oevent)) {
@ -932,7 +934,8 @@ COMPAT_SYSCALL_DEFINE6(fanotify_mark,
*/ */
static int __init fanotify_user_setup(void) static int __init fanotify_user_setup(void)
{ {
fanotify_mark_cache = KMEM_CACHE(fsnotify_mark, SLAB_PANIC); fanotify_mark_cache = KMEM_CACHE(fsnotify_mark,
SLAB_PANIC|SLAB_ACCOUNT);
fanotify_event_cachep = KMEM_CACHE(fanotify_event_info, SLAB_PANIC); fanotify_event_cachep = KMEM_CACHE(fanotify_event_info, SLAB_PANIC);
if (IS_ENABLED(CONFIG_FANOTIFY_ACCESS_PERMISSIONS)) { if (IS_ENABLED(CONFIG_FANOTIFY_ACCESS_PERMISSIONS)) {
fanotify_perm_event_cachep = fanotify_perm_event_cachep =

View File

@ -22,6 +22,7 @@
#include <linux/srcu.h> #include <linux/srcu.h>
#include <linux/rculist.h> #include <linux/rculist.h>
#include <linux/wait.h> #include <linux/wait.h>
#include <linux/memcontrol.h>
#include <linux/fsnotify_backend.h> #include <linux/fsnotify_backend.h>
#include "fsnotify.h" #include "fsnotify.h"
@ -36,6 +37,8 @@ static void fsnotify_final_destroy_group(struct fsnotify_group *group)
if (group->ops->free_group_priv) if (group->ops->free_group_priv)
group->ops->free_group_priv(group); group->ops->free_group_priv(group);
mem_cgroup_put(group->memcg);
kfree(group); kfree(group);
} }

View File

@ -31,6 +31,7 @@
#include <linux/types.h> #include <linux/types.h>
#include <linux/sched.h> #include <linux/sched.h>
#include <linux/sched/user.h> #include <linux/sched/user.h>
#include <linux/sched/mm.h>
#include "inotify.h" #include "inotify.h"
@ -98,7 +99,11 @@ int inotify_handle_event(struct fsnotify_group *group,
i_mark = container_of(inode_mark, struct inotify_inode_mark, i_mark = container_of(inode_mark, struct inotify_inode_mark,
fsn_mark); fsn_mark);
event = kmalloc(alloc_len, GFP_KERNEL); /* Whoever is interested in the event, pays for the allocation. */
memalloc_use_memcg(group->memcg);
event = kmalloc(alloc_len, GFP_KERNEL_ACCOUNT);
memalloc_unuse_memcg();
if (unlikely(!event)) { if (unlikely(!event)) {
/* /*
* Treat lost event due to ENOMEM the same way as queue * Treat lost event due to ENOMEM the same way as queue

View File

@ -38,6 +38,7 @@
#include <linux/uaccess.h> #include <linux/uaccess.h>
#include <linux/poll.h> #include <linux/poll.h>
#include <linux/wait.h> #include <linux/wait.h>
#include <linux/memcontrol.h>
#include "inotify.h" #include "inotify.h"
#include "../fdinfo.h" #include "../fdinfo.h"
@ -639,6 +640,7 @@ static struct fsnotify_group *inotify_new_group(unsigned int max_events)
oevent->name_len = 0; oevent->name_len = 0;
group->max_events = max_events; group->max_events = max_events;
group->memcg = get_mem_cgroup_from_mm(current->mm);
spin_lock_init(&group->inotify_data.idr_lock); spin_lock_init(&group->inotify_data.idr_lock);
idr_init(&group->inotify_data.idr); idr_init(&group->inotify_data.idr);
@ -815,7 +817,8 @@ static int __init inotify_user_setup(void)
BUG_ON(hweight32(ALL_INOTIFY_BITS) != 22); BUG_ON(hweight32(ALL_INOTIFY_BITS) != 22);
inotify_inode_mark_cachep = KMEM_CACHE(inotify_inode_mark, SLAB_PANIC); inotify_inode_mark_cachep = KMEM_CACHE(inotify_inode_mark,
SLAB_PANIC|SLAB_ACCOUNT);
inotify_max_queued_events = 16384; inotify_max_queued_events = 16384;
init_user_ns.ucount_max[UCOUNT_INOTIFY_INSTANCES] = 128; init_user_ns.ucount_max[UCOUNT_INOTIFY_INSTANCES] = 128;

View File

@ -93,13 +93,11 @@ static void ntfs_end_buffer_async_read(struct buffer_head *bh, int uptodate)
ofs = 0; ofs = 0;
if (file_ofs < init_size) if (file_ofs < init_size)
ofs = init_size - file_ofs; ofs = init_size - file_ofs;
local_irq_save(flags);
kaddr = kmap_atomic(page); kaddr = kmap_atomic(page);
memset(kaddr + bh_offset(bh) + ofs, 0, memset(kaddr + bh_offset(bh) + ofs, 0,
bh->b_size - ofs); bh->b_size - ofs);
flush_dcache_page(page); flush_dcache_page(page);
kunmap_atomic(kaddr); kunmap_atomic(kaddr);
local_irq_restore(flags);
} }
} else { } else {
clear_buffer_uptodate(bh); clear_buffer_uptodate(bh);
@ -146,13 +144,11 @@ static void ntfs_end_buffer_async_read(struct buffer_head *bh, int uptodate)
recs = PAGE_SIZE / rec_size; recs = PAGE_SIZE / rec_size;
/* Should have been verified before we got here... */ /* Should have been verified before we got here... */
BUG_ON(!recs); BUG_ON(!recs);
local_irq_save(flags);
kaddr = kmap_atomic(page); kaddr = kmap_atomic(page);
for (i = 0; i < recs; i++) for (i = 0; i < recs; i++)
post_read_mst_fixup((NTFS_RECORD*)(kaddr + post_read_mst_fixup((NTFS_RECORD*)(kaddr +
i * rec_size), rec_size); i * rec_size), rec_size);
kunmap_atomic(kaddr); kunmap_atomic(kaddr);
local_irq_restore(flags);
flush_dcache_page(page); flush_dcache_page(page);
if (likely(page_uptodate && !PageError(page))) if (likely(page_uptodate && !PageError(page)))
SetPageUptodate(page); SetPageUptodate(page);
@ -926,7 +922,7 @@ static int ntfs_write_mst_block(struct page *page,
ntfs_volume *vol = ni->vol; ntfs_volume *vol = ni->vol;
u8 *kaddr; u8 *kaddr;
unsigned int rec_size = ni->itype.index.block_size; unsigned int rec_size = ni->itype.index.block_size;
ntfs_inode *locked_nis[PAGE_SIZE / rec_size]; ntfs_inode *locked_nis[PAGE_SIZE / NTFS_BLOCK_SIZE];
struct buffer_head *bh, *head, *tbh, *rec_start_bh; struct buffer_head *bh, *head, *tbh, *rec_start_bh;
struct buffer_head *bhs[MAX_BUF_PER_PAGE]; struct buffer_head *bhs[MAX_BUF_PER_PAGE];
runlist_element *rl; runlist_element *rl;
@ -935,6 +931,9 @@ static int ntfs_write_mst_block(struct page *page,
bool sync, is_mft, page_is_dirty, rec_is_dirty; bool sync, is_mft, page_is_dirty, rec_is_dirty;
unsigned char bh_size_bits; unsigned char bh_size_bits;
if (WARN_ON(rec_size < NTFS_BLOCK_SIZE))
return -EINVAL;
ntfs_debug("Entering for inode 0x%lx, attribute type 0x%x, page index " ntfs_debug("Entering for inode 0x%lx, attribute type 0x%x, page index "
"0x%lx.", vi->i_ino, ni->type, page->index); "0x%lx.", vi->i_ino, ni->type, page->index);
BUG_ON(!NInoNonResident(ni)); BUG_ON(!NInoNonResident(ni));

View File

@ -128,6 +128,7 @@ static inline void handle_bounds_compressed_page(struct page *page,
/** /**
* ntfs_decompress - decompress a compression block into an array of pages * ntfs_decompress - decompress a compression block into an array of pages
* @dest_pages: destination array of pages * @dest_pages: destination array of pages
* @completed_pages: scratch space to track completed pages
* @dest_index: current index into @dest_pages (IN/OUT) * @dest_index: current index into @dest_pages (IN/OUT)
* @dest_ofs: current offset within @dest_pages[@dest_index] (IN/OUT) * @dest_ofs: current offset within @dest_pages[@dest_index] (IN/OUT)
* @dest_max_index: maximum index into @dest_pages (IN) * @dest_max_index: maximum index into @dest_pages (IN)
@ -162,10 +163,10 @@ static inline void handle_bounds_compressed_page(struct page *page,
* Note to hackers: This function may not sleep until it has finished accessing * Note to hackers: This function may not sleep until it has finished accessing
* the compression block @cb_start as it is a per-CPU buffer. * the compression block @cb_start as it is a per-CPU buffer.
*/ */
static int ntfs_decompress(struct page *dest_pages[], int *dest_index, static int ntfs_decompress(struct page *dest_pages[], int completed_pages[],
int *dest_ofs, const int dest_max_index, const int dest_max_ofs, int *dest_index, int *dest_ofs, const int dest_max_index,
const int xpage, char *xpage_done, u8 *const cb_start, const int dest_max_ofs, const int xpage, char *xpage_done,
const u32 cb_size, const loff_t i_size, u8 *const cb_start, const u32 cb_size, const loff_t i_size,
const s64 initialized_size) const s64 initialized_size)
{ {
/* /*
@ -190,9 +191,6 @@ static int ntfs_decompress(struct page *dest_pages[], int *dest_index,
/* Variables for tag and token parsing. */ /* Variables for tag and token parsing. */
u8 tag; /* Current tag. */ u8 tag; /* Current tag. */
int token; /* Loop counter for the eight tokens in tag. */ int token; /* Loop counter for the eight tokens in tag. */
/* Need this because we can't sleep, so need two stages. */
int completed_pages[dest_max_index - *dest_index + 1];
int nr_completed_pages = 0; int nr_completed_pages = 0;
/* Default error code. */ /* Default error code. */
@ -516,6 +514,7 @@ int ntfs_read_compressed_block(struct page *page)
unsigned int cb_clusters, cb_max_ofs; unsigned int cb_clusters, cb_max_ofs;
int block, max_block, cb_max_page, bhs_size, nr_bhs, err = 0; int block, max_block, cb_max_page, bhs_size, nr_bhs, err = 0;
struct page **pages; struct page **pages;
int *completed_pages;
unsigned char xpage_done = 0; unsigned char xpage_done = 0;
ntfs_debug("Entering, page->index = 0x%lx, cb_size = 0x%x, nr_pages = " ntfs_debug("Entering, page->index = 0x%lx, cb_size = 0x%x, nr_pages = "
@ -528,14 +527,16 @@ int ntfs_read_compressed_block(struct page *page)
BUG_ON(ni->name_len); BUG_ON(ni->name_len);
pages = kmalloc_array(nr_pages, sizeof(struct page *), GFP_NOFS); pages = kmalloc_array(nr_pages, sizeof(struct page *), GFP_NOFS);
completed_pages = kmalloc_array(nr_pages + 1, sizeof(int), GFP_NOFS);
/* Allocate memory to store the buffer heads we need. */ /* Allocate memory to store the buffer heads we need. */
bhs_size = cb_size / block_size * sizeof(struct buffer_head *); bhs_size = cb_size / block_size * sizeof(struct buffer_head *);
bhs = kmalloc(bhs_size, GFP_NOFS); bhs = kmalloc(bhs_size, GFP_NOFS);
if (unlikely(!pages || !bhs)) { if (unlikely(!pages || !bhs || !completed_pages)) {
kfree(bhs); kfree(bhs);
kfree(pages); kfree(pages);
kfree(completed_pages);
unlock_page(page); unlock_page(page);
ntfs_error(vol->sb, "Failed to allocate internal buffers."); ntfs_error(vol->sb, "Failed to allocate internal buffers.");
return -ENOMEM; return -ENOMEM;
@ -562,6 +563,7 @@ int ntfs_read_compressed_block(struct page *page)
if (xpage >= max_page) { if (xpage >= max_page) {
kfree(bhs); kfree(bhs);
kfree(pages); kfree(pages);
kfree(completed_pages);
zero_user(page, 0, PAGE_SIZE); zero_user(page, 0, PAGE_SIZE);
ntfs_debug("Compressed read outside i_size - truncated?"); ntfs_debug("Compressed read outside i_size - truncated?");
SetPageUptodate(page); SetPageUptodate(page);
@ -854,10 +856,10 @@ lock_retry_remap:
unsigned int prev_cur_page = cur_page; unsigned int prev_cur_page = cur_page;
ntfs_debug("Found compressed compression block."); ntfs_debug("Found compressed compression block.");
err = ntfs_decompress(pages, &cur_page, &cur_ofs, err = ntfs_decompress(pages, completed_pages, &cur_page,
cb_max_page, cb_max_ofs, xpage, &xpage_done, &cur_ofs, cb_max_page, cb_max_ofs, xpage,
cb_pos, cb_size - (cb_pos - cb), i_size, &xpage_done, cb_pos, cb_size - (cb_pos - cb),
initialized_size); i_size, initialized_size);
/* /*
* We can sleep from now on, lock already dropped by * We can sleep from now on, lock already dropped by
* ntfs_decompress(). * ntfs_decompress().
@ -912,6 +914,7 @@ lock_retry_remap:
/* We no longer need the list of pages. */ /* We no longer need the list of pages. */
kfree(pages); kfree(pages);
kfree(completed_pages);
/* If we have completed the requested page, we return success. */ /* If we have completed the requested page, we return success. */
if (likely(xpage_done)) if (likely(xpage_done))
@ -956,5 +959,6 @@ err_out:
} }
} }
kfree(pages); kfree(pages);
kfree(completed_pages);
return -EIO; return -EIO;
} }

View File

@ -667,18 +667,18 @@ static int ntfs_read_locked_inode(struct inode *vi)
* mtime is the last change of the data within the file. Not changed * mtime is the last change of the data within the file. Not changed
* when only metadata is changed, e.g. a rename doesn't affect mtime. * when only metadata is changed, e.g. a rename doesn't affect mtime.
*/ */
vi->i_mtime = timespec_to_timespec64(ntfs2utc(si->last_data_change_time)); vi->i_mtime = ntfs2utc(si->last_data_change_time);
/* /*
* ctime is the last change of the metadata of the file. This obviously * ctime is the last change of the metadata of the file. This obviously
* always changes, when mtime is changed. ctime can be changed on its * always changes, when mtime is changed. ctime can be changed on its
* own, mtime is then not changed, e.g. when a file is renamed. * own, mtime is then not changed, e.g. when a file is renamed.
*/ */
vi->i_ctime = timespec_to_timespec64(ntfs2utc(si->last_mft_change_time)); vi->i_ctime = ntfs2utc(si->last_mft_change_time);
/* /*
* Last access to the data within the file. Not changed during a rename * Last access to the data within the file. Not changed during a rename
* for example but changed whenever the file is written to. * for example but changed whenever the file is written to.
*/ */
vi->i_atime = timespec_to_timespec64(ntfs2utc(si->last_access_time)); vi->i_atime = ntfs2utc(si->last_access_time);
/* Find the attribute list attribute if present. */ /* Find the attribute list attribute if present. */
ntfs_attr_reinit_search_ctx(ctx); ntfs_attr_reinit_search_ctx(ctx);
@ -2997,7 +2997,7 @@ int __ntfs_write_inode(struct inode *vi, int sync)
si = (STANDARD_INFORMATION*)((u8*)ctx->attr + si = (STANDARD_INFORMATION*)((u8*)ctx->attr +
le16_to_cpu(ctx->attr->data.resident.value_offset)); le16_to_cpu(ctx->attr->data.resident.value_offset));
/* Update the access times if they have changed. */ /* Update the access times if they have changed. */
nt = utc2ntfs(timespec64_to_timespec(vi->i_mtime)); nt = utc2ntfs(vi->i_mtime);
if (si->last_data_change_time != nt) { if (si->last_data_change_time != nt) {
ntfs_debug("Updating mtime for inode 0x%lx: old = 0x%llx, " ntfs_debug("Updating mtime for inode 0x%lx: old = 0x%llx, "
"new = 0x%llx", vi->i_ino, (long long) "new = 0x%llx", vi->i_ino, (long long)
@ -3006,7 +3006,7 @@ int __ntfs_write_inode(struct inode *vi, int sync)
si->last_data_change_time = nt; si->last_data_change_time = nt;
modified = true; modified = true;
} }
nt = utc2ntfs(timespec64_to_timespec(vi->i_ctime)); nt = utc2ntfs(vi->i_ctime);
if (si->last_mft_change_time != nt) { if (si->last_mft_change_time != nt) {
ntfs_debug("Updating ctime for inode 0x%lx: old = 0x%llx, " ntfs_debug("Updating ctime for inode 0x%lx: old = 0x%llx, "
"new = 0x%llx", vi->i_ino, (long long) "new = 0x%llx", vi->i_ino, (long long)
@ -3015,7 +3015,7 @@ int __ntfs_write_inode(struct inode *vi, int sync)
si->last_mft_change_time = nt; si->last_mft_change_time = nt;
modified = true; modified = true;
} }
nt = utc2ntfs(timespec64_to_timespec(vi->i_atime)); nt = utc2ntfs(vi->i_atime);
if (si->last_access_time != nt) { if (si->last_access_time != nt) {
ntfs_debug("Updating atime for inode 0x%lx: old = 0x%llx, " ntfs_debug("Updating atime for inode 0x%lx: old = 0x%llx, "
"new = 0x%llx", vi->i_ino, "new = 0x%llx", vi->i_ino,

View File

@ -35,6 +35,8 @@
#include "mft.h" #include "mft.h"
#include "ntfs.h" #include "ntfs.h"
#define MAX_BHS (PAGE_SIZE / NTFS_BLOCK_SIZE)
/** /**
* map_mft_record_page - map the page in which a specific mft record resides * map_mft_record_page - map the page in which a specific mft record resides
* @ni: ntfs inode whose mft record page to map * @ni: ntfs inode whose mft record page to map
@ -469,7 +471,7 @@ int ntfs_sync_mft_mirror(ntfs_volume *vol, const unsigned long mft_no,
struct page *page; struct page *page;
unsigned int blocksize = vol->sb->s_blocksize; unsigned int blocksize = vol->sb->s_blocksize;
int max_bhs = vol->mft_record_size / blocksize; int max_bhs = vol->mft_record_size / blocksize;
struct buffer_head *bhs[max_bhs]; struct buffer_head *bhs[MAX_BHS];
struct buffer_head *bh, *head; struct buffer_head *bh, *head;
u8 *kmirr; u8 *kmirr;
runlist_element *rl; runlist_element *rl;
@ -479,6 +481,8 @@ int ntfs_sync_mft_mirror(ntfs_volume *vol, const unsigned long mft_no,
ntfs_debug("Entering for inode 0x%lx.", mft_no); ntfs_debug("Entering for inode 0x%lx.", mft_no);
BUG_ON(!max_bhs); BUG_ON(!max_bhs);
if (WARN_ON(max_bhs > MAX_BHS))
return -EINVAL;
if (unlikely(!vol->mftmirr_ino)) { if (unlikely(!vol->mftmirr_ino)) {
/* This could happen during umount... */ /* This could happen during umount... */
err = ntfs_sync_mft_mirror_umount(vol, mft_no, m); err = ntfs_sync_mft_mirror_umount(vol, mft_no, m);
@ -674,7 +678,7 @@ int write_mft_record_nolock(ntfs_inode *ni, MFT_RECORD *m, int sync)
unsigned int blocksize = vol->sb->s_blocksize; unsigned int blocksize = vol->sb->s_blocksize;
unsigned char blocksize_bits = vol->sb->s_blocksize_bits; unsigned char blocksize_bits = vol->sb->s_blocksize_bits;
int max_bhs = vol->mft_record_size / blocksize; int max_bhs = vol->mft_record_size / blocksize;
struct buffer_head *bhs[max_bhs]; struct buffer_head *bhs[MAX_BHS];
struct buffer_head *bh, *head; struct buffer_head *bh, *head;
runlist_element *rl; runlist_element *rl;
unsigned int block_start, block_end, m_start, m_end; unsigned int block_start, block_end, m_start, m_end;
@ -684,6 +688,10 @@ int write_mft_record_nolock(ntfs_inode *ni, MFT_RECORD *m, int sync)
BUG_ON(NInoAttr(ni)); BUG_ON(NInoAttr(ni));
BUG_ON(!max_bhs); BUG_ON(!max_bhs);
BUG_ON(!PageLocked(page)); BUG_ON(!PageLocked(page));
if (WARN_ON(max_bhs > MAX_BHS)) {
err = -EINVAL;
goto err_out;
}
/* /*
* If the ntfs_inode is clean no need to do anything. If it is dirty, * If the ntfs_inode is clean no need to do anything. If it is dirty,
* mark it as clean now so that it can be redirtied later on if needed. * mark it as clean now so that it can be redirtied later on if needed.

View File

@ -36,16 +36,16 @@
* Convert the Linux UTC time @ts to its corresponding NTFS time and return * Convert the Linux UTC time @ts to its corresponding NTFS time and return
* that in little endian format. * that in little endian format.
* *
* Linux stores time in a struct timespec consisting of a time_t (long at * Linux stores time in a struct timespec64 consisting of a time64_t tv_sec
* present) tv_sec and a long tv_nsec where tv_sec is the number of 1-second * and a long tv_nsec where tv_sec is the number of 1-second intervals since
* intervals since 1st January 1970, 00:00:00 UTC and tv_nsec is the number of * 1st January 1970, 00:00:00 UTC and tv_nsec is the number of 1-nano-second
* 1-nano-second intervals since the value of tv_sec. * intervals since the value of tv_sec.
* *
* NTFS uses Microsoft's standard time format which is stored in a s64 and is * NTFS uses Microsoft's standard time format which is stored in a s64 and is
* measured as the number of 100-nano-second intervals since 1st January 1601, * measured as the number of 100-nano-second intervals since 1st January 1601,
* 00:00:00 UTC. * 00:00:00 UTC.
*/ */
static inline sle64 utc2ntfs(const struct timespec ts) static inline sle64 utc2ntfs(const struct timespec64 ts)
{ {
/* /*
* Convert the seconds to 100ns intervals, add the nano-seconds * Convert the seconds to 100ns intervals, add the nano-seconds
@ -63,7 +63,10 @@ static inline sle64 utc2ntfs(const struct timespec ts)
*/ */
static inline sle64 get_current_ntfs_time(void) static inline sle64 get_current_ntfs_time(void)
{ {
return utc2ntfs(current_kernel_time()); struct timespec64 ts;
ktime_get_coarse_real_ts64(&ts);
return utc2ntfs(ts);
} }
/** /**
@ -73,18 +76,18 @@ static inline sle64 get_current_ntfs_time(void)
* Convert the little endian NTFS time @time to its corresponding Linux UTC * Convert the little endian NTFS time @time to its corresponding Linux UTC
* time and return that in cpu format. * time and return that in cpu format.
* *
* Linux stores time in a struct timespec consisting of a time_t (long at * Linux stores time in a struct timespec64 consisting of a time64_t tv_sec
* present) tv_sec and a long tv_nsec where tv_sec is the number of 1-second * and a long tv_nsec where tv_sec is the number of 1-second intervals since
* intervals since 1st January 1970, 00:00:00 UTC and tv_nsec is the number of * 1st January 1970, 00:00:00 UTC and tv_nsec is the number of 1-nano-second
* 1-nano-second intervals since the value of tv_sec. * intervals since the value of tv_sec.
* *
* NTFS uses Microsoft's standard time format which is stored in a s64 and is * NTFS uses Microsoft's standard time format which is stored in a s64 and is
* measured as the number of 100 nano-second intervals since 1st January 1601, * measured as the number of 100 nano-second intervals since 1st January 1601,
* 00:00:00 UTC. * 00:00:00 UTC.
*/ */
static inline struct timespec ntfs2utc(const sle64 time) static inline struct timespec64 ntfs2utc(const sle64 time)
{ {
struct timespec ts; struct timespec64 ts;
/* Subtract the NTFS time offset. */ /* Subtract the NTFS time offset. */
u64 t = (u64)(sle64_to_cpu(time) - NTFS_TIME_OFFSET); u64 t = (u64)(sle64_to_cpu(time) - NTFS_TIME_OFFSET);

View File

@ -932,13 +932,11 @@ static int ocfs2_validate_extent_block(struct super_block *sb,
goto bail; goto bail;
} }
if (le32_to_cpu(eb->h_fs_generation) != OCFS2_SB(sb)->fs_generation) { if (le32_to_cpu(eb->h_fs_generation) != OCFS2_SB(sb)->fs_generation)
rc = ocfs2_error(sb, rc = ocfs2_error(sb,
"Extent block #%llu has an invalid h_fs_generation of #%u\n", "Extent block #%llu has an invalid h_fs_generation of #%u\n",
(unsigned long long)bh->b_blocknr, (unsigned long long)bh->b_blocknr,
le32_to_cpu(eb->h_fs_generation)); le32_to_cpu(eb->h_fs_generation));
goto bail;
}
bail: bail:
return rc; return rc;
} }
@ -1481,19 +1479,17 @@ static int ocfs2_find_branch_target(struct ocfs2_extent_tree *et,
while(le16_to_cpu(el->l_tree_depth) > 1) { while(le16_to_cpu(el->l_tree_depth) > 1) {
if (le16_to_cpu(el->l_next_free_rec) == 0) { if (le16_to_cpu(el->l_next_free_rec) == 0) {
ocfs2_error(ocfs2_metadata_cache_get_super(et->et_ci), status = ocfs2_error(ocfs2_metadata_cache_get_super(et->et_ci),
"Owner %llu has empty extent list (next_free_rec == 0)\n", "Owner %llu has empty extent list (next_free_rec == 0)\n",
(unsigned long long)ocfs2_metadata_cache_owner(et->et_ci)); (unsigned long long)ocfs2_metadata_cache_owner(et->et_ci));
status = -EIO;
goto bail; goto bail;
} }
i = le16_to_cpu(el->l_next_free_rec) - 1; i = le16_to_cpu(el->l_next_free_rec) - 1;
blkno = le64_to_cpu(el->l_recs[i].e_blkno); blkno = le64_to_cpu(el->l_recs[i].e_blkno);
if (!blkno) { if (!blkno) {
ocfs2_error(ocfs2_metadata_cache_get_super(et->et_ci), status = ocfs2_error(ocfs2_metadata_cache_get_super(et->et_ci),
"Owner %llu has extent list where extent # %d has no physical block start\n", "Owner %llu has extent list where extent # %d has no physical block start\n",
(unsigned long long)ocfs2_metadata_cache_owner(et->et_ci), i); (unsigned long long)ocfs2_metadata_cache_owner(et->et_ci), i);
status = -EIO;
goto bail; goto bail;
} }
@ -1598,10 +1594,8 @@ static int ocfs2_grow_tree(handle_t *handle, struct ocfs2_extent_tree *et,
* the new data. */ * the new data. */
ret = ocfs2_add_branch(handle, et, bh, last_eb_bh, ret = ocfs2_add_branch(handle, et, bh, last_eb_bh,
meta_ac); meta_ac);
if (ret < 0) { if (ret < 0)
mlog_errno(ret); mlog_errno(ret);
goto out;
}
out: out:
if (final_depth) if (final_depth)
@ -3214,11 +3208,10 @@ rightmost_no_delete:
goto rightmost_no_delete; goto rightmost_no_delete;
if (le16_to_cpu(el->l_next_free_rec) == 0) { if (le16_to_cpu(el->l_next_free_rec) == 0) {
ret = -EIO; ret = ocfs2_error(ocfs2_metadata_cache_get_super(et->et_ci),
ocfs2_error(ocfs2_metadata_cache_get_super(et->et_ci), "Owner %llu has empty extent block at %llu\n",
"Owner %llu has empty extent block at %llu\n", (unsigned long long)ocfs2_metadata_cache_owner(et->et_ci),
(unsigned long long)ocfs2_metadata_cache_owner(et->et_ci), (unsigned long long)le64_to_cpu(eb->h_blkno));
(unsigned long long)le64_to_cpu(eb->h_blkno));
goto out; goto out;
} }
@ -4411,12 +4404,11 @@ static int ocfs2_figure_merge_contig_type(struct ocfs2_extent_tree *et,
le16_to_cpu(new_el->l_count)) { le16_to_cpu(new_el->l_count)) {
bh = path_leaf_bh(left_path); bh = path_leaf_bh(left_path);
eb = (struct ocfs2_extent_block *)bh->b_data; eb = (struct ocfs2_extent_block *)bh->b_data;
ocfs2_error(sb, status = ocfs2_error(sb,
"Extent block #%llu has an invalid l_next_free_rec of %d. It should have matched the l_count of %d\n", "Extent block #%llu has an invalid l_next_free_rec of %d. It should have matched the l_count of %d\n",
(unsigned long long)le64_to_cpu(eb->h_blkno), (unsigned long long)le64_to_cpu(eb->h_blkno),
le16_to_cpu(new_el->l_next_free_rec), le16_to_cpu(new_el->l_next_free_rec),
le16_to_cpu(new_el->l_count)); le16_to_cpu(new_el->l_count));
status = -EINVAL;
goto free_left_path; goto free_left_path;
} }
rec = &new_el->l_recs[ rec = &new_el->l_recs[
@ -4466,11 +4458,10 @@ static int ocfs2_figure_merge_contig_type(struct ocfs2_extent_tree *et,
if (le16_to_cpu(new_el->l_next_free_rec) <= 1) { if (le16_to_cpu(new_el->l_next_free_rec) <= 1) {
bh = path_leaf_bh(right_path); bh = path_leaf_bh(right_path);
eb = (struct ocfs2_extent_block *)bh->b_data; eb = (struct ocfs2_extent_block *)bh->b_data;
ocfs2_error(sb, status = ocfs2_error(sb,
"Extent block #%llu has an invalid l_next_free_rec of %d\n", "Extent block #%llu has an invalid l_next_free_rec of %d\n",
(unsigned long long)le64_to_cpu(eb->h_blkno), (unsigned long long)le64_to_cpu(eb->h_blkno),
le16_to_cpu(new_el->l_next_free_rec)); le16_to_cpu(new_el->l_next_free_rec));
status = -EINVAL;
goto free_right_path; goto free_right_path;
} }
rec = &new_el->l_recs[1]; rec = &new_el->l_recs[1];
@ -5523,10 +5514,8 @@ static int ocfs2_truncate_rec(handle_t *handle,
ocfs2_journal_dirty(handle, path_leaf_bh(path)); ocfs2_journal_dirty(handle, path_leaf_bh(path));
ret = ocfs2_rotate_tree_left(handle, et, path, dealloc); ret = ocfs2_rotate_tree_left(handle, et, path, dealloc);
if (ret) { if (ret)
mlog_errno(ret); mlog_errno(ret);
goto out;
}
out: out:
ocfs2_free_path(left_path); ocfs2_free_path(left_path);
@ -5659,10 +5648,8 @@ int ocfs2_remove_extent(handle_t *handle,
ret = ocfs2_truncate_rec(handle, et, path, index, dealloc, ret = ocfs2_truncate_rec(handle, et, path, index, dealloc,
cpos, len); cpos, len);
if (ret) { if (ret)
mlog_errno(ret); mlog_errno(ret);
goto out;
}
} }
out: out:
@ -5707,7 +5694,6 @@ static int ocfs2_reserve_blocks_for_rec_trunc(struct inode *inode,
if (ret < 0) { if (ret < 0) {
if (ret != -ENOSPC) if (ret != -ENOSPC)
mlog_errno(ret); mlog_errno(ret);
goto out;
} }
} }

View File

@ -127,13 +127,13 @@ enum o2hb_heartbeat_modes {
O2HB_HEARTBEAT_NUM_MODES, O2HB_HEARTBEAT_NUM_MODES,
}; };
char *o2hb_heartbeat_mode_desc[O2HB_HEARTBEAT_NUM_MODES] = { static const char *o2hb_heartbeat_mode_desc[O2HB_HEARTBEAT_NUM_MODES] = {
"local", /* O2HB_HEARTBEAT_LOCAL */ "local", /* O2HB_HEARTBEAT_LOCAL */
"global", /* O2HB_HEARTBEAT_GLOBAL */ "global", /* O2HB_HEARTBEAT_GLOBAL */
}; };
unsigned int o2hb_dead_threshold = O2HB_DEFAULT_DEAD_THRESHOLD; unsigned int o2hb_dead_threshold = O2HB_DEFAULT_DEAD_THRESHOLD;
unsigned int o2hb_heartbeat_mode = O2HB_HEARTBEAT_LOCAL; static unsigned int o2hb_heartbeat_mode = O2HB_HEARTBEAT_LOCAL;
/* /*
* o2hb_dependent_users tracks the number of registered callbacks that depend * o2hb_dependent_users tracks the number of registered callbacks that depend
@ -141,7 +141,7 @@ unsigned int o2hb_heartbeat_mode = O2HB_HEARTBEAT_LOCAL;
* However only o2dlm depends on the heartbeat. It does not want the heartbeat * However only o2dlm depends on the heartbeat. It does not want the heartbeat
* to stop while a dlm domain is still active. * to stop while a dlm domain is still active.
*/ */
unsigned int o2hb_dependent_users; static unsigned int o2hb_dependent_users;
/* /*
* In global heartbeat mode, all regions are pinned if there are one or more * In global heartbeat mode, all regions are pinned if there are one or more
@ -2486,7 +2486,7 @@ unlock:
return ret; return ret;
} }
void o2hb_region_dec_user(const char *region_uuid) static void o2hb_region_dec_user(const char *region_uuid)
{ {
spin_lock(&o2hb_live_lock); spin_lock(&o2hb_live_lock);

View File

@ -35,9 +35,9 @@
* cluster references throughout where nodes are looked up */ * cluster references throughout where nodes are looked up */
struct o2nm_cluster *o2nm_single_cluster = NULL; struct o2nm_cluster *o2nm_single_cluster = NULL;
char *o2nm_fence_method_desc[O2NM_FENCE_METHODS] = { static const char *o2nm_fence_method_desc[O2NM_FENCE_METHODS] = {
"reset", /* O2NM_FENCE_RESET */ "reset", /* O2NM_FENCE_RESET */
"panic", /* O2NM_FENCE_PANIC */ "panic", /* O2NM_FENCE_PANIC */
}; };
static inline void o2nm_lock_subsystem(void); static inline void o2nm_lock_subsystem(void);

View File

@ -872,8 +872,6 @@ int o2net_register_handler(u32 msg_type, u32 key, u32 max_len,
"for type %u key %08x\n", msg_type, key); "for type %u key %08x\n", msg_type, key);
} }
write_unlock(&o2net_handler_lock); write_unlock(&o2net_handler_lock);
if (ret)
goto out;
out: out:
if (ret) if (ret)

View File

@ -96,7 +96,7 @@ struct ocfs2_unblock_ctl {
}; };
/* Lockdep class keys */ /* Lockdep class keys */
struct lock_class_key lockdep_keys[OCFS2_NUM_LOCK_TYPES]; static struct lock_class_key lockdep_keys[OCFS2_NUM_LOCK_TYPES];
static int ocfs2_check_meta_downconvert(struct ocfs2_lock_res *lockres, static int ocfs2_check_meta_downconvert(struct ocfs2_lock_res *lockres,
int new_level); int new_level);

View File

@ -637,10 +637,8 @@ static int ocfs2_truncate_for_delete(struct ocfs2_super *osb,
handle = NULL; handle = NULL;
status = ocfs2_commit_truncate(osb, inode, fe_bh); status = ocfs2_commit_truncate(osb, inode, fe_bh);
if (status < 0) { if (status < 0)
mlog_errno(status); mlog_errno(status);
goto out;
}
} }
out: out:
@ -1499,7 +1497,6 @@ static int ocfs2_filecheck_validate_inode_block(struct super_block *sb,
(unsigned long long)bh->b_blocknr, (unsigned long long)bh->b_blocknr,
le32_to_cpu(di->i_fs_generation)); le32_to_cpu(di->i_fs_generation));
rc = -OCFS2_FILECHECK_ERR_GENERATION; rc = -OCFS2_FILECHECK_ERR_GENERATION;
goto bail;
} }
bail: bail:

View File

@ -663,11 +663,10 @@ int ocfs2_reserve_local_alloc_bits(struct ocfs2_super *osb,
#ifdef CONFIG_OCFS2_DEBUG_FS #ifdef CONFIG_OCFS2_DEBUG_FS
if (le32_to_cpu(alloc->id1.bitmap1.i_used) != if (le32_to_cpu(alloc->id1.bitmap1.i_used) !=
ocfs2_local_alloc_count_bits(alloc)) { ocfs2_local_alloc_count_bits(alloc)) {
ocfs2_error(osb->sb, "local alloc inode %llu says it has %u used bits, but a count shows %u\n", status = ocfs2_error(osb->sb, "local alloc inode %llu says it has %u used bits, but a count shows %u\n",
(unsigned long long)le64_to_cpu(alloc->i_blkno), (unsigned long long)le64_to_cpu(alloc->i_blkno),
le32_to_cpu(alloc->id1.bitmap1.i_used), le32_to_cpu(alloc->id1.bitmap1.i_used),
ocfs2_local_alloc_count_bits(alloc)); ocfs2_local_alloc_count_bits(alloc));
status = -EIO;
goto bail; goto bail;
} }
#endif #endif

View File

@ -137,14 +137,13 @@ static int ocfs2_read_quota_block(struct inode *inode, u64 v_block,
int rc = 0; int rc = 0;
struct buffer_head *tmp = *bh; struct buffer_head *tmp = *bh;
if (i_size_read(inode) >> inode->i_sb->s_blocksize_bits <= v_block) { if (i_size_read(inode) >> inode->i_sb->s_blocksize_bits <= v_block)
ocfs2_error(inode->i_sb, return ocfs2_error(inode->i_sb,
"Quota file %llu is probably corrupted! Requested to read block %Lu but file has size only %Lu\n", "Quota file %llu is probably corrupted! Requested to read block %Lu but file has size only %Lu\n",
(unsigned long long)OCFS2_I(inode)->ip_blkno, (unsigned long long)OCFS2_I(inode)->ip_blkno,
(unsigned long long)v_block, (unsigned long long)v_block,
(unsigned long long)i_size_read(inode)); (unsigned long long)i_size_read(inode));
return -EIO;
}
rc = ocfs2_read_virt_blocks(inode, v_block, 1, &tmp, 0, rc = ocfs2_read_virt_blocks(inode, v_block, 1, &tmp, 0,
ocfs2_validate_quota_block); ocfs2_validate_quota_block);
if (rc) if (rc)

View File

@ -90,23 +90,22 @@ EXPORT_SYMBOL(seq_open);
static int traverse(struct seq_file *m, loff_t offset) static int traverse(struct seq_file *m, loff_t offset)
{ {
loff_t pos = 0, index; loff_t pos = 0;
int error = 0; int error = 0;
void *p; void *p;
m->version = 0; m->version = 0;
index = 0; m->index = 0;
m->count = m->from = 0; m->count = m->from = 0;
if (!offset) { if (!offset)
m->index = index;
return 0; return 0;
}
if (!m->buf) { if (!m->buf) {
m->buf = seq_buf_alloc(m->size = PAGE_SIZE); m->buf = seq_buf_alloc(m->size = PAGE_SIZE);
if (!m->buf) if (!m->buf)
return -ENOMEM; return -ENOMEM;
} }
p = m->op->start(m, &index); p = m->op->start(m, &m->index);
while (p) { while (p) {
error = PTR_ERR(p); error = PTR_ERR(p);
if (IS_ERR(p)) if (IS_ERR(p))
@ -123,20 +122,15 @@ static int traverse(struct seq_file *m, loff_t offset)
if (pos + m->count > offset) { if (pos + m->count > offset) {
m->from = offset - pos; m->from = offset - pos;
m->count -= m->from; m->count -= m->from;
m->index = index;
break; break;
} }
pos += m->count; pos += m->count;
m->count = 0; m->count = 0;
if (pos == offset) { p = m->op->next(m, p, &m->index);
index++; if (pos == offset)
m->index = index;
break; break;
}
p = m->op->next(m, p, &index);
} }
m->op->stop(m, p); m->op->stop(m, p);
m->index = index;
return error; return error;
Eoverflow: Eoverflow:
@ -160,7 +154,6 @@ ssize_t seq_read(struct file *file, char __user *buf, size_t size, loff_t *ppos)
{ {
struct seq_file *m = file->private_data; struct seq_file *m = file->private_data;
size_t copied = 0; size_t copied = 0;
loff_t pos;
size_t n; size_t n;
void *p; void *p;
int err = 0; int err = 0;
@ -223,16 +216,12 @@ ssize_t seq_read(struct file *file, char __user *buf, size_t size, loff_t *ppos)
size -= n; size -= n;
buf += n; buf += n;
copied += n; copied += n;
if (!m->count) {
m->from = 0;
m->index++;
}
if (!size) if (!size)
goto Done; goto Done;
} }
/* we need at least one record in buffer */ /* we need at least one record in buffer */
pos = m->index; m->from = 0;
p = m->op->start(m, &pos); p = m->op->start(m, &m->index);
while (1) { while (1) {
err = PTR_ERR(p); err = PTR_ERR(p);
if (!p || IS_ERR(p)) if (!p || IS_ERR(p))
@ -243,8 +232,7 @@ ssize_t seq_read(struct file *file, char __user *buf, size_t size, loff_t *ppos)
if (unlikely(err)) if (unlikely(err))
m->count = 0; m->count = 0;
if (unlikely(!m->count)) { if (unlikely(!m->count)) {
p = m->op->next(m, p, &pos); p = m->op->next(m, p, &m->index);
m->index = pos;
continue; continue;
} }
if (m->count < m->size) if (m->count < m->size)
@ -256,29 +244,33 @@ ssize_t seq_read(struct file *file, char __user *buf, size_t size, loff_t *ppos)
if (!m->buf) if (!m->buf)
goto Enomem; goto Enomem;
m->version = 0; m->version = 0;
pos = m->index; p = m->op->start(m, &m->index);
p = m->op->start(m, &pos);
} }
m->op->stop(m, p); m->op->stop(m, p);
m->count = 0; m->count = 0;
goto Done; goto Done;
Fill: Fill:
/* they want more? let's try to get some more */ /* they want more? let's try to get some more */
while (m->count < size) { while (1) {
size_t offs = m->count; size_t offs = m->count;
loff_t next = pos; loff_t pos = m->index;
p = m->op->next(m, p, &next);
p = m->op->next(m, p, &m->index);
if (pos == m->index)
/* Buggy ->next function */
m->index++;
if (!p || IS_ERR(p)) { if (!p || IS_ERR(p)) {
err = PTR_ERR(p); err = PTR_ERR(p);
break; break;
} }
if (m->count >= size)
break;
err = m->op->show(m, p); err = m->op->show(m, p);
if (seq_has_overflowed(m) || err) { if (seq_has_overflowed(m) || err) {
m->count = offs; m->count = offs;
if (likely(err <= 0)) if (likely(err <= 0))
break; break;
} }
pos = next;
} }
m->op->stop(m, p); m->op->stop(m, p);
n = min(m->count, size); n = min(m->count, size);
@ -287,11 +279,7 @@ Fill:
goto Efault; goto Efault;
copied += n; copied += n;
m->count -= n; m->count -= n;
if (m->count) m->from = n;
m->from = n;
else
pos++;
m->index = pos;
Done: Done:
if (!copied) if (!copied)
copied = err; copied = err;

View File

@ -144,6 +144,9 @@ static unsigned long super_cache_count(struct shrinker *shrink,
total_objects += list_lru_shrink_count(&sb->s_dentry_lru, sc); total_objects += list_lru_shrink_count(&sb->s_dentry_lru, sc);
total_objects += list_lru_shrink_count(&sb->s_inode_lru, sc); total_objects += list_lru_shrink_count(&sb->s_inode_lru, sc);
if (!total_objects)
return SHRINK_EMPTY;
total_objects = vfs_pressure_ratio(total_objects); total_objects = vfs_pressure_ratio(total_objects);
return total_objects; return total_objects;
} }
@ -244,10 +247,6 @@ static struct super_block *alloc_super(struct file_system_type *type, int flags,
INIT_LIST_HEAD(&s->s_inodes_wb); INIT_LIST_HEAD(&s->s_inodes_wb);
spin_lock_init(&s->s_inode_wblist_lock); spin_lock_init(&s->s_inode_wblist_lock);
if (list_lru_init_memcg(&s->s_dentry_lru))
goto fail;
if (list_lru_init_memcg(&s->s_inode_lru))
goto fail;
s->s_count = 1; s->s_count = 1;
atomic_set(&s->s_active, 1); atomic_set(&s->s_active, 1);
mutex_init(&s->s_vfs_rename_mutex); mutex_init(&s->s_vfs_rename_mutex);
@ -265,6 +264,10 @@ static struct super_block *alloc_super(struct file_system_type *type, int flags,
s->s_shrink.flags = SHRINKER_NUMA_AWARE | SHRINKER_MEMCG_AWARE; s->s_shrink.flags = SHRINKER_NUMA_AWARE | SHRINKER_MEMCG_AWARE;
if (prealloc_shrinker(&s->s_shrink)) if (prealloc_shrinker(&s->s_shrink))
goto fail; goto fail;
if (list_lru_init_memcg(&s->s_dentry_lru, &s->s_shrink))
goto fail;
if (list_lru_init_memcg(&s->s_inode_lru, &s->s_shrink))
goto fail;
return s; return s;
fail: fail:

View File

@ -547,7 +547,7 @@ static u64 ufs_add_fragments(struct inode *inode, u64 fragment,
/* /*
* Block can be extended * Block can be extended
*/ */
ucg->cg_time = cpu_to_fs32(sb, get_seconds()); ucg->cg_time = ufs_get_seconds(sb);
for (i = newcount; i < (uspi->s_fpb - fragoff); i++) for (i = newcount; i < (uspi->s_fpb - fragoff); i++)
if (ubh_isclr (UCPI_UBH(ucpi), ucpi->c_freeoff, fragno + i)) if (ubh_isclr (UCPI_UBH(ucpi), ucpi->c_freeoff, fragno + i))
break; break;
@ -639,7 +639,7 @@ cg_found:
if (!ufs_cg_chkmagic(sb, ucg)) if (!ufs_cg_chkmagic(sb, ucg))
ufs_panic (sb, "ufs_alloc_fragments", ufs_panic (sb, "ufs_alloc_fragments",
"internal error, bad magic number on cg %u", cgno); "internal error, bad magic number on cg %u", cgno);
ucg->cg_time = cpu_to_fs32(sb, get_seconds()); ucg->cg_time = ufs_get_seconds(sb);
if (count == uspi->s_fpb) { if (count == uspi->s_fpb) {
result = ufs_alloccg_block (inode, ucpi, goal, err); result = ufs_alloccg_block (inode, ucpi, goal, err);

View File

@ -89,7 +89,7 @@ void ufs_free_inode (struct inode * inode)
if (!ufs_cg_chkmagic(sb, ucg)) if (!ufs_cg_chkmagic(sb, ucg))
ufs_panic (sb, "ufs_free_fragments", "internal error, bad cg magic number"); ufs_panic (sb, "ufs_free_fragments", "internal error, bad cg magic number");
ucg->cg_time = cpu_to_fs32(sb, get_seconds()); ucg->cg_time = ufs_get_seconds(sb);
is_directory = S_ISDIR(inode->i_mode); is_directory = S_ISDIR(inode->i_mode);

View File

@ -698,7 +698,7 @@ static int ufs_sync_fs(struct super_block *sb, int wait)
usb1 = ubh_get_usb_first(uspi); usb1 = ubh_get_usb_first(uspi);
usb3 = ubh_get_usb_third(uspi); usb3 = ubh_get_usb_third(uspi);
usb1->fs_time = cpu_to_fs32(sb, get_seconds()); usb1->fs_time = ufs_get_seconds(sb);
if ((flags & UFS_ST_MASK) == UFS_ST_SUN || if ((flags & UFS_ST_MASK) == UFS_ST_SUN ||
(flags & UFS_ST_MASK) == UFS_ST_SUNOS || (flags & UFS_ST_MASK) == UFS_ST_SUNOS ||
(flags & UFS_ST_MASK) == UFS_ST_SUNx86) (flags & UFS_ST_MASK) == UFS_ST_SUNx86)
@ -1342,7 +1342,7 @@ static int ufs_remount (struct super_block *sb, int *mount_flags, char *data)
*/ */
if (*mount_flags & SB_RDONLY) { if (*mount_flags & SB_RDONLY) {
ufs_put_super_internal(sb); ufs_put_super_internal(sb);
usb1->fs_time = cpu_to_fs32(sb, get_seconds()); usb1->fs_time = ufs_get_seconds(sb);
if ((flags & UFS_ST_MASK) == UFS_ST_SUN if ((flags & UFS_ST_MASK) == UFS_ST_SUN
|| (flags & UFS_ST_MASK) == UFS_ST_SUNOS || (flags & UFS_ST_MASK) == UFS_ST_SUNOS
|| (flags & UFS_ST_MASK) == UFS_ST_SUNx86) || (flags & UFS_ST_MASK) == UFS_ST_SUNx86)

View File

@ -590,3 +590,17 @@ static inline int ufs_is_data_ptr_zero(struct ufs_sb_private_info *uspi,
else else
return *(__fs32 *)p == 0; return *(__fs32 *)p == 0;
} }
static inline __fs32 ufs_get_seconds(struct super_block *sbp)
{
time64_t now = ktime_get_real_seconds();
/* Signed 32-bit interpretation wraps around in 2038, which
* happens in ufs1 inode stamps but not ufs2 using 64-bits
* stamps. For superblock and blockgroup, let's assume
* unsigned 32-bit stamps, which are good until y2106.
* Wrap around rather than clamp here to make the dirty
* file system detection work in the superblock stamp.
*/
return cpu_to_fs32(sbp, lower_32_bits(now));
}

View File

@ -1849,17 +1849,14 @@ static void userfaultfd_show_fdinfo(struct seq_file *m, struct file *f)
{ {
struct userfaultfd_ctx *ctx = f->private_data; struct userfaultfd_ctx *ctx = f->private_data;
wait_queue_entry_t *wq; wait_queue_entry_t *wq;
struct userfaultfd_wait_queue *uwq;
unsigned long pending = 0, total = 0; unsigned long pending = 0, total = 0;
spin_lock(&ctx->fault_pending_wqh.lock); spin_lock(&ctx->fault_pending_wqh.lock);
list_for_each_entry(wq, &ctx->fault_pending_wqh.head, entry) { list_for_each_entry(wq, &ctx->fault_pending_wqh.head, entry) {
uwq = container_of(wq, struct userfaultfd_wait_queue, wq);
pending++; pending++;
total++; total++;
} }
list_for_each_entry(wq, &ctx->fault_wqh.head, entry) { list_for_each_entry(wq, &ctx->fault_wqh.head, entry) {
uwq = container_of(wq, struct userfaultfd_wait_queue, wq);
total++; total++;
} }
spin_unlock(&ctx->fault_pending_wqh.lock); spin_unlock(&ctx->fault_pending_wqh.lock);

View File

@ -1169,7 +1169,7 @@ xfs_file_mmap(
file_accessed(filp); file_accessed(filp);
vma->vm_ops = &xfs_file_vm_ops; vma->vm_ops = &xfs_file_vm_ops;
if (IS_DAX(file_inode(filp))) if (IS_DAX(file_inode(filp)))
vma->vm_flags |= VM_MIXEDMAP | VM_HUGEPAGE; vma->vm_flags |= VM_HUGEPAGE;
return 0; return 0;
} }

View File

@ -1095,6 +1095,24 @@ static inline bool arch_has_pfn_modify_check(void)
} }
#endif /* !_HAVE_ARCH_PFN_MODIFY_ALLOWED */ #endif /* !_HAVE_ARCH_PFN_MODIFY_ALLOWED */
/*
* Architecture PAGE_KERNEL_* fallbacks
*
* Some architectures don't define certain PAGE_KERNEL_* flags. This is either
* because they really don't support them, or the port needs to be updated to
* reflect the required functionality. Below are a set of relatively safe
* fallbacks, as best effort, which we can count on in lieu of the architectures
* not defining them on their own yet.
*/
#ifndef PAGE_KERNEL_RO
# define PAGE_KERNEL_RO PAGE_KERNEL
#endif
#ifndef PAGE_KERNEL_EXEC
# define PAGE_KERNEL_EXEC PAGE_KERNEL
#endif
#endif /* !__ASSEMBLY__ */ #endif /* !__ASSEMBLY__ */
#ifndef io_remap_pfn_range #ifndef io_remap_pfn_range

View File

@ -53,7 +53,7 @@
({ \ ({ \
BUILD_BUG_ON_MSG(!__builtin_constant_p(_mask), \ BUILD_BUG_ON_MSG(!__builtin_constant_p(_mask), \
_pfx "mask is not constant"); \ _pfx "mask is not constant"); \
BUILD_BUG_ON_MSG(!(_mask), _pfx "mask is zero"); \ BUILD_BUG_ON_MSG((_mask) == 0, _pfx "mask is zero"); \
BUILD_BUG_ON_MSG(__builtin_constant_p(_val) ? \ BUILD_BUG_ON_MSG(__builtin_constant_p(_val) ? \
~((_mask) >> __bf_shf(_mask)) & (_val) : 0, \ ~((_mask) >> __bf_shf(_mask)) & (_val) : 0, \
_pfx "value too large for the field"); \ _pfx "value too large for the field"); \

View File

@ -33,7 +33,7 @@ extern int cma_init_reserved_mem(phys_addr_t base, phys_addr_t size,
const char *name, const char *name,
struct cma **res_cma); struct cma **res_cma);
extern struct page *cma_alloc(struct cma *cma, size_t count, unsigned int align, extern struct page *cma_alloc(struct cma *cma, size_t count, unsigned int align,
gfp_t gfp_mask); bool no_warn);
extern bool cma_release(struct cma *cma, const struct page *pages, unsigned int count); extern bool cma_release(struct cma *cma, const struct page *pages, unsigned int count);
extern int cma_for_each_area(int (*it)(struct cma *cma, void *data), void *data); extern int cma_for_each_area(int (*it)(struct cma *cma, void *data), void *data);

View File

@ -112,7 +112,7 @@ static inline int dma_declare_contiguous(struct device *dev, phys_addr_t size,
} }
struct page *dma_alloc_from_contiguous(struct device *dev, size_t count, struct page *dma_alloc_from_contiguous(struct device *dev, size_t count,
unsigned int order, gfp_t gfp_mask); unsigned int order, bool no_warn);
bool dma_release_from_contiguous(struct device *dev, struct page *pages, bool dma_release_from_contiguous(struct device *dev, struct page *pages,
int count); int count);
@ -145,7 +145,7 @@ int dma_declare_contiguous(struct device *dev, phys_addr_t size,
static inline static inline
struct page *dma_alloc_from_contiguous(struct device *dev, size_t count, struct page *dma_alloc_from_contiguous(struct device *dev, size_t count,
unsigned int order, gfp_t gfp_mask) unsigned int order, bool no_warn)
{ {
return NULL; return NULL;
} }

View File

@ -179,7 +179,6 @@ typedef int (dio_iodone_t)(struct kiocb *iocb, loff_t offset,
#define ATTR_ATIME_SET (1 << 7) #define ATTR_ATIME_SET (1 << 7)
#define ATTR_MTIME_SET (1 << 8) #define ATTR_MTIME_SET (1 << 8)
#define ATTR_FORCE (1 << 9) /* Not a change, but a change it */ #define ATTR_FORCE (1 << 9) /* Not a change, but a change it */
#define ATTR_ATTR_FLAG (1 << 10)
#define ATTR_KILL_SUID (1 << 11) #define ATTR_KILL_SUID (1 << 11)
#define ATTR_KILL_SGID (1 << 12) #define ATTR_KILL_SGID (1 << 12)
#define ATTR_FILE (1 << 13) #define ATTR_FILE (1 << 13)
@ -345,6 +344,10 @@ struct address_space_operations {
/* Set a page dirty. Return true if this dirtied it */ /* Set a page dirty. Return true if this dirtied it */
int (*set_page_dirty)(struct page *page); int (*set_page_dirty)(struct page *page);
/*
* Reads in the requested pages. Unlike ->readpage(), this is
* PURELY used for read-ahead!.
*/
int (*readpages)(struct file *filp, struct address_space *mapping, int (*readpages)(struct file *filp, struct address_space *mapping,
struct list_head *pages, unsigned nr_pages); struct list_head *pages, unsigned nr_pages);

View File

@ -84,6 +84,8 @@ struct fsnotify_event_private_data;
struct fsnotify_fname; struct fsnotify_fname;
struct fsnotify_iter_info; struct fsnotify_iter_info;
struct mem_cgroup;
/* /*
* Each group much define these ops. The fsnotify infrastructure will call * Each group much define these ops. The fsnotify infrastructure will call
* these operations for each relevant group. * these operations for each relevant group.
@ -127,6 +129,8 @@ struct fsnotify_event {
* everything will be cleaned up. * everything will be cleaned up.
*/ */
struct fsnotify_group { struct fsnotify_group {
const struct fsnotify_ops *ops; /* how this group handles things */
/* /*
* How the refcnt is used is up to each group. When the refcnt hits 0 * How the refcnt is used is up to each group. When the refcnt hits 0
* fsnotify will clean up all of the resources associated with this group. * fsnotify will clean up all of the resources associated with this group.
@ -137,8 +141,6 @@ struct fsnotify_group {
*/ */
refcount_t refcnt; /* things with interest in this group */ refcount_t refcnt; /* things with interest in this group */
const struct fsnotify_ops *ops; /* how this group handles things */
/* needed to send notification to userspace */ /* needed to send notification to userspace */
spinlock_t notification_lock; /* protect the notification_list */ spinlock_t notification_lock; /* protect the notification_list */
struct list_head notification_list; /* list of event_holder this group needs to send to userspace */ struct list_head notification_list; /* list of event_holder this group needs to send to userspace */
@ -160,6 +162,8 @@ struct fsnotify_group {
atomic_t num_marks; /* 1 for each mark and 1 for not being atomic_t num_marks; /* 1 for each mark and 1 for not being
* past the point of no return when freeing * past the point of no return when freeing
* a group */ * a group */
atomic_t user_waits; /* Number of tasks waiting for user
* response */
struct list_head marks_list; /* all inode marks for this group */ struct list_head marks_list; /* all inode marks for this group */
struct fasync_struct *fsn_fa; /* async notification */ struct fasync_struct *fsn_fa; /* async notification */
@ -167,8 +171,8 @@ struct fsnotify_group {
struct fsnotify_event *overflow_event; /* Event we queue when the struct fsnotify_event *overflow_event; /* Event we queue when the
* notification list is too * notification list is too
* full */ * full */
atomic_t user_waits; /* Number of tasks waiting for user
* response */ struct mem_cgroup *memcg; /* memcg to charge allocations */
/* groups can define private fields here or use the void *private */ /* groups can define private fields here or use the void *private */
union { union {

View File

@ -348,9 +348,6 @@ struct hstate {
struct huge_bootmem_page { struct huge_bootmem_page {
struct list_head list; struct list_head list;
struct hstate *hstate; struct hstate *hstate;
#ifdef CONFIG_HIGHMEM
phys_addr_t phys;
#endif
}; };
struct page *alloc_huge_page(struct vm_area_struct *vma, struct page *alloc_huge_page(struct vm_area_struct *vma,

View File

@ -20,7 +20,7 @@ extern pmd_t kasan_zero_pmd[PTRS_PER_PMD];
extern pud_t kasan_zero_pud[PTRS_PER_PUD]; extern pud_t kasan_zero_pud[PTRS_PER_PUD];
extern p4d_t kasan_zero_p4d[MAX_PTRS_PER_P4D]; extern p4d_t kasan_zero_p4d[MAX_PTRS_PER_P4D];
void kasan_populate_zero_shadow(const void *shadow_start, int kasan_populate_zero_shadow(const void *shadow_start,
const void *shadow_end); const void *shadow_end);
static inline void *kasan_mem_to_shadow(const void *addr) static inline void *kasan_mem_to_shadow(const void *addr)
@ -71,6 +71,9 @@ struct kasan_cache {
int kasan_module_alloc(void *addr, size_t size); int kasan_module_alloc(void *addr, size_t size);
void kasan_free_shadow(const struct vm_struct *vm); void kasan_free_shadow(const struct vm_struct *vm);
int kasan_add_zero_shadow(void *start, unsigned long size);
void kasan_remove_zero_shadow(void *start, unsigned long size);
size_t ksize(const void *); size_t ksize(const void *);
static inline void kasan_unpoison_slab(const void *ptr) { ksize(ptr); } static inline void kasan_unpoison_slab(const void *ptr) { ksize(ptr); }
size_t kasan_metadata_size(struct kmem_cache *cache); size_t kasan_metadata_size(struct kmem_cache *cache);
@ -124,6 +127,14 @@ static inline bool kasan_slab_free(struct kmem_cache *s, void *object,
static inline int kasan_module_alloc(void *addr, size_t size) { return 0; } static inline int kasan_module_alloc(void *addr, size_t size) { return 0; }
static inline void kasan_free_shadow(const struct vm_struct *vm) {} static inline void kasan_free_shadow(const struct vm_struct *vm) {}
static inline int kasan_add_zero_shadow(void *start, unsigned long size)
{
return 0;
}
static inline void kasan_remove_zero_shadow(void *start,
unsigned long size)
{}
static inline void kasan_unpoison_slab(const void *ptr) { } static inline void kasan_unpoison_slab(const void *ptr) { }
static inline size_t kasan_metadata_size(struct kmem_cache *cache) { return 0; } static inline size_t kasan_metadata_size(struct kmem_cache *cache) { return 0; }

View File

@ -42,7 +42,7 @@ struct list_lru_node {
spinlock_t lock; spinlock_t lock;
/* global list, used for the root cgroup in cgroup aware lrus */ /* global list, used for the root cgroup in cgroup aware lrus */
struct list_lru_one lru; struct list_lru_one lru;
#if defined(CONFIG_MEMCG) && !defined(CONFIG_SLOB) #ifdef CONFIG_MEMCG_KMEM
/* for cgroup aware lrus points to per cgroup lists, otherwise NULL */ /* for cgroup aware lrus points to per cgroup lists, otherwise NULL */
struct list_lru_memcg __rcu *memcg_lrus; struct list_lru_memcg __rcu *memcg_lrus;
#endif #endif
@ -51,21 +51,25 @@ struct list_lru_node {
struct list_lru { struct list_lru {
struct list_lru_node *node; struct list_lru_node *node;
#if defined(CONFIG_MEMCG) && !defined(CONFIG_SLOB) #ifdef CONFIG_MEMCG_KMEM
struct list_head list; struct list_head list;
int shrinker_id;
#endif #endif
}; };
void list_lru_destroy(struct list_lru *lru); void list_lru_destroy(struct list_lru *lru);
int __list_lru_init(struct list_lru *lru, bool memcg_aware, int __list_lru_init(struct list_lru *lru, bool memcg_aware,
struct lock_class_key *key); struct lock_class_key *key, struct shrinker *shrinker);
#define list_lru_init(lru) __list_lru_init((lru), false, NULL) #define list_lru_init(lru) \
#define list_lru_init_key(lru, key) __list_lru_init((lru), false, (key)) __list_lru_init((lru), false, NULL, NULL)
#define list_lru_init_memcg(lru) __list_lru_init((lru), true, NULL) #define list_lru_init_key(lru, key) \
__list_lru_init((lru), false, (key), NULL)
#define list_lru_init_memcg(lru, shrinker) \
__list_lru_init((lru), true, NULL, shrinker)
int memcg_update_all_list_lrus(int num_memcgs); int memcg_update_all_list_lrus(int num_memcgs);
void memcg_drain_all_list_lrus(int src_idx, int dst_idx); void memcg_drain_all_list_lrus(int src_idx, struct mem_cgroup *dst_memcg);
/** /**
* list_lru_add: add an element to the lru list's tail * list_lru_add: add an element to the lru list's tail
@ -162,6 +166,23 @@ unsigned long list_lru_walk_one(struct list_lru *lru,
int nid, struct mem_cgroup *memcg, int nid, struct mem_cgroup *memcg,
list_lru_walk_cb isolate, void *cb_arg, list_lru_walk_cb isolate, void *cb_arg,
unsigned long *nr_to_walk); unsigned long *nr_to_walk);
/**
* list_lru_walk_one_irq: walk a list_lru, isolating and disposing freeable items.
* @lru: the lru pointer.
* @nid: the node id to scan from.
* @memcg: the cgroup to scan from.
* @isolate: callback function that is resposible for deciding what to do with
* the item currently being scanned
* @cb_arg: opaque type that will be passed to @isolate
* @nr_to_walk: how many items to scan.
*
* Same as @list_lru_walk_one except that the spinlock is acquired with
* spin_lock_irq().
*/
unsigned long list_lru_walk_one_irq(struct list_lru *lru,
int nid, struct mem_cgroup *memcg,
list_lru_walk_cb isolate, void *cb_arg,
unsigned long *nr_to_walk);
unsigned long list_lru_walk_node(struct list_lru *lru, int nid, unsigned long list_lru_walk_node(struct list_lru *lru, int nid,
list_lru_walk_cb isolate, void *cb_arg, list_lru_walk_cb isolate, void *cb_arg,
unsigned long *nr_to_walk); unsigned long *nr_to_walk);
@ -174,6 +195,14 @@ list_lru_shrink_walk(struct list_lru *lru, struct shrink_control *sc,
&sc->nr_to_scan); &sc->nr_to_scan);
} }
static inline unsigned long
list_lru_shrink_walk_irq(struct list_lru *lru, struct shrink_control *sc,
list_lru_walk_cb isolate, void *cb_arg)
{
return list_lru_walk_one_irq(lru, sc->nid, sc->memcg, isolate, cb_arg,
&sc->nr_to_scan);
}
static inline unsigned long static inline unsigned long
list_lru_walk(struct list_lru *lru, list_lru_walk_cb isolate, list_lru_walk(struct list_lru *lru, list_lru_walk_cb isolate,
void *cb_arg, unsigned long nr_to_walk) void *cb_arg, unsigned long nr_to_walk)

Some files were not shown because too many files have changed in this diff Show More