mirror of
https://mirrors.bfsu.edu.cn/git/linux.git
synced 2025-01-18 11:54:37 +08:00
Merge branch 'akpm' (patches from Andrew)
Merge more updates from Andrew Morton: - procfs updates - various misc bits - lib/ updates - epoll updates - autofs - fatfs - a few more MM bits * emailed patches from Andrew Morton <akpm@linux-foundation.org>: (58 commits) mm/page_io.c: fix polled swap page in checkpatch: add Co-developed-by to signature tags docs: fix Co-Developed-by docs drivers/base/platform.c: kmemleak ignore a known leak fs: don't open code lru_to_page() fs/: remove caller signal_pending branch predictions mm/: remove caller signal_pending branch predictions arch/arc/mm/fault.c: remove caller signal_pending_branch predictions kernel/sched/: remove caller signal_pending branch predictions kernel/locking/mutex.c: remove caller signal_pending branch predictions mm: select HAVE_MOVE_PMD on x86 for faster mremap mm: speed up mremap by 20x on large regions mm: treewide: remove unused address argument from pte_alloc functions initramfs: cleanup incomplete rootfs scripts/gdb: fix lx-version string output kernel/kcov.c: mark write_comp_data() as notrace kernel/sysctl: add panic_print into sysctl panic: add options to print system info when panic happens bfs: extra sanity checking and static inode bitmap exec: separate MM_ANONPAGES and RLIMIT_STACK accounting ...
This commit is contained in:
commit
a65981109f
@ -3092,6 +3092,14 @@
|
|||||||
timeout < 0: reboot immediately
|
timeout < 0: reboot immediately
|
||||||
Format: <timeout>
|
Format: <timeout>
|
||||||
|
|
||||||
|
panic_print= Bitmask for printing system info when panic happens.
|
||||||
|
User can chose combination of the following bits:
|
||||||
|
bit 0: print all tasks info
|
||||||
|
bit 1: print system memory info
|
||||||
|
bit 2: print timer info
|
||||||
|
bit 3: print locks info if CONFIG_LOCKDEP is on
|
||||||
|
bit 4: print ftrace buffer
|
||||||
|
|
||||||
panic_on_warn panic() instead of WARN(). Useful to cause kdump
|
panic_on_warn panic() instead of WARN(). Useful to cause kdump
|
||||||
on a WARN().
|
on a WARN().
|
||||||
|
|
||||||
|
@ -443,6 +443,9 @@ In function prototypes, include parameter names with their data types.
|
|||||||
Although this is not required by the C language, it is preferred in Linux
|
Although this is not required by the C language, it is preferred in Linux
|
||||||
because it is a simple way to add valuable information for the reader.
|
because it is a simple way to add valuable information for the reader.
|
||||||
|
|
||||||
|
Do not use the `extern' keyword with function prototypes as this makes
|
||||||
|
lines longer and isn't strictly necessary.
|
||||||
|
|
||||||
|
|
||||||
7) Centralized exiting of functions
|
7) Centralized exiting of functions
|
||||||
-----------------------------------
|
-----------------------------------
|
||||||
|
@ -510,7 +510,7 @@ tracking your trees, and to people trying to troubleshoot bugs in your
|
|||||||
tree.
|
tree.
|
||||||
|
|
||||||
|
|
||||||
12) When to use Acked-by:, Cc:, and Co-Developed-by:
|
12) When to use Acked-by:, Cc:, and Co-developed-by:
|
||||||
-------------------------------------------------------
|
-------------------------------------------------------
|
||||||
|
|
||||||
The Signed-off-by: tag indicates that the signer was involved in the
|
The Signed-off-by: tag indicates that the signer was involved in the
|
||||||
@ -543,7 +543,7 @@ person it names - but it should indicate that this person was copied on the
|
|||||||
patch. This tag documents that potentially interested parties
|
patch. This tag documents that potentially interested parties
|
||||||
have been included in the discussion.
|
have been included in the discussion.
|
||||||
|
|
||||||
A Co-Developed-by: states that the patch was also created by another developer
|
A Co-developed-by: states that the patch was also created by another developer
|
||||||
along with the original author. This is useful at times when multiple people
|
along with the original author. This is useful at times when multiple people
|
||||||
work on a single patch. Note, this person also needs to have a Signed-off-by:
|
work on a single patch. Note, this person also needs to have a Signed-off-by:
|
||||||
line in the patch as well.
|
line in the patch as well.
|
||||||
|
@ -60,6 +60,7 @@ show up in /proc/sys/kernel:
|
|||||||
- panic_on_stackoverflow
|
- panic_on_stackoverflow
|
||||||
- panic_on_unrecovered_nmi
|
- panic_on_unrecovered_nmi
|
||||||
- panic_on_warn
|
- panic_on_warn
|
||||||
|
- panic_print
|
||||||
- panic_on_rcu_stall
|
- panic_on_rcu_stall
|
||||||
- perf_cpu_time_max_percent
|
- perf_cpu_time_max_percent
|
||||||
- perf_event_paranoid
|
- perf_event_paranoid
|
||||||
@ -654,6 +655,22 @@ a kernel rebuild when attempting to kdump at the location of a WARN().
|
|||||||
|
|
||||||
==============================================================
|
==============================================================
|
||||||
|
|
||||||
|
panic_print:
|
||||||
|
|
||||||
|
Bitmask for printing system info when panic happens. User can chose
|
||||||
|
combination of the following bits:
|
||||||
|
|
||||||
|
bit 0: print all tasks info
|
||||||
|
bit 1: print system memory info
|
||||||
|
bit 2: print timer info
|
||||||
|
bit 3: print locks info if CONFIG_LOCKDEP is on
|
||||||
|
bit 4: print ftrace buffer
|
||||||
|
|
||||||
|
So for example to print tasks and memory info on panic, user can:
|
||||||
|
echo 3 > /proc/sys/kernel/panic_print
|
||||||
|
|
||||||
|
==============================================================
|
||||||
|
|
||||||
panic_on_rcu_stall:
|
panic_on_rcu_stall:
|
||||||
|
|
||||||
When set to 1, calls panic() after RCU stall detection messages. This
|
When set to 1, calls panic() after RCU stall detection messages. This
|
||||||
|
@ -535,6 +535,11 @@ config HAVE_IRQ_TIME_ACCOUNTING
|
|||||||
Archs need to ensure they use a high enough resolution clock to
|
Archs need to ensure they use a high enough resolution clock to
|
||||||
support irq time accounting and then call enable_sched_clock_irqtime().
|
support irq time accounting and then call enable_sched_clock_irqtime().
|
||||||
|
|
||||||
|
config HAVE_MOVE_PMD
|
||||||
|
bool
|
||||||
|
help
|
||||||
|
Archs that select this are able to move page tables at the PMD level.
|
||||||
|
|
||||||
config HAVE_ARCH_TRANSPARENT_HUGEPAGE
|
config HAVE_ARCH_TRANSPARENT_HUGEPAGE
|
||||||
bool
|
bool
|
||||||
|
|
||||||
|
@ -391,9 +391,9 @@ static inline unsigned long __fls(unsigned long x)
|
|||||||
return fls64(x) - 1;
|
return fls64(x) - 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline int fls(int x)
|
static inline int fls(unsigned int x)
|
||||||
{
|
{
|
||||||
return fls64((unsigned int) x);
|
return fls64(x);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -52,7 +52,7 @@ pmd_free(struct mm_struct *mm, pmd_t *pmd)
|
|||||||
}
|
}
|
||||||
|
|
||||||
static inline pte_t *
|
static inline pte_t *
|
||||||
pte_alloc_one_kernel(struct mm_struct *mm, unsigned long address)
|
pte_alloc_one_kernel(struct mm_struct *mm)
|
||||||
{
|
{
|
||||||
pte_t *pte = (pte_t *)__get_free_page(GFP_KERNEL|__GFP_ZERO);
|
pte_t *pte = (pte_t *)__get_free_page(GFP_KERNEL|__GFP_ZERO);
|
||||||
return pte;
|
return pte;
|
||||||
@ -65,9 +65,9 @@ pte_free_kernel(struct mm_struct *mm, pte_t *pte)
|
|||||||
}
|
}
|
||||||
|
|
||||||
static inline pgtable_t
|
static inline pgtable_t
|
||||||
pte_alloc_one(struct mm_struct *mm, unsigned long address)
|
pte_alloc_one(struct mm_struct *mm)
|
||||||
{
|
{
|
||||||
pte_t *pte = pte_alloc_one_kernel(mm, address);
|
pte_t *pte = pte_alloc_one_kernel(mm);
|
||||||
struct page *page;
|
struct page *page;
|
||||||
|
|
||||||
if (!pte)
|
if (!pte)
|
||||||
|
@ -278,7 +278,7 @@ static inline __attribute__ ((const)) int clz(unsigned int x)
|
|||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline int constant_fls(int x)
|
static inline int constant_fls(unsigned int x)
|
||||||
{
|
{
|
||||||
int r = 32;
|
int r = 32;
|
||||||
|
|
||||||
@ -312,7 +312,7 @@ static inline int constant_fls(int x)
|
|||||||
* @result: [1-32]
|
* @result: [1-32]
|
||||||
* fls(1) = 1, fls(0x80000000) = 32, fls(0) = 0
|
* fls(1) = 1, fls(0x80000000) = 32, fls(0) = 0
|
||||||
*/
|
*/
|
||||||
static inline __attribute__ ((const)) int fls(unsigned long x)
|
static inline __attribute__ ((const)) int fls(unsigned int x)
|
||||||
{
|
{
|
||||||
if (__builtin_constant_p(x))
|
if (__builtin_constant_p(x))
|
||||||
return constant_fls(x);
|
return constant_fls(x);
|
||||||
|
@ -90,8 +90,7 @@ static inline int __get_order_pte(void)
|
|||||||
return get_order(PTRS_PER_PTE * sizeof(pte_t));
|
return get_order(PTRS_PER_PTE * sizeof(pte_t));
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm,
|
static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm)
|
||||||
unsigned long address)
|
|
||||||
{
|
{
|
||||||
pte_t *pte;
|
pte_t *pte;
|
||||||
|
|
||||||
@ -102,7 +101,7 @@ static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm,
|
|||||||
}
|
}
|
||||||
|
|
||||||
static inline pgtable_t
|
static inline pgtable_t
|
||||||
pte_alloc_one(struct mm_struct *mm, unsigned long address)
|
pte_alloc_one(struct mm_struct *mm)
|
||||||
{
|
{
|
||||||
pgtable_t pte_pg;
|
pgtable_t pte_pg;
|
||||||
struct page *page;
|
struct page *page;
|
||||||
|
@ -142,7 +142,7 @@ good_area:
|
|||||||
fault = handle_mm_fault(vma, address, flags);
|
fault = handle_mm_fault(vma, address, flags);
|
||||||
|
|
||||||
/* If Pagefault was interrupted by SIGKILL, exit page fault "early" */
|
/* If Pagefault was interrupted by SIGKILL, exit page fault "early" */
|
||||||
if (unlikely(fatal_signal_pending(current))) {
|
if (fatal_signal_pending(current)) {
|
||||||
if ((fault & VM_FAULT_ERROR) && !(fault & VM_FAULT_RETRY))
|
if ((fault & VM_FAULT_ERROR) && !(fault & VM_FAULT_RETRY))
|
||||||
up_read(&mm->mmap_sem);
|
up_read(&mm->mmap_sem);
|
||||||
if (user_mode(regs))
|
if (user_mode(regs))
|
||||||
|
@ -81,7 +81,7 @@ static inline void clean_pte_table(pte_t *pte)
|
|||||||
* +------------+
|
* +------------+
|
||||||
*/
|
*/
|
||||||
static inline pte_t *
|
static inline pte_t *
|
||||||
pte_alloc_one_kernel(struct mm_struct *mm, unsigned long addr)
|
pte_alloc_one_kernel(struct mm_struct *mm)
|
||||||
{
|
{
|
||||||
pte_t *pte;
|
pte_t *pte;
|
||||||
|
|
||||||
@ -93,7 +93,7 @@ pte_alloc_one_kernel(struct mm_struct *mm, unsigned long addr)
|
|||||||
}
|
}
|
||||||
|
|
||||||
static inline pgtable_t
|
static inline pgtable_t
|
||||||
pte_alloc_one(struct mm_struct *mm, unsigned long addr)
|
pte_alloc_one(struct mm_struct *mm)
|
||||||
{
|
{
|
||||||
struct page *pte;
|
struct page *pte;
|
||||||
|
|
||||||
|
@ -91,13 +91,13 @@ extern pgd_t *pgd_alloc(struct mm_struct *mm);
|
|||||||
extern void pgd_free(struct mm_struct *mm, pgd_t *pgdp);
|
extern void pgd_free(struct mm_struct *mm, pgd_t *pgdp);
|
||||||
|
|
||||||
static inline pte_t *
|
static inline pte_t *
|
||||||
pte_alloc_one_kernel(struct mm_struct *mm, unsigned long addr)
|
pte_alloc_one_kernel(struct mm_struct *mm)
|
||||||
{
|
{
|
||||||
return (pte_t *)__get_free_page(PGALLOC_GFP);
|
return (pte_t *)__get_free_page(PGALLOC_GFP);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline pgtable_t
|
static inline pgtable_t
|
||||||
pte_alloc_one(struct mm_struct *mm, unsigned long addr)
|
pte_alloc_one(struct mm_struct *mm)
|
||||||
{
|
{
|
||||||
struct page *pte;
|
struct page *pte;
|
||||||
|
|
||||||
|
@ -54,7 +54,7 @@ static inline unsigned long __ffs(unsigned long x)
|
|||||||
* This is defined the same way as ffs.
|
* This is defined the same way as ffs.
|
||||||
* Note fls(0) = 0, fls(1) = 1, fls(0x80000000) = 32.
|
* Note fls(0) = 0, fls(1) = 1, fls(0x80000000) = 32.
|
||||||
*/
|
*/
|
||||||
static inline int fls(int x)
|
static inline int fls(unsigned int x)
|
||||||
{
|
{
|
||||||
if (!x)
|
if (!x)
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -40,7 +40,7 @@ static __always_inline unsigned long __ffs(unsigned long x)
|
|||||||
/*
|
/*
|
||||||
* asm-generic/bitops/fls.h
|
* asm-generic/bitops/fls.h
|
||||||
*/
|
*/
|
||||||
static __always_inline int fls(int x)
|
static __always_inline int fls(unsigned int x)
|
||||||
{
|
{
|
||||||
asm volatile(
|
asm volatile(
|
||||||
"ff1 %0\n"
|
"ff1 %0\n"
|
||||||
|
@ -211,7 +211,7 @@ static inline long ffz(int x)
|
|||||||
* This is defined the same way as ffs.
|
* This is defined the same way as ffs.
|
||||||
* Note fls(0) = 0, fls(1) = 1, fls(0x80000000) = 32.
|
* Note fls(0) = 0, fls(1) = 1, fls(0x80000000) = 32.
|
||||||
*/
|
*/
|
||||||
static inline int fls(int x)
|
static inline int fls(unsigned int x)
|
||||||
{
|
{
|
||||||
int r;
|
int r;
|
||||||
|
|
||||||
|
@ -59,8 +59,7 @@ static inline void pgd_free(struct mm_struct *mm, pgd_t *pgd)
|
|||||||
free_page((unsigned long) pgd);
|
free_page((unsigned long) pgd);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline struct page *pte_alloc_one(struct mm_struct *mm,
|
static inline struct page *pte_alloc_one(struct mm_struct *mm)
|
||||||
unsigned long address)
|
|
||||||
{
|
{
|
||||||
struct page *pte;
|
struct page *pte;
|
||||||
|
|
||||||
@ -75,8 +74,7 @@ static inline struct page *pte_alloc_one(struct mm_struct *mm,
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* _kernel variant gets to use a different allocator */
|
/* _kernel variant gets to use a different allocator */
|
||||||
static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm,
|
static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm)
|
||||||
unsigned long address)
|
|
||||||
{
|
{
|
||||||
gfp_t flags = GFP_KERNEL | __GFP_ZERO;
|
gfp_t flags = GFP_KERNEL | __GFP_ZERO;
|
||||||
return (pte_t *) __get_free_page(flags);
|
return (pte_t *) __get_free_page(flags);
|
||||||
|
@ -388,8 +388,7 @@ ia64_fls (unsigned long x)
|
|||||||
* Find the last (most significant) bit set. Returns 0 for x==0 and
|
* Find the last (most significant) bit set. Returns 0 for x==0 and
|
||||||
* bits are numbered from 1..32 (e.g., fls(9) == 4).
|
* bits are numbered from 1..32 (e.g., fls(9) == 4).
|
||||||
*/
|
*/
|
||||||
static inline int
|
static inline int fls(unsigned int t)
|
||||||
fls (int t)
|
|
||||||
{
|
{
|
||||||
unsigned long x = t & 0xffffffffu;
|
unsigned long x = t & 0xffffffffu;
|
||||||
|
|
||||||
|
@ -83,7 +83,7 @@ pmd_populate_kernel(struct mm_struct *mm, pmd_t * pmd_entry, pte_t * pte)
|
|||||||
pmd_val(*pmd_entry) = __pa(pte);
|
pmd_val(*pmd_entry) = __pa(pte);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline pgtable_t pte_alloc_one(struct mm_struct *mm, unsigned long addr)
|
static inline pgtable_t pte_alloc_one(struct mm_struct *mm)
|
||||||
{
|
{
|
||||||
struct page *page;
|
struct page *page;
|
||||||
void *pg;
|
void *pg;
|
||||||
@ -99,8 +99,7 @@ static inline pgtable_t pte_alloc_one(struct mm_struct *mm, unsigned long addr)
|
|||||||
return page;
|
return page;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm,
|
static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm)
|
||||||
unsigned long addr)
|
|
||||||
{
|
{
|
||||||
return quicklist_alloc(0, GFP_KERNEL, NULL);
|
return quicklist_alloc(0, GFP_KERNEL, NULL);
|
||||||
}
|
}
|
||||||
|
@ -502,7 +502,7 @@ static inline unsigned long __ffs(unsigned long x)
|
|||||||
/*
|
/*
|
||||||
* fls: find last bit set.
|
* fls: find last bit set.
|
||||||
*/
|
*/
|
||||||
static inline int fls(int x)
|
static inline int fls(unsigned int x)
|
||||||
{
|
{
|
||||||
int cnt;
|
int cnt;
|
||||||
|
|
||||||
|
@ -12,8 +12,7 @@ extern inline void pte_free_kernel(struct mm_struct *mm, pte_t *pte)
|
|||||||
|
|
||||||
extern const char bad_pmd_string[];
|
extern const char bad_pmd_string[];
|
||||||
|
|
||||||
extern inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm,
|
extern inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm)
|
||||||
unsigned long address)
|
|
||||||
{
|
{
|
||||||
unsigned long page = __get_free_page(GFP_DMA);
|
unsigned long page = __get_free_page(GFP_DMA);
|
||||||
|
|
||||||
@ -32,8 +31,6 @@ extern inline pmd_t *pmd_alloc_kernel(pgd_t *pgd, unsigned long address)
|
|||||||
#define pmd_alloc_one_fast(mm, address) ({ BUG(); ((pmd_t *)1); })
|
#define pmd_alloc_one_fast(mm, address) ({ BUG(); ((pmd_t *)1); })
|
||||||
#define pmd_alloc_one(mm, address) ({ BUG(); ((pmd_t *)2); })
|
#define pmd_alloc_one(mm, address) ({ BUG(); ((pmd_t *)2); })
|
||||||
|
|
||||||
#define pte_alloc_one_fast(mm, addr) pte_alloc_one(mm, addr)
|
|
||||||
|
|
||||||
#define pmd_populate(mm, pmd, page) (pmd_val(*pmd) = \
|
#define pmd_populate(mm, pmd, page) (pmd_val(*pmd) = \
|
||||||
(unsigned long)(page_address(page)))
|
(unsigned long)(page_address(page)))
|
||||||
|
|
||||||
@ -50,8 +47,7 @@ static inline void __pte_free_tlb(struct mmu_gather *tlb, pgtable_t page,
|
|||||||
|
|
||||||
#define __pmd_free_tlb(tlb, pmd, address) do { } while (0)
|
#define __pmd_free_tlb(tlb, pmd, address) do { } while (0)
|
||||||
|
|
||||||
static inline struct page *pte_alloc_one(struct mm_struct *mm,
|
static inline struct page *pte_alloc_one(struct mm_struct *mm)
|
||||||
unsigned long address)
|
|
||||||
{
|
{
|
||||||
struct page *page = alloc_pages(GFP_DMA, 0);
|
struct page *page = alloc_pages(GFP_DMA, 0);
|
||||||
pte_t *pte;
|
pte_t *pte;
|
||||||
|
@ -8,7 +8,7 @@
|
|||||||
extern pmd_t *get_pointer_table(void);
|
extern pmd_t *get_pointer_table(void);
|
||||||
extern int free_pointer_table(pmd_t *);
|
extern int free_pointer_table(pmd_t *);
|
||||||
|
|
||||||
static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm, unsigned long address)
|
static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm)
|
||||||
{
|
{
|
||||||
pte_t *pte;
|
pte_t *pte;
|
||||||
|
|
||||||
@ -28,7 +28,7 @@ static inline void pte_free_kernel(struct mm_struct *mm, pte_t *pte)
|
|||||||
free_page((unsigned long) pte);
|
free_page((unsigned long) pte);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline pgtable_t pte_alloc_one(struct mm_struct *mm, unsigned long address)
|
static inline pgtable_t pte_alloc_one(struct mm_struct *mm)
|
||||||
{
|
{
|
||||||
struct page *page;
|
struct page *page;
|
||||||
pte_t *pte;
|
pte_t *pte;
|
||||||
|
@ -35,8 +35,7 @@ do { \
|
|||||||
tlb_remove_page((tlb), pte); \
|
tlb_remove_page((tlb), pte); \
|
||||||
} while (0)
|
} while (0)
|
||||||
|
|
||||||
static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm,
|
static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm)
|
||||||
unsigned long address)
|
|
||||||
{
|
{
|
||||||
unsigned long page = __get_free_page(GFP_KERNEL);
|
unsigned long page = __get_free_page(GFP_KERNEL);
|
||||||
|
|
||||||
@ -47,8 +46,7 @@ static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm,
|
|||||||
return (pte_t *) (page);
|
return (pte_t *) (page);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline pgtable_t pte_alloc_one(struct mm_struct *mm,
|
static inline pgtable_t pte_alloc_one(struct mm_struct *mm)
|
||||||
unsigned long address)
|
|
||||||
{
|
{
|
||||||
struct page *page = alloc_pages(GFP_KERNEL, 0);
|
struct page *page = alloc_pages(GFP_KERNEL, 0);
|
||||||
|
|
||||||
|
@ -108,10 +108,9 @@ static inline void free_pgd_slow(pgd_t *pgd)
|
|||||||
#define pmd_alloc_one_fast(mm, address) ({ BUG(); ((pmd_t *)1); })
|
#define pmd_alloc_one_fast(mm, address) ({ BUG(); ((pmd_t *)1); })
|
||||||
#define pmd_alloc_one(mm, address) ({ BUG(); ((pmd_t *)2); })
|
#define pmd_alloc_one(mm, address) ({ BUG(); ((pmd_t *)2); })
|
||||||
|
|
||||||
extern pte_t *pte_alloc_one_kernel(struct mm_struct *mm, unsigned long addr);
|
extern pte_t *pte_alloc_one_kernel(struct mm_struct *mm);
|
||||||
|
|
||||||
static inline struct page *pte_alloc_one(struct mm_struct *mm,
|
static inline struct page *pte_alloc_one(struct mm_struct *mm)
|
||||||
unsigned long address)
|
|
||||||
{
|
{
|
||||||
struct page *ptepage;
|
struct page *ptepage;
|
||||||
|
|
||||||
@ -132,20 +131,6 @@ static inline struct page *pte_alloc_one(struct mm_struct *mm,
|
|||||||
return ptepage;
|
return ptepage;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline pte_t *pte_alloc_one_fast(struct mm_struct *mm,
|
|
||||||
unsigned long address)
|
|
||||||
{
|
|
||||||
unsigned long *ret;
|
|
||||||
|
|
||||||
ret = pte_quicklist;
|
|
||||||
if (ret != NULL) {
|
|
||||||
pte_quicklist = (unsigned long *)(*ret);
|
|
||||||
ret[0] = 0;
|
|
||||||
pgtable_cache_size--;
|
|
||||||
}
|
|
||||||
return (pte_t *)ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline void pte_free_fast(pte_t *pte)
|
static inline void pte_free_fast(pte_t *pte)
|
||||||
{
|
{
|
||||||
*(unsigned long **)pte = pte_quicklist;
|
*(unsigned long **)pte = pte_quicklist;
|
||||||
|
@ -235,8 +235,7 @@ unsigned long iopa(unsigned long addr)
|
|||||||
return pa;
|
return pa;
|
||||||
}
|
}
|
||||||
|
|
||||||
__ref pte_t *pte_alloc_one_kernel(struct mm_struct *mm,
|
__ref pte_t *pte_alloc_one_kernel(struct mm_struct *mm)
|
||||||
unsigned long address)
|
|
||||||
{
|
{
|
||||||
pte_t *pte;
|
pte_t *pte;
|
||||||
if (mem_init_done) {
|
if (mem_init_done) {
|
||||||
|
@ -555,7 +555,7 @@ static inline unsigned long __ffs(unsigned long word)
|
|||||||
* This is defined the same way as ffs.
|
* This is defined the same way as ffs.
|
||||||
* Note fls(0) = 0, fls(1) = 1, fls(0x80000000) = 32.
|
* Note fls(0) = 0, fls(1) = 1, fls(0x80000000) = 32.
|
||||||
*/
|
*/
|
||||||
static inline int fls(int x)
|
static inline int fls(unsigned int x)
|
||||||
{
|
{
|
||||||
int r;
|
int r;
|
||||||
|
|
||||||
|
@ -50,14 +50,12 @@ static inline void pgd_free(struct mm_struct *mm, pgd_t *pgd)
|
|||||||
free_pages((unsigned long)pgd, PGD_ORDER);
|
free_pages((unsigned long)pgd, PGD_ORDER);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm,
|
static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm)
|
||||||
unsigned long address)
|
|
||||||
{
|
{
|
||||||
return (pte_t *)__get_free_pages(GFP_KERNEL | __GFP_ZERO, PTE_ORDER);
|
return (pte_t *)__get_free_pages(GFP_KERNEL | __GFP_ZERO, PTE_ORDER);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline struct page *pte_alloc_one(struct mm_struct *mm,
|
static inline struct page *pte_alloc_one(struct mm_struct *mm)
|
||||||
unsigned long address)
|
|
||||||
{
|
{
|
||||||
struct page *pte;
|
struct page *pte;
|
||||||
|
|
||||||
|
@ -22,8 +22,7 @@ extern void pgd_free(struct mm_struct *mm, pgd_t * pgd);
|
|||||||
|
|
||||||
#define check_pgt_cache() do { } while (0)
|
#define check_pgt_cache() do { } while (0)
|
||||||
|
|
||||||
static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm,
|
static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm)
|
||||||
unsigned long addr)
|
|
||||||
{
|
{
|
||||||
pte_t *pte;
|
pte_t *pte;
|
||||||
|
|
||||||
@ -34,7 +33,7 @@ static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm,
|
|||||||
return pte;
|
return pte;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline pgtable_t pte_alloc_one(struct mm_struct *mm, unsigned long addr)
|
static inline pgtable_t pte_alloc_one(struct mm_struct *mm)
|
||||||
{
|
{
|
||||||
pgtable_t pte;
|
pgtable_t pte;
|
||||||
|
|
||||||
|
@ -37,8 +37,7 @@ static inline void pgd_free(struct mm_struct *mm, pgd_t *pgd)
|
|||||||
free_pages((unsigned long)pgd, PGD_ORDER);
|
free_pages((unsigned long)pgd, PGD_ORDER);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm,
|
static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm)
|
||||||
unsigned long address)
|
|
||||||
{
|
{
|
||||||
pte_t *pte;
|
pte_t *pte;
|
||||||
|
|
||||||
@ -47,8 +46,7 @@ static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm,
|
|||||||
return pte;
|
return pte;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline pgtable_t pte_alloc_one(struct mm_struct *mm,
|
static inline pgtable_t pte_alloc_one(struct mm_struct *mm)
|
||||||
unsigned long address)
|
|
||||||
{
|
{
|
||||||
struct page *pte;
|
struct page *pte;
|
||||||
|
|
||||||
|
@ -15,7 +15,7 @@
|
|||||||
|
|
||||||
#ifdef CONFIG_OPENRISC_HAVE_INST_FL1
|
#ifdef CONFIG_OPENRISC_HAVE_INST_FL1
|
||||||
|
|
||||||
static inline int fls(int x)
|
static inline int fls(unsigned int x)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
|
@ -70,10 +70,9 @@ static inline void pgd_free(struct mm_struct *mm, pgd_t *pgd)
|
|||||||
free_page((unsigned long)pgd);
|
free_page((unsigned long)pgd);
|
||||||
}
|
}
|
||||||
|
|
||||||
extern pte_t *pte_alloc_one_kernel(struct mm_struct *mm, unsigned long address);
|
extern pte_t *pte_alloc_one_kernel(struct mm_struct *mm);
|
||||||
|
|
||||||
static inline struct page *pte_alloc_one(struct mm_struct *mm,
|
static inline struct page *pte_alloc_one(struct mm_struct *mm)
|
||||||
unsigned long address)
|
|
||||||
{
|
{
|
||||||
struct page *pte;
|
struct page *pte;
|
||||||
pte = alloc_pages(GFP_KERNEL, 0);
|
pte = alloc_pages(GFP_KERNEL, 0);
|
||||||
|
@ -118,8 +118,7 @@ EXPORT_SYMBOL(iounmap);
|
|||||||
* the memblock infrastructure.
|
* the memblock infrastructure.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
pte_t __ref *pte_alloc_one_kernel(struct mm_struct *mm,
|
pte_t __ref *pte_alloc_one_kernel(struct mm_struct *mm)
|
||||||
unsigned long address)
|
|
||||||
{
|
{
|
||||||
pte_t *pte;
|
pte_t *pte;
|
||||||
|
|
||||||
|
@ -188,7 +188,7 @@ static __inline__ int ffs(int x)
|
|||||||
* fls(0) = 0, fls(1) = 1, fls(0x80000000) = 32.
|
* fls(0) = 0, fls(1) = 1, fls(0x80000000) = 32.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
static __inline__ int fls(int x)
|
static __inline__ int fls(unsigned int x)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
if (!x)
|
if (!x)
|
||||||
|
@ -122,7 +122,7 @@ pmd_populate_kernel(struct mm_struct *mm, pmd_t *pmd, pte_t *pte)
|
|||||||
#define pmd_pgtable(pmd) pmd_page(pmd)
|
#define pmd_pgtable(pmd) pmd_page(pmd)
|
||||||
|
|
||||||
static inline pgtable_t
|
static inline pgtable_t
|
||||||
pte_alloc_one(struct mm_struct *mm, unsigned long address)
|
pte_alloc_one(struct mm_struct *mm)
|
||||||
{
|
{
|
||||||
struct page *page = alloc_page(GFP_KERNEL|__GFP_ZERO);
|
struct page *page = alloc_page(GFP_KERNEL|__GFP_ZERO);
|
||||||
if (!page)
|
if (!page)
|
||||||
@ -135,7 +135,7 @@ pte_alloc_one(struct mm_struct *mm, unsigned long address)
|
|||||||
}
|
}
|
||||||
|
|
||||||
static inline pte_t *
|
static inline pte_t *
|
||||||
pte_alloc_one_kernel(struct mm_struct *mm, unsigned long addr)
|
pte_alloc_one_kernel(struct mm_struct *mm)
|
||||||
{
|
{
|
||||||
pte_t *pte = (pte_t *)__get_free_page(GFP_KERNEL|__GFP_ZERO);
|
pte_t *pte = (pte_t *)__get_free_page(GFP_KERNEL|__GFP_ZERO);
|
||||||
return pte;
|
return pte;
|
||||||
|
@ -61,10 +61,10 @@ static inline void pmd_populate(struct mm_struct *mm, pmd_t *pmdp,
|
|||||||
|
|
||||||
#define pmd_pgtable(pmd) ((pgtable_t)pmd_page_vaddr(pmd))
|
#define pmd_pgtable(pmd) ((pgtable_t)pmd_page_vaddr(pmd))
|
||||||
|
|
||||||
extern pte_t *pte_alloc_one_kernel(struct mm_struct *mm, unsigned long addr);
|
extern pte_t *pte_alloc_one_kernel(struct mm_struct *mm);
|
||||||
extern pgtable_t pte_alloc_one(struct mm_struct *mm, unsigned long addr);
|
extern pgtable_t pte_alloc_one(struct mm_struct *mm);
|
||||||
void pte_frag_destroy(void *pte_frag);
|
void pte_frag_destroy(void *pte_frag);
|
||||||
pte_t *pte_fragment_alloc(struct mm_struct *mm, unsigned long vmaddr, int kernel);
|
pte_t *pte_fragment_alloc(struct mm_struct *mm, int kernel);
|
||||||
void pte_fragment_free(unsigned long *table, int kernel);
|
void pte_fragment_free(unsigned long *table, int kernel);
|
||||||
|
|
||||||
static inline void pte_free_kernel(struct mm_struct *mm, pte_t *pte)
|
static inline void pte_free_kernel(struct mm_struct *mm, pte_t *pte)
|
||||||
|
@ -39,7 +39,7 @@ extern struct vmemmap_backing *vmemmap_list;
|
|||||||
extern struct kmem_cache *pgtable_cache[];
|
extern struct kmem_cache *pgtable_cache[];
|
||||||
#define PGT_CACHE(shift) pgtable_cache[shift]
|
#define PGT_CACHE(shift) pgtable_cache[shift]
|
||||||
|
|
||||||
extern pte_t *pte_fragment_alloc(struct mm_struct *, unsigned long, int);
|
extern pte_t *pte_fragment_alloc(struct mm_struct *, int);
|
||||||
extern pmd_t *pmd_fragment_alloc(struct mm_struct *, unsigned long);
|
extern pmd_t *pmd_fragment_alloc(struct mm_struct *, unsigned long);
|
||||||
extern void pte_fragment_free(unsigned long *, int);
|
extern void pte_fragment_free(unsigned long *, int);
|
||||||
extern void pmd_fragment_free(unsigned long *);
|
extern void pmd_fragment_free(unsigned long *);
|
||||||
@ -190,16 +190,14 @@ static inline pgtable_t pmd_pgtable(pmd_t pmd)
|
|||||||
return (pgtable_t)pmd_page_vaddr(pmd);
|
return (pgtable_t)pmd_page_vaddr(pmd);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm,
|
static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm)
|
||||||
unsigned long address)
|
|
||||||
{
|
{
|
||||||
return (pte_t *)pte_fragment_alloc(mm, address, 1);
|
return (pte_t *)pte_fragment_alloc(mm, 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline pgtable_t pte_alloc_one(struct mm_struct *mm,
|
static inline pgtable_t pte_alloc_one(struct mm_struct *mm)
|
||||||
unsigned long address)
|
|
||||||
{
|
{
|
||||||
return (pgtable_t)pte_fragment_alloc(mm, address, 0);
|
return (pgtable_t)pte_fragment_alloc(mm, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void pte_free_kernel(struct mm_struct *mm, pte_t *pte)
|
static inline void pte_free_kernel(struct mm_struct *mm, pte_t *pte)
|
||||||
|
@ -79,10 +79,10 @@ static inline void pmd_populate(struct mm_struct *mm, pmd_t *pmdp,
|
|||||||
#define pmd_pgtable(pmd) ((pgtable_t)pmd_page_vaddr(pmd))
|
#define pmd_pgtable(pmd) ((pgtable_t)pmd_page_vaddr(pmd))
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
extern pte_t *pte_alloc_one_kernel(struct mm_struct *mm, unsigned long addr);
|
extern pte_t *pte_alloc_one_kernel(struct mm_struct *mm);
|
||||||
extern pgtable_t pte_alloc_one(struct mm_struct *mm, unsigned long addr);
|
extern pgtable_t pte_alloc_one(struct mm_struct *mm);
|
||||||
void pte_frag_destroy(void *pte_frag);
|
void pte_frag_destroy(void *pte_frag);
|
||||||
pte_t *pte_fragment_alloc(struct mm_struct *mm, unsigned long vmaddr, int kernel);
|
pte_t *pte_fragment_alloc(struct mm_struct *mm, int kernel);
|
||||||
void pte_fragment_free(unsigned long *table, int kernel);
|
void pte_fragment_free(unsigned long *table, int kernel);
|
||||||
|
|
||||||
static inline void pte_free_kernel(struct mm_struct *mm, pte_t *pte)
|
static inline void pte_free_kernel(struct mm_struct *mm, pte_t *pte)
|
||||||
|
@ -93,14 +93,12 @@ static inline void pmd_free(struct mm_struct *mm, pmd_t *pmd)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm,
|
static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm)
|
||||||
unsigned long address)
|
|
||||||
{
|
{
|
||||||
return (pte_t *)__get_free_page(GFP_KERNEL | __GFP_ZERO);
|
return (pte_t *)__get_free_page(GFP_KERNEL | __GFP_ZERO);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline pgtable_t pte_alloc_one(struct mm_struct *mm,
|
static inline pgtable_t pte_alloc_one(struct mm_struct *mm)
|
||||||
unsigned long address)
|
|
||||||
{
|
{
|
||||||
struct page *page;
|
struct page *page;
|
||||||
pte_t *pte;
|
pte_t *pte;
|
||||||
|
@ -95,7 +95,7 @@ static pte_t *__alloc_for_ptecache(struct mm_struct *mm, int kernel)
|
|||||||
return (pte_t *)ret;
|
return (pte_t *)ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
pte_t *pte_fragment_alloc(struct mm_struct *mm, unsigned long vmaddr, int kernel)
|
pte_t *pte_fragment_alloc(struct mm_struct *mm, int kernel)
|
||||||
{
|
{
|
||||||
pte_t *pte;
|
pte_t *pte;
|
||||||
|
|
||||||
|
@ -43,17 +43,17 @@ EXPORT_SYMBOL(ioremap_bot); /* aka VMALLOC_END */
|
|||||||
|
|
||||||
extern char etext[], _stext[], _sinittext[], _einittext[];
|
extern char etext[], _stext[], _sinittext[], _einittext[];
|
||||||
|
|
||||||
__ref pte_t *pte_alloc_one_kernel(struct mm_struct *mm, unsigned long address)
|
__ref pte_t *pte_alloc_one_kernel(struct mm_struct *mm)
|
||||||
{
|
{
|
||||||
if (!slab_is_available())
|
if (!slab_is_available())
|
||||||
return memblock_alloc(PTE_FRAG_SIZE, PTE_FRAG_SIZE);
|
return memblock_alloc(PTE_FRAG_SIZE, PTE_FRAG_SIZE);
|
||||||
|
|
||||||
return (pte_t *)pte_fragment_alloc(mm, address, 1);
|
return (pte_t *)pte_fragment_alloc(mm, 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
pgtable_t pte_alloc_one(struct mm_struct *mm, unsigned long address)
|
pgtable_t pte_alloc_one(struct mm_struct *mm)
|
||||||
{
|
{
|
||||||
return (pgtable_t)pte_fragment_alloc(mm, address, 0);
|
return (pgtable_t)pte_fragment_alloc(mm, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
void __iomem *
|
void __iomem *
|
||||||
|
@ -82,15 +82,13 @@ static inline void pmd_free(struct mm_struct *mm, pmd_t *pmd)
|
|||||||
|
|
||||||
#endif /* __PAGETABLE_PMD_FOLDED */
|
#endif /* __PAGETABLE_PMD_FOLDED */
|
||||||
|
|
||||||
static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm,
|
static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm)
|
||||||
unsigned long address)
|
|
||||||
{
|
{
|
||||||
return (pte_t *)__get_free_page(
|
return (pte_t *)__get_free_page(
|
||||||
GFP_KERNEL | __GFP_RETRY_MAYFAIL | __GFP_ZERO);
|
GFP_KERNEL | __GFP_RETRY_MAYFAIL | __GFP_ZERO);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline struct page *pte_alloc_one(struct mm_struct *mm,
|
static inline struct page *pte_alloc_one(struct mm_struct *mm)
|
||||||
unsigned long address)
|
|
||||||
{
|
{
|
||||||
struct page *pte;
|
struct page *pte;
|
||||||
|
|
||||||
|
@ -397,9 +397,9 @@ static inline int fls64(unsigned long word)
|
|||||||
* This is defined the same way as ffs.
|
* This is defined the same way as ffs.
|
||||||
* Note fls(0) = 0, fls(1) = 1, fls(0x80000000) = 32.
|
* Note fls(0) = 0, fls(1) = 1, fls(0x80000000) = 32.
|
||||||
*/
|
*/
|
||||||
static inline int fls(int word)
|
static inline int fls(unsigned int word)
|
||||||
{
|
{
|
||||||
return fls64((unsigned int)word);
|
return fls64(word);
|
||||||
}
|
}
|
||||||
|
|
||||||
#else /* CONFIG_HAVE_MARCH_Z9_109_FEATURES */
|
#else /* CONFIG_HAVE_MARCH_Z9_109_FEATURES */
|
||||||
|
@ -139,8 +139,8 @@ static inline void pmd_populate(struct mm_struct *mm,
|
|||||||
/*
|
/*
|
||||||
* page table entry allocation/free routines.
|
* page table entry allocation/free routines.
|
||||||
*/
|
*/
|
||||||
#define pte_alloc_one_kernel(mm, vmaddr) ((pte_t *) page_table_alloc(mm))
|
#define pte_alloc_one_kernel(mm) ((pte_t *)page_table_alloc(mm))
|
||||||
#define pte_alloc_one(mm, vmaddr) ((pte_t *) page_table_alloc(mm))
|
#define pte_alloc_one(mm) ((pte_t *)page_table_alloc(mm))
|
||||||
|
|
||||||
#define pte_free_kernel(mm, pte) page_table_free(mm, (unsigned long *) pte)
|
#define pte_free_kernel(mm, pte) page_table_free(mm, (unsigned long *) pte)
|
||||||
#define pte_free(mm, pte) page_table_free(mm, (unsigned long *) pte)
|
#define pte_free(mm, pte) page_table_free(mm, (unsigned long *) pte)
|
||||||
|
@ -32,14 +32,12 @@ static inline void pmd_populate(struct mm_struct *mm, pmd_t *pmd,
|
|||||||
/*
|
/*
|
||||||
* Allocate and free page tables.
|
* Allocate and free page tables.
|
||||||
*/
|
*/
|
||||||
static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm,
|
static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm)
|
||||||
unsigned long address)
|
|
||||||
{
|
{
|
||||||
return quicklist_alloc(QUICK_PT, GFP_KERNEL, NULL);
|
return quicklist_alloc(QUICK_PT, GFP_KERNEL, NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline pgtable_t pte_alloc_one(struct mm_struct *mm,
|
static inline pgtable_t pte_alloc_one(struct mm_struct *mm)
|
||||||
unsigned long address)
|
|
||||||
{
|
{
|
||||||
struct page *page;
|
struct page *page;
|
||||||
void *pg;
|
void *pg;
|
||||||
|
@ -58,10 +58,9 @@ void pmd_populate(struct mm_struct *mm, pmd_t *pmdp, struct page *ptep);
|
|||||||
void pmd_set(pmd_t *pmdp, pte_t *ptep);
|
void pmd_set(pmd_t *pmdp, pte_t *ptep);
|
||||||
#define pmd_populate_kernel(MM, PMD, PTE) pmd_set(PMD, PTE)
|
#define pmd_populate_kernel(MM, PMD, PTE) pmd_set(PMD, PTE)
|
||||||
|
|
||||||
pgtable_t pte_alloc_one(struct mm_struct *mm, unsigned long address);
|
pgtable_t pte_alloc_one(struct mm_struct *mm);
|
||||||
|
|
||||||
static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm,
|
static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm)
|
||||||
unsigned long address)
|
|
||||||
{
|
{
|
||||||
return srmmu_get_nocache(PTE_SIZE, PTE_SIZE);
|
return srmmu_get_nocache(PTE_SIZE, PTE_SIZE);
|
||||||
}
|
}
|
||||||
|
@ -60,10 +60,8 @@ static inline void pmd_free(struct mm_struct *mm, pmd_t *pmd)
|
|||||||
kmem_cache_free(pgtable_cache, pmd);
|
kmem_cache_free(pgtable_cache, pmd);
|
||||||
}
|
}
|
||||||
|
|
||||||
pte_t *pte_alloc_one_kernel(struct mm_struct *mm,
|
pte_t *pte_alloc_one_kernel(struct mm_struct *mm);
|
||||||
unsigned long address);
|
pgtable_t pte_alloc_one(struct mm_struct *mm);
|
||||||
pgtable_t pte_alloc_one(struct mm_struct *mm,
|
|
||||||
unsigned long address);
|
|
||||||
void pte_free_kernel(struct mm_struct *mm, pte_t *pte);
|
void pte_free_kernel(struct mm_struct *mm, pte_t *pte);
|
||||||
void pte_free(struct mm_struct *mm, pgtable_t ptepage);
|
void pte_free(struct mm_struct *mm, pgtable_t ptepage);
|
||||||
|
|
||||||
|
@ -2925,8 +2925,7 @@ void __flush_tlb_all(void)
|
|||||||
: : "r" (pstate));
|
: : "r" (pstate));
|
||||||
}
|
}
|
||||||
|
|
||||||
pte_t *pte_alloc_one_kernel(struct mm_struct *mm,
|
pte_t *pte_alloc_one_kernel(struct mm_struct *mm)
|
||||||
unsigned long address)
|
|
||||||
{
|
{
|
||||||
struct page *page = alloc_page(GFP_KERNEL | __GFP_ZERO);
|
struct page *page = alloc_page(GFP_KERNEL | __GFP_ZERO);
|
||||||
pte_t *pte = NULL;
|
pte_t *pte = NULL;
|
||||||
@ -2937,8 +2936,7 @@ pte_t *pte_alloc_one_kernel(struct mm_struct *mm,
|
|||||||
return pte;
|
return pte;
|
||||||
}
|
}
|
||||||
|
|
||||||
pgtable_t pte_alloc_one(struct mm_struct *mm,
|
pgtable_t pte_alloc_one(struct mm_struct *mm)
|
||||||
unsigned long address)
|
|
||||||
{
|
{
|
||||||
struct page *page = alloc_page(GFP_KERNEL | __GFP_ZERO);
|
struct page *page = alloc_page(GFP_KERNEL | __GFP_ZERO);
|
||||||
if (!page)
|
if (!page)
|
||||||
|
@ -364,12 +364,12 @@ pgd_t *get_pgd_fast(void)
|
|||||||
* Alignments up to the page size are the same for physical and virtual
|
* Alignments up to the page size are the same for physical and virtual
|
||||||
* addresses of the nocache area.
|
* addresses of the nocache area.
|
||||||
*/
|
*/
|
||||||
pgtable_t pte_alloc_one(struct mm_struct *mm, unsigned long address)
|
pgtable_t pte_alloc_one(struct mm_struct *mm)
|
||||||
{
|
{
|
||||||
unsigned long pte;
|
unsigned long pte;
|
||||||
struct page *page;
|
struct page *page;
|
||||||
|
|
||||||
if ((pte = (unsigned long)pte_alloc_one_kernel(mm, address)) == 0)
|
if ((pte = (unsigned long)pte_alloc_one_kernel(mm)) == 0)
|
||||||
return NULL;
|
return NULL;
|
||||||
page = pfn_to_page(__nocache_pa(pte) >> PAGE_SHIFT);
|
page = pfn_to_page(__nocache_pa(pte) >> PAGE_SHIFT);
|
||||||
if (!pgtable_page_ctor(page)) {
|
if (!pgtable_page_ctor(page)) {
|
||||||
|
@ -25,8 +25,8 @@
|
|||||||
extern pgd_t *pgd_alloc(struct mm_struct *);
|
extern pgd_t *pgd_alloc(struct mm_struct *);
|
||||||
extern void pgd_free(struct mm_struct *mm, pgd_t *pgd);
|
extern void pgd_free(struct mm_struct *mm, pgd_t *pgd);
|
||||||
|
|
||||||
extern pte_t *pte_alloc_one_kernel(struct mm_struct *, unsigned long);
|
extern pte_t *pte_alloc_one_kernel(struct mm_struct *);
|
||||||
extern pgtable_t pte_alloc_one(struct mm_struct *, unsigned long);
|
extern pgtable_t pte_alloc_one(struct mm_struct *);
|
||||||
|
|
||||||
static inline void pte_free_kernel(struct mm_struct *mm, pte_t *pte)
|
static inline void pte_free_kernel(struct mm_struct *mm, pte_t *pte)
|
||||||
{
|
{
|
||||||
|
@ -199,7 +199,7 @@ void pgd_free(struct mm_struct *mm, pgd_t *pgd)
|
|||||||
free_page((unsigned long) pgd);
|
free_page((unsigned long) pgd);
|
||||||
}
|
}
|
||||||
|
|
||||||
pte_t *pte_alloc_one_kernel(struct mm_struct *mm, unsigned long address)
|
pte_t *pte_alloc_one_kernel(struct mm_struct *mm)
|
||||||
{
|
{
|
||||||
pte_t *pte;
|
pte_t *pte;
|
||||||
|
|
||||||
@ -207,7 +207,7 @@ pte_t *pte_alloc_one_kernel(struct mm_struct *mm, unsigned long address)
|
|||||||
return pte;
|
return pte;
|
||||||
}
|
}
|
||||||
|
|
||||||
pgtable_t pte_alloc_one(struct mm_struct *mm, unsigned long address)
|
pgtable_t pte_alloc_one(struct mm_struct *mm)
|
||||||
{
|
{
|
||||||
struct page *pte;
|
struct page *pte;
|
||||||
|
|
||||||
|
@ -22,7 +22,7 @@
|
|||||||
* the cntlz instruction for much better code efficiency.
|
* the cntlz instruction for much better code efficiency.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
static inline int fls(int x)
|
static inline int fls(unsigned int x)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
|
@ -34,7 +34,7 @@ extern void free_pgd_slow(struct mm_struct *mm, pgd_t *pgd);
|
|||||||
* Allocate one PTE table.
|
* Allocate one PTE table.
|
||||||
*/
|
*/
|
||||||
static inline pte_t *
|
static inline pte_t *
|
||||||
pte_alloc_one_kernel(struct mm_struct *mm, unsigned long addr)
|
pte_alloc_one_kernel(struct mm_struct *mm)
|
||||||
{
|
{
|
||||||
pte_t *pte;
|
pte_t *pte;
|
||||||
|
|
||||||
@ -46,7 +46,7 @@ pte_alloc_one_kernel(struct mm_struct *mm, unsigned long addr)
|
|||||||
}
|
}
|
||||||
|
|
||||||
static inline pgtable_t
|
static inline pgtable_t
|
||||||
pte_alloc_one(struct mm_struct *mm, unsigned long addr)
|
pte_alloc_one(struct mm_struct *mm)
|
||||||
{
|
{
|
||||||
struct page *pte;
|
struct page *pte;
|
||||||
|
|
||||||
|
@ -172,6 +172,7 @@ config X86
|
|||||||
select HAVE_MEMBLOCK_NODE_MAP
|
select HAVE_MEMBLOCK_NODE_MAP
|
||||||
select HAVE_MIXED_BREAKPOINTS_REGS
|
select HAVE_MIXED_BREAKPOINTS_REGS
|
||||||
select HAVE_MOD_ARCH_SPECIFIC
|
select HAVE_MOD_ARCH_SPECIFIC
|
||||||
|
select HAVE_MOVE_PMD
|
||||||
select HAVE_NMI
|
select HAVE_NMI
|
||||||
select HAVE_OPROFILE
|
select HAVE_OPROFILE
|
||||||
select HAVE_OPTPROBES
|
select HAVE_OPTPROBES
|
||||||
|
@ -448,7 +448,7 @@ static __always_inline int ffs(int x)
|
|||||||
* set bit if value is nonzero. The last (most significant) bit is
|
* set bit if value is nonzero. The last (most significant) bit is
|
||||||
* at position 32.
|
* at position 32.
|
||||||
*/
|
*/
|
||||||
static __always_inline int fls(int x)
|
static __always_inline int fls(unsigned int x)
|
||||||
{
|
{
|
||||||
int r;
|
int r;
|
||||||
|
|
||||||
|
@ -47,8 +47,8 @@ extern gfp_t __userpte_alloc_gfp;
|
|||||||
extern pgd_t *pgd_alloc(struct mm_struct *);
|
extern pgd_t *pgd_alloc(struct mm_struct *);
|
||||||
extern void pgd_free(struct mm_struct *mm, pgd_t *pgd);
|
extern void pgd_free(struct mm_struct *mm, pgd_t *pgd);
|
||||||
|
|
||||||
extern pte_t *pte_alloc_one_kernel(struct mm_struct *, unsigned long);
|
extern pte_t *pte_alloc_one_kernel(struct mm_struct *);
|
||||||
extern pgtable_t pte_alloc_one(struct mm_struct *, unsigned long);
|
extern pgtable_t pte_alloc_one(struct mm_struct *);
|
||||||
|
|
||||||
/* Should really implement gc for free page table pages. This could be
|
/* Should really implement gc for free page table pages. This could be
|
||||||
done with a reference count in struct page. */
|
done with a reference count in struct page. */
|
||||||
|
@ -23,12 +23,12 @@ EXPORT_SYMBOL(physical_mask);
|
|||||||
|
|
||||||
gfp_t __userpte_alloc_gfp = PGALLOC_GFP | PGALLOC_USER_GFP;
|
gfp_t __userpte_alloc_gfp = PGALLOC_GFP | PGALLOC_USER_GFP;
|
||||||
|
|
||||||
pte_t *pte_alloc_one_kernel(struct mm_struct *mm, unsigned long address)
|
pte_t *pte_alloc_one_kernel(struct mm_struct *mm)
|
||||||
{
|
{
|
||||||
return (pte_t *)__get_free_page(PGALLOC_GFP & ~__GFP_ACCOUNT);
|
return (pte_t *)__get_free_page(PGALLOC_GFP & ~__GFP_ACCOUNT);
|
||||||
}
|
}
|
||||||
|
|
||||||
pgtable_t pte_alloc_one(struct mm_struct *mm, unsigned long address)
|
pgtable_t pte_alloc_one(struct mm_struct *mm)
|
||||||
{
|
{
|
||||||
struct page *pte;
|
struct page *pte;
|
||||||
|
|
||||||
|
@ -38,8 +38,7 @@ static inline void pgd_free(struct mm_struct *mm, pgd_t *pgd)
|
|||||||
free_page((unsigned long)pgd);
|
free_page((unsigned long)pgd);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm,
|
static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm)
|
||||||
unsigned long address)
|
|
||||||
{
|
{
|
||||||
pte_t *ptep;
|
pte_t *ptep;
|
||||||
int i;
|
int i;
|
||||||
@ -52,13 +51,12 @@ static inline pte_t *pte_alloc_one_kernel(struct mm_struct *mm,
|
|||||||
return ptep;
|
return ptep;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline pgtable_t pte_alloc_one(struct mm_struct *mm,
|
static inline pgtable_t pte_alloc_one(struct mm_struct *mm)
|
||||||
unsigned long addr)
|
|
||||||
{
|
{
|
||||||
pte_t *pte;
|
pte_t *pte;
|
||||||
struct page *page;
|
struct page *page;
|
||||||
|
|
||||||
pte = pte_alloc_one_kernel(mm, addr);
|
pte = pte_alloc_one_kernel(mm);
|
||||||
if (!pte)
|
if (!pte)
|
||||||
return NULL;
|
return NULL;
|
||||||
page = virt_to_page(pte);
|
page = virt_to_page(pte);
|
||||||
|
@ -26,6 +26,7 @@
|
|||||||
#include <linux/clk/clk-conf.h>
|
#include <linux/clk/clk-conf.h>
|
||||||
#include <linux/limits.h>
|
#include <linux/limits.h>
|
||||||
#include <linux/property.h>
|
#include <linux/property.h>
|
||||||
|
#include <linux/kmemleak.h>
|
||||||
|
|
||||||
#include "base.h"
|
#include "base.h"
|
||||||
#include "power/power.h"
|
#include "power/power.h"
|
||||||
@ -524,6 +525,8 @@ struct platform_device *platform_device_register_full(
|
|||||||
if (!pdev->dev.dma_mask)
|
if (!pdev->dev.dma_mask)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
|
kmemleak_ignore(pdev->dev.dma_mask);
|
||||||
|
|
||||||
*pdev->dev.dma_mask = pdevinfo->dma_mask;
|
*pdev->dev.dma_mask = pdevinfo->dma_mask;
|
||||||
pdev->dev.coherent_dma_mask = pdevinfo->dma_mask;
|
pdev->dev.coherent_dma_mask = pdevinfo->dma_mask;
|
||||||
}
|
}
|
||||||
|
@ -20,7 +20,7 @@ struct udmabuf {
|
|||||||
struct page **pages;
|
struct page **pages;
|
||||||
};
|
};
|
||||||
|
|
||||||
static int udmabuf_vm_fault(struct vm_fault *vmf)
|
static vm_fault_t udmabuf_vm_fault(struct vm_fault *vmf)
|
||||||
{
|
{
|
||||||
struct vm_area_struct *vma = vmf->vma;
|
struct vm_area_struct *vma = vmf->vma;
|
||||||
struct udmabuf *ubuf = vma->vm_private_data;
|
struct udmabuf *ubuf = vma->vm_private_data;
|
||||||
|
@ -333,7 +333,7 @@ int __init firmware_map_add_early(u64 start, u64 end, const char *type)
|
|||||||
{
|
{
|
||||||
struct firmware_map_entry *entry;
|
struct firmware_map_entry *entry;
|
||||||
|
|
||||||
entry = memblock_alloc(sizeof(struct firmware_map_entry),
|
entry = memblock_alloc_nopanic(sizeof(struct firmware_map_entry),
|
||||||
SMP_CACHE_BYTES);
|
SMP_CACHE_BYTES);
|
||||||
if (WARN_ON(!entry))
|
if (WARN_ON(!entry))
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
@ -17,6 +17,7 @@
|
|||||||
#include <linux/writeback.h>
|
#include <linux/writeback.h>
|
||||||
#include <linux/gfp.h>
|
#include <linux/gfp.h>
|
||||||
#include <linux/task_io_accounting_ops.h>
|
#include <linux/task_io_accounting_ops.h>
|
||||||
|
#include <linux/mm.h>
|
||||||
#include "internal.h"
|
#include "internal.h"
|
||||||
|
|
||||||
static int afs_file_mmap(struct file *file, struct vm_area_struct *vma);
|
static int afs_file_mmap(struct file *file, struct vm_area_struct *vma);
|
||||||
@ -441,7 +442,7 @@ static int afs_readpages_one(struct file *file, struct address_space *mapping,
|
|||||||
/* Count the number of contiguous pages at the front of the list. Note
|
/* Count the number of contiguous pages at the front of the list. Note
|
||||||
* that the list goes prev-wards rather than next-wards.
|
* that the list goes prev-wards rather than next-wards.
|
||||||
*/
|
*/
|
||||||
first = list_entry(pages->prev, struct page, lru);
|
first = lru_to_page(pages);
|
||||||
index = first->index + 1;
|
index = first->index + 1;
|
||||||
n = 1;
|
n = 1;
|
||||||
for (p = first->lru.prev; p != pages; p = p->prev) {
|
for (p = first->lru.prev; p != pages; p = p->prev) {
|
||||||
@ -473,7 +474,7 @@ static int afs_readpages_one(struct file *file, struct address_space *mapping,
|
|||||||
* page at the end of the file.
|
* page at the end of the file.
|
||||||
*/
|
*/
|
||||||
do {
|
do {
|
||||||
page = list_entry(pages->prev, struct page, lru);
|
page = lru_to_page(pages);
|
||||||
list_del(&page->lru);
|
list_del(&page->lru);
|
||||||
index = page->index;
|
index = page->index;
|
||||||
if (add_to_page_cache_lru(page, mapping, index,
|
if (add_to_page_cache_lru(page, mapping, index,
|
||||||
|
@ -247,7 +247,7 @@ int afs_wait_for_fs_probes(struct afs_server_list *slist, unsigned long untried)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!still_probing || unlikely(signal_pending(current)))
|
if (!still_probing || signal_pending(current))
|
||||||
goto stop;
|
goto stop;
|
||||||
schedule();
|
schedule();
|
||||||
}
|
}
|
||||||
|
@ -248,7 +248,7 @@ int afs_wait_for_vl_probes(struct afs_vlserver_list *vllist,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!still_probing || unlikely(signal_pending(current)))
|
if (!still_probing || signal_pending(current))
|
||||||
goto stop;
|
goto stop;
|
||||||
schedule();
|
schedule();
|
||||||
}
|
}
|
||||||
|
@ -42,6 +42,8 @@
|
|||||||
#endif
|
#endif
|
||||||
#define pr_fmt(fmt) KBUILD_MODNAME ":pid:%d:%s: " fmt, current->pid, __func__
|
#define pr_fmt(fmt) KBUILD_MODNAME ":pid:%d:%s: " fmt, current->pid, __func__
|
||||||
|
|
||||||
|
extern struct file_system_type autofs_fs_type;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Unified info structure. This is pointed to by both the dentry and
|
* Unified info structure. This is pointed to by both the dentry and
|
||||||
* inode structures. Each file in the filesystem has an instance of this
|
* inode structures. Each file in the filesystem has an instance of this
|
||||||
@ -101,16 +103,19 @@ struct autofs_wait_queue {
|
|||||||
|
|
||||||
#define AUTOFS_SBI_MAGIC 0x6d4a556d
|
#define AUTOFS_SBI_MAGIC 0x6d4a556d
|
||||||
|
|
||||||
|
#define AUTOFS_SBI_CATATONIC 0x0001
|
||||||
|
#define AUTOFS_SBI_STRICTEXPIRE 0x0002
|
||||||
|
|
||||||
struct autofs_sb_info {
|
struct autofs_sb_info {
|
||||||
u32 magic;
|
u32 magic;
|
||||||
int pipefd;
|
int pipefd;
|
||||||
struct file *pipe;
|
struct file *pipe;
|
||||||
struct pid *oz_pgrp;
|
struct pid *oz_pgrp;
|
||||||
int catatonic;
|
|
||||||
int version;
|
int version;
|
||||||
int sub_version;
|
int sub_version;
|
||||||
int min_proto;
|
int min_proto;
|
||||||
int max_proto;
|
int max_proto;
|
||||||
|
unsigned int flags;
|
||||||
unsigned long exp_timeout;
|
unsigned long exp_timeout;
|
||||||
unsigned int type;
|
unsigned int type;
|
||||||
struct super_block *sb;
|
struct super_block *sb;
|
||||||
@ -126,8 +131,7 @@ struct autofs_sb_info {
|
|||||||
|
|
||||||
static inline struct autofs_sb_info *autofs_sbi(struct super_block *sb)
|
static inline struct autofs_sb_info *autofs_sbi(struct super_block *sb)
|
||||||
{
|
{
|
||||||
return sb->s_magic != AUTOFS_SUPER_MAGIC ?
|
return (struct autofs_sb_info *)(sb->s_fs_info);
|
||||||
NULL : (struct autofs_sb_info *)(sb->s_fs_info);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline struct autofs_info *autofs_dentry_ino(struct dentry *dentry)
|
static inline struct autofs_info *autofs_dentry_ino(struct dentry *dentry)
|
||||||
@ -141,7 +145,8 @@ static inline struct autofs_info *autofs_dentry_ino(struct dentry *dentry)
|
|||||||
*/
|
*/
|
||||||
static inline int autofs_oz_mode(struct autofs_sb_info *sbi)
|
static inline int autofs_oz_mode(struct autofs_sb_info *sbi)
|
||||||
{
|
{
|
||||||
return sbi->catatonic || task_pgrp(current) == sbi->oz_pgrp;
|
return ((sbi->flags & AUTOFS_SBI_CATATONIC) ||
|
||||||
|
task_pgrp(current) == sbi->oz_pgrp);
|
||||||
}
|
}
|
||||||
|
|
||||||
struct inode *autofs_get_inode(struct super_block *, umode_t);
|
struct inode *autofs_get_inode(struct super_block *, umode_t);
|
||||||
|
@ -151,22 +151,6 @@ out:
|
|||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* Get the autofs super block info struct from the file opened on
|
|
||||||
* the autofs mount point.
|
|
||||||
*/
|
|
||||||
static struct autofs_sb_info *autofs_dev_ioctl_sbi(struct file *f)
|
|
||||||
{
|
|
||||||
struct autofs_sb_info *sbi = NULL;
|
|
||||||
struct inode *inode;
|
|
||||||
|
|
||||||
if (f) {
|
|
||||||
inode = file_inode(f);
|
|
||||||
sbi = autofs_sbi(inode->i_sb);
|
|
||||||
}
|
|
||||||
return sbi;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Return autofs dev ioctl version */
|
/* Return autofs dev ioctl version */
|
||||||
static int autofs_dev_ioctl_version(struct file *fp,
|
static int autofs_dev_ioctl_version(struct file *fp,
|
||||||
struct autofs_sb_info *sbi,
|
struct autofs_sb_info *sbi,
|
||||||
@ -366,7 +350,7 @@ static int autofs_dev_ioctl_setpipefd(struct file *fp,
|
|||||||
pipefd = param->setpipefd.pipefd;
|
pipefd = param->setpipefd.pipefd;
|
||||||
|
|
||||||
mutex_lock(&sbi->wq_mutex);
|
mutex_lock(&sbi->wq_mutex);
|
||||||
if (!sbi->catatonic) {
|
if (!(sbi->flags & AUTOFS_SBI_CATATONIC)) {
|
||||||
mutex_unlock(&sbi->wq_mutex);
|
mutex_unlock(&sbi->wq_mutex);
|
||||||
return -EBUSY;
|
return -EBUSY;
|
||||||
} else {
|
} else {
|
||||||
@ -393,7 +377,7 @@ static int autofs_dev_ioctl_setpipefd(struct file *fp,
|
|||||||
swap(sbi->oz_pgrp, new_pid);
|
swap(sbi->oz_pgrp, new_pid);
|
||||||
sbi->pipefd = pipefd;
|
sbi->pipefd = pipefd;
|
||||||
sbi->pipe = pipe;
|
sbi->pipe = pipe;
|
||||||
sbi->catatonic = 0;
|
sbi->flags &= ~AUTOFS_SBI_CATATONIC;
|
||||||
}
|
}
|
||||||
out:
|
out:
|
||||||
put_pid(new_pid);
|
put_pid(new_pid);
|
||||||
@ -658,6 +642,8 @@ static int _autofs_dev_ioctl(unsigned int command,
|
|||||||
if (cmd != AUTOFS_DEV_IOCTL_VERSION_CMD &&
|
if (cmd != AUTOFS_DEV_IOCTL_VERSION_CMD &&
|
||||||
cmd != AUTOFS_DEV_IOCTL_OPENMOUNT_CMD &&
|
cmd != AUTOFS_DEV_IOCTL_OPENMOUNT_CMD &&
|
||||||
cmd != AUTOFS_DEV_IOCTL_CLOSEMOUNT_CMD) {
|
cmd != AUTOFS_DEV_IOCTL_CLOSEMOUNT_CMD) {
|
||||||
|
struct super_block *sb;
|
||||||
|
|
||||||
fp = fget(param->ioctlfd);
|
fp = fget(param->ioctlfd);
|
||||||
if (!fp) {
|
if (!fp) {
|
||||||
if (cmd == AUTOFS_DEV_IOCTL_ISMOUNTPOINT_CMD)
|
if (cmd == AUTOFS_DEV_IOCTL_ISMOUNTPOINT_CMD)
|
||||||
@ -666,12 +652,13 @@ static int _autofs_dev_ioctl(unsigned int command,
|
|||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
sbi = autofs_dev_ioctl_sbi(fp);
|
sb = file_inode(fp)->i_sb;
|
||||||
if (!sbi || sbi->magic != AUTOFS_SBI_MAGIC) {
|
if (sb->s_type != &autofs_fs_type) {
|
||||||
err = -EINVAL;
|
err = -EINVAL;
|
||||||
fput(fp);
|
fput(fp);
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
sbi = autofs_sbi(sb);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Admin needs to be able to set the mount catatonic in
|
* Admin needs to be able to set the mount catatonic in
|
||||||
|
@ -16,7 +16,7 @@ static struct dentry *autofs_mount(struct file_system_type *fs_type,
|
|||||||
return mount_nodev(fs_type, flags, data, autofs_fill_super);
|
return mount_nodev(fs_type, flags, data, autofs_fill_super);
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct file_system_type autofs_fs_type = {
|
struct file_system_type autofs_fs_type = {
|
||||||
.owner = THIS_MODULE,
|
.owner = THIS_MODULE,
|
||||||
.name = "autofs",
|
.name = "autofs",
|
||||||
.mount = autofs_mount,
|
.mount = autofs_mount,
|
||||||
|
@ -87,6 +87,8 @@ static int autofs_show_options(struct seq_file *m, struct dentry *root)
|
|||||||
seq_printf(m, ",direct");
|
seq_printf(m, ",direct");
|
||||||
else
|
else
|
||||||
seq_printf(m, ",indirect");
|
seq_printf(m, ",indirect");
|
||||||
|
if (sbi->flags & AUTOFS_SBI_STRICTEXPIRE)
|
||||||
|
seq_printf(m, ",strictexpire");
|
||||||
#ifdef CONFIG_CHECKPOINT_RESTORE
|
#ifdef CONFIG_CHECKPOINT_RESTORE
|
||||||
if (sbi->pipe)
|
if (sbi->pipe)
|
||||||
seq_printf(m, ",pipe_ino=%ld", file_inode(sbi->pipe)->i_ino);
|
seq_printf(m, ",pipe_ino=%ld", file_inode(sbi->pipe)->i_ino);
|
||||||
@ -109,7 +111,7 @@ static const struct super_operations autofs_sops = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
enum {Opt_err, Opt_fd, Opt_uid, Opt_gid, Opt_pgrp, Opt_minproto, Opt_maxproto,
|
enum {Opt_err, Opt_fd, Opt_uid, Opt_gid, Opt_pgrp, Opt_minproto, Opt_maxproto,
|
||||||
Opt_indirect, Opt_direct, Opt_offset};
|
Opt_indirect, Opt_direct, Opt_offset, Opt_strictexpire};
|
||||||
|
|
||||||
static const match_table_t tokens = {
|
static const match_table_t tokens = {
|
||||||
{Opt_fd, "fd=%u"},
|
{Opt_fd, "fd=%u"},
|
||||||
@ -121,24 +123,28 @@ static const match_table_t tokens = {
|
|||||||
{Opt_indirect, "indirect"},
|
{Opt_indirect, "indirect"},
|
||||||
{Opt_direct, "direct"},
|
{Opt_direct, "direct"},
|
||||||
{Opt_offset, "offset"},
|
{Opt_offset, "offset"},
|
||||||
|
{Opt_strictexpire, "strictexpire"},
|
||||||
{Opt_err, NULL}
|
{Opt_err, NULL}
|
||||||
};
|
};
|
||||||
|
|
||||||
static int parse_options(char *options, int *pipefd, kuid_t *uid, kgid_t *gid,
|
static int parse_options(char *options,
|
||||||
int *pgrp, bool *pgrp_set, unsigned int *type,
|
struct inode *root, int *pgrp, bool *pgrp_set,
|
||||||
int *minproto, int *maxproto)
|
struct autofs_sb_info *sbi)
|
||||||
{
|
{
|
||||||
char *p;
|
char *p;
|
||||||
substring_t args[MAX_OPT_ARGS];
|
substring_t args[MAX_OPT_ARGS];
|
||||||
int option;
|
int option;
|
||||||
|
int pipefd = -1;
|
||||||
|
kuid_t uid;
|
||||||
|
kgid_t gid;
|
||||||
|
|
||||||
*uid = current_uid();
|
root->i_uid = current_uid();
|
||||||
*gid = current_gid();
|
root->i_gid = current_gid();
|
||||||
|
|
||||||
*minproto = AUTOFS_MIN_PROTO_VERSION;
|
sbi->min_proto = AUTOFS_MIN_PROTO_VERSION;
|
||||||
*maxproto = AUTOFS_MAX_PROTO_VERSION;
|
sbi->max_proto = AUTOFS_MAX_PROTO_VERSION;
|
||||||
|
|
||||||
*pipefd = -1;
|
sbi->pipefd = -1;
|
||||||
|
|
||||||
if (!options)
|
if (!options)
|
||||||
return 1;
|
return 1;
|
||||||
@ -152,22 +158,25 @@ static int parse_options(char *options, int *pipefd, kuid_t *uid, kgid_t *gid,
|
|||||||
token = match_token(p, tokens, args);
|
token = match_token(p, tokens, args);
|
||||||
switch (token) {
|
switch (token) {
|
||||||
case Opt_fd:
|
case Opt_fd:
|
||||||
if (match_int(args, pipefd))
|
if (match_int(args, &pipefd))
|
||||||
return 1;
|
return 1;
|
||||||
|
sbi->pipefd = pipefd;
|
||||||
break;
|
break;
|
||||||
case Opt_uid:
|
case Opt_uid:
|
||||||
if (match_int(args, &option))
|
if (match_int(args, &option))
|
||||||
return 1;
|
return 1;
|
||||||
*uid = make_kuid(current_user_ns(), option);
|
uid = make_kuid(current_user_ns(), option);
|
||||||
if (!uid_valid(*uid))
|
if (!uid_valid(uid))
|
||||||
return 1;
|
return 1;
|
||||||
|
root->i_uid = uid;
|
||||||
break;
|
break;
|
||||||
case Opt_gid:
|
case Opt_gid:
|
||||||
if (match_int(args, &option))
|
if (match_int(args, &option))
|
||||||
return 1;
|
return 1;
|
||||||
*gid = make_kgid(current_user_ns(), option);
|
gid = make_kgid(current_user_ns(), option);
|
||||||
if (!gid_valid(*gid))
|
if (!gid_valid(gid))
|
||||||
return 1;
|
return 1;
|
||||||
|
root->i_gid = gid;
|
||||||
break;
|
break;
|
||||||
case Opt_pgrp:
|
case Opt_pgrp:
|
||||||
if (match_int(args, &option))
|
if (match_int(args, &option))
|
||||||
@ -178,27 +187,30 @@ static int parse_options(char *options, int *pipefd, kuid_t *uid, kgid_t *gid,
|
|||||||
case Opt_minproto:
|
case Opt_minproto:
|
||||||
if (match_int(args, &option))
|
if (match_int(args, &option))
|
||||||
return 1;
|
return 1;
|
||||||
*minproto = option;
|
sbi->min_proto = option;
|
||||||
break;
|
break;
|
||||||
case Opt_maxproto:
|
case Opt_maxproto:
|
||||||
if (match_int(args, &option))
|
if (match_int(args, &option))
|
||||||
return 1;
|
return 1;
|
||||||
*maxproto = option;
|
sbi->max_proto = option;
|
||||||
break;
|
break;
|
||||||
case Opt_indirect:
|
case Opt_indirect:
|
||||||
set_autofs_type_indirect(type);
|
set_autofs_type_indirect(&sbi->type);
|
||||||
break;
|
break;
|
||||||
case Opt_direct:
|
case Opt_direct:
|
||||||
set_autofs_type_direct(type);
|
set_autofs_type_direct(&sbi->type);
|
||||||
break;
|
break;
|
||||||
case Opt_offset:
|
case Opt_offset:
|
||||||
set_autofs_type_offset(type);
|
set_autofs_type_offset(&sbi->type);
|
||||||
|
break;
|
||||||
|
case Opt_strictexpire:
|
||||||
|
sbi->flags |= AUTOFS_SBI_STRICTEXPIRE;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return (*pipefd < 0);
|
return (sbi->pipefd < 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
int autofs_fill_super(struct super_block *s, void *data, int silent)
|
int autofs_fill_super(struct super_block *s, void *data, int silent)
|
||||||
@ -206,7 +218,6 @@ int autofs_fill_super(struct super_block *s, void *data, int silent)
|
|||||||
struct inode *root_inode;
|
struct inode *root_inode;
|
||||||
struct dentry *root;
|
struct dentry *root;
|
||||||
struct file *pipe;
|
struct file *pipe;
|
||||||
int pipefd;
|
|
||||||
struct autofs_sb_info *sbi;
|
struct autofs_sb_info *sbi;
|
||||||
struct autofs_info *ino;
|
struct autofs_info *ino;
|
||||||
int pgrp = 0;
|
int pgrp = 0;
|
||||||
@ -222,12 +233,12 @@ int autofs_fill_super(struct super_block *s, void *data, int silent)
|
|||||||
sbi->magic = AUTOFS_SBI_MAGIC;
|
sbi->magic = AUTOFS_SBI_MAGIC;
|
||||||
sbi->pipefd = -1;
|
sbi->pipefd = -1;
|
||||||
sbi->pipe = NULL;
|
sbi->pipe = NULL;
|
||||||
sbi->catatonic = 1;
|
|
||||||
sbi->exp_timeout = 0;
|
sbi->exp_timeout = 0;
|
||||||
sbi->oz_pgrp = NULL;
|
sbi->oz_pgrp = NULL;
|
||||||
sbi->sb = s;
|
sbi->sb = s;
|
||||||
sbi->version = 0;
|
sbi->version = 0;
|
||||||
sbi->sub_version = 0;
|
sbi->sub_version = 0;
|
||||||
|
sbi->flags = AUTOFS_SBI_CATATONIC;
|
||||||
set_autofs_type_indirect(&sbi->type);
|
set_autofs_type_indirect(&sbi->type);
|
||||||
sbi->min_proto = 0;
|
sbi->min_proto = 0;
|
||||||
sbi->max_proto = 0;
|
sbi->max_proto = 0;
|
||||||
@ -262,9 +273,7 @@ int autofs_fill_super(struct super_block *s, void *data, int silent)
|
|||||||
root->d_fsdata = ino;
|
root->d_fsdata = ino;
|
||||||
|
|
||||||
/* Can this call block? */
|
/* Can this call block? */
|
||||||
if (parse_options(data, &pipefd, &root_inode->i_uid, &root_inode->i_gid,
|
if (parse_options(data, root_inode, &pgrp, &pgrp_set, sbi)) {
|
||||||
&pgrp, &pgrp_set, &sbi->type, &sbi->min_proto,
|
|
||||||
&sbi->max_proto)) {
|
|
||||||
pr_err("called with bogus options\n");
|
pr_err("called with bogus options\n");
|
||||||
goto fail_dput;
|
goto fail_dput;
|
||||||
}
|
}
|
||||||
@ -303,8 +312,9 @@ int autofs_fill_super(struct super_block *s, void *data, int silent)
|
|||||||
root_inode->i_fop = &autofs_root_operations;
|
root_inode->i_fop = &autofs_root_operations;
|
||||||
root_inode->i_op = &autofs_dir_inode_operations;
|
root_inode->i_op = &autofs_dir_inode_operations;
|
||||||
|
|
||||||
pr_debug("pipe fd = %d, pgrp = %u\n", pipefd, pid_nr(sbi->oz_pgrp));
|
pr_debug("pipe fd = %d, pgrp = %u\n",
|
||||||
pipe = fget(pipefd);
|
sbi->pipefd, pid_nr(sbi->oz_pgrp));
|
||||||
|
pipe = fget(sbi->pipefd);
|
||||||
|
|
||||||
if (!pipe) {
|
if (!pipe) {
|
||||||
pr_err("could not open pipe file descriptor\n");
|
pr_err("could not open pipe file descriptor\n");
|
||||||
@ -314,8 +324,7 @@ int autofs_fill_super(struct super_block *s, void *data, int silent)
|
|||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
goto fail_fput;
|
goto fail_fput;
|
||||||
sbi->pipe = pipe;
|
sbi->pipe = pipe;
|
||||||
sbi->pipefd = pipefd;
|
sbi->flags &= ~AUTOFS_SBI_CATATONIC;
|
||||||
sbi->catatonic = 0;
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Success! Install the root dentry now to indicate completion.
|
* Success! Install the root dentry now to indicate completion.
|
||||||
|
@ -275,8 +275,11 @@ static int autofs_mount_wait(const struct path *path, bool rcu_walk)
|
|||||||
pr_debug("waiting for mount name=%pd\n", path->dentry);
|
pr_debug("waiting for mount name=%pd\n", path->dentry);
|
||||||
status = autofs_wait(sbi, path, NFY_MOUNT);
|
status = autofs_wait(sbi, path, NFY_MOUNT);
|
||||||
pr_debug("mount wait done status=%d\n", status);
|
pr_debug("mount wait done status=%d\n", status);
|
||||||
|
ino->last_used = jiffies;
|
||||||
|
return status;
|
||||||
}
|
}
|
||||||
ino->last_used = jiffies;
|
if (!(sbi->flags & AUTOFS_SBI_STRICTEXPIRE))
|
||||||
|
ino->last_used = jiffies;
|
||||||
return status;
|
return status;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -510,7 +513,8 @@ static struct dentry *autofs_lookup(struct inode *dir,
|
|||||||
sbi = autofs_sbi(dir->i_sb);
|
sbi = autofs_sbi(dir->i_sb);
|
||||||
|
|
||||||
pr_debug("pid = %u, pgrp = %u, catatonic = %d, oz_mode = %d\n",
|
pr_debug("pid = %u, pgrp = %u, catatonic = %d, oz_mode = %d\n",
|
||||||
current->pid, task_pgrp_nr(current), sbi->catatonic,
|
current->pid, task_pgrp_nr(current),
|
||||||
|
sbi->flags & AUTOFS_SBI_CATATONIC,
|
||||||
autofs_oz_mode(sbi));
|
autofs_oz_mode(sbi));
|
||||||
|
|
||||||
active = autofs_lookup_active(dentry);
|
active = autofs_lookup_active(dentry);
|
||||||
@ -563,7 +567,7 @@ static int autofs_dir_symlink(struct inode *dir,
|
|||||||
* autofs mount is catatonic but the state of an autofs
|
* autofs mount is catatonic but the state of an autofs
|
||||||
* file system needs to be preserved over restarts.
|
* file system needs to be preserved over restarts.
|
||||||
*/
|
*/
|
||||||
if (sbi->catatonic)
|
if (sbi->flags & AUTOFS_SBI_CATATONIC)
|
||||||
return -EACCES;
|
return -EACCES;
|
||||||
|
|
||||||
BUG_ON(!ino);
|
BUG_ON(!ino);
|
||||||
@ -626,7 +630,7 @@ static int autofs_dir_unlink(struct inode *dir, struct dentry *dentry)
|
|||||||
* autofs mount is catatonic but the state of an autofs
|
* autofs mount is catatonic but the state of an autofs
|
||||||
* file system needs to be preserved over restarts.
|
* file system needs to be preserved over restarts.
|
||||||
*/
|
*/
|
||||||
if (sbi->catatonic)
|
if (sbi->flags & AUTOFS_SBI_CATATONIC)
|
||||||
return -EACCES;
|
return -EACCES;
|
||||||
|
|
||||||
if (atomic_dec_and_test(&ino->count)) {
|
if (atomic_dec_and_test(&ino->count)) {
|
||||||
@ -714,7 +718,7 @@ static int autofs_dir_rmdir(struct inode *dir, struct dentry *dentry)
|
|||||||
* autofs mount is catatonic but the state of an autofs
|
* autofs mount is catatonic but the state of an autofs
|
||||||
* file system needs to be preserved over restarts.
|
* file system needs to be preserved over restarts.
|
||||||
*/
|
*/
|
||||||
if (sbi->catatonic)
|
if (sbi->flags & AUTOFS_SBI_CATATONIC)
|
||||||
return -EACCES;
|
return -EACCES;
|
||||||
|
|
||||||
spin_lock(&sbi->lookup_lock);
|
spin_lock(&sbi->lookup_lock);
|
||||||
@ -759,7 +763,7 @@ static int autofs_dir_mkdir(struct inode *dir,
|
|||||||
* autofs mount is catatonic but the state of an autofs
|
* autofs mount is catatonic but the state of an autofs
|
||||||
* file system needs to be preserved over restarts.
|
* file system needs to be preserved over restarts.
|
||||||
*/
|
*/
|
||||||
if (sbi->catatonic)
|
if (sbi->flags & AUTOFS_SBI_CATATONIC)
|
||||||
return -EACCES;
|
return -EACCES;
|
||||||
|
|
||||||
pr_debug("dentry %p, creating %pd\n", dentry, dentry);
|
pr_debug("dentry %p, creating %pd\n", dentry, dentry);
|
||||||
|
@ -20,14 +20,14 @@ void autofs_catatonic_mode(struct autofs_sb_info *sbi)
|
|||||||
struct autofs_wait_queue *wq, *nwq;
|
struct autofs_wait_queue *wq, *nwq;
|
||||||
|
|
||||||
mutex_lock(&sbi->wq_mutex);
|
mutex_lock(&sbi->wq_mutex);
|
||||||
if (sbi->catatonic) {
|
if (sbi->flags & AUTOFS_SBI_CATATONIC) {
|
||||||
mutex_unlock(&sbi->wq_mutex);
|
mutex_unlock(&sbi->wq_mutex);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
pr_debug("entering catatonic mode\n");
|
pr_debug("entering catatonic mode\n");
|
||||||
|
|
||||||
sbi->catatonic = 1;
|
sbi->flags |= AUTOFS_SBI_CATATONIC;
|
||||||
wq = sbi->queues;
|
wq = sbi->queues;
|
||||||
sbi->queues = NULL; /* Erase all wait queues */
|
sbi->queues = NULL; /* Erase all wait queues */
|
||||||
while (wq) {
|
while (wq) {
|
||||||
@ -255,7 +255,7 @@ static int validate_request(struct autofs_wait_queue **wait,
|
|||||||
struct autofs_wait_queue *wq;
|
struct autofs_wait_queue *wq;
|
||||||
struct autofs_info *ino;
|
struct autofs_info *ino;
|
||||||
|
|
||||||
if (sbi->catatonic)
|
if (sbi->flags & AUTOFS_SBI_CATATONIC)
|
||||||
return -ENOENT;
|
return -ENOENT;
|
||||||
|
|
||||||
/* Wait in progress, continue; */
|
/* Wait in progress, continue; */
|
||||||
@ -290,7 +290,7 @@ static int validate_request(struct autofs_wait_queue **wait,
|
|||||||
if (mutex_lock_interruptible(&sbi->wq_mutex))
|
if (mutex_lock_interruptible(&sbi->wq_mutex))
|
||||||
return -EINTR;
|
return -EINTR;
|
||||||
|
|
||||||
if (sbi->catatonic)
|
if (sbi->flags & AUTOFS_SBI_CATATONIC)
|
||||||
return -ENOENT;
|
return -ENOENT;
|
||||||
|
|
||||||
wq = autofs_find_wait(sbi, qstr);
|
wq = autofs_find_wait(sbi, qstr);
|
||||||
@ -359,7 +359,7 @@ int autofs_wait(struct autofs_sb_info *sbi,
|
|||||||
pid_t tgid;
|
pid_t tgid;
|
||||||
|
|
||||||
/* In catatonic mode, we don't wait for nobody */
|
/* In catatonic mode, we don't wait for nobody */
|
||||||
if (sbi->catatonic)
|
if (sbi->flags & AUTOFS_SBI_CATATONIC)
|
||||||
return -ENOENT;
|
return -ENOENT;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
11
fs/bfs/bfs.h
11
fs/bfs/bfs.h
@ -1,13 +1,20 @@
|
|||||||
/* SPDX-License-Identifier: GPL-2.0 */
|
/* SPDX-License-Identifier: GPL-2.0 */
|
||||||
/*
|
/*
|
||||||
* fs/bfs/bfs.h
|
* fs/bfs/bfs.h
|
||||||
* Copyright (C) 1999 Tigran Aivazian <tigran@veritas.com>
|
* Copyright (C) 1999-2018 Tigran Aivazian <aivazian.tigran@gmail.com>
|
||||||
*/
|
*/
|
||||||
#ifndef _FS_BFS_BFS_H
|
#ifndef _FS_BFS_BFS_H
|
||||||
#define _FS_BFS_BFS_H
|
#define _FS_BFS_BFS_H
|
||||||
|
|
||||||
#include <linux/bfs_fs.h>
|
#include <linux/bfs_fs.h>
|
||||||
|
|
||||||
|
/* In theory BFS supports up to 512 inodes, numbered from 2 (for /) up to 513 inclusive.
|
||||||
|
In actual fact, attempting to create the 512th inode (i.e. inode No. 513 or file No. 511)
|
||||||
|
will fail with ENOSPC in bfs_add_entry(): the root directory cannot contain so many entries, counting '..'.
|
||||||
|
So, mkfs.bfs(8) should really limit its -N option to 511 and not 512. For now, we just print a warning
|
||||||
|
if a filesystem is mounted with such "impossible to fill up" number of inodes */
|
||||||
|
#define BFS_MAX_LASTI 513
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* BFS file system in-core superblock info
|
* BFS file system in-core superblock info
|
||||||
*/
|
*/
|
||||||
@ -17,7 +24,7 @@ struct bfs_sb_info {
|
|||||||
unsigned long si_freei;
|
unsigned long si_freei;
|
||||||
unsigned long si_lf_eblk;
|
unsigned long si_lf_eblk;
|
||||||
unsigned long si_lasti;
|
unsigned long si_lasti;
|
||||||
unsigned long *si_imap;
|
DECLARE_BITMAP(si_imap, BFS_MAX_LASTI+1);
|
||||||
struct mutex bfs_lock;
|
struct mutex bfs_lock;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -2,8 +2,8 @@
|
|||||||
/*
|
/*
|
||||||
* fs/bfs/dir.c
|
* fs/bfs/dir.c
|
||||||
* BFS directory operations.
|
* BFS directory operations.
|
||||||
* Copyright (C) 1999,2000 Tigran Aivazian <tigran@veritas.com>
|
* Copyright (C) 1999-2018 Tigran Aivazian <aivazian.tigran@gmail.com>
|
||||||
* Made endianness-clean by Andrew Stribblehill <ads@wompom.org> 2005
|
* Made endianness-clean by Andrew Stribblehill <ads@wompom.org> 2005
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <linux/time.h>
|
#include <linux/time.h>
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
/*
|
/*
|
||||||
* fs/bfs/file.c
|
* fs/bfs/file.c
|
||||||
* BFS file operations.
|
* BFS file operations.
|
||||||
* Copyright (C) 1999,2000 Tigran Aivazian <tigran@veritas.com>
|
* Copyright (C) 1999-2018 Tigran Aivazian <aivazian.tigran@gmail.com>
|
||||||
*
|
*
|
||||||
* Make the file block allocation algorithm understand the size
|
* Make the file block allocation algorithm understand the size
|
||||||
* of the underlying block device.
|
* of the underlying block device.
|
||||||
|
@ -1,10 +1,9 @@
|
|||||||
/*
|
/*
|
||||||
* fs/bfs/inode.c
|
* fs/bfs/inode.c
|
||||||
* BFS superblock and inode operations.
|
* BFS superblock and inode operations.
|
||||||
* Copyright (C) 1999-2006 Tigran Aivazian <aivazian.tigran@gmail.com>
|
* Copyright (C) 1999-2018 Tigran Aivazian <aivazian.tigran@gmail.com>
|
||||||
* From fs/minix, Copyright (C) 1991, 1992 Linus Torvalds.
|
* From fs/minix, Copyright (C) 1991, 1992 Linus Torvalds.
|
||||||
*
|
* Made endianness-clean by Andrew Stribblehill <ads@wompom.org>, 2005.
|
||||||
* Made endianness-clean by Andrew Stribblehill <ads@wompom.org>, 2005.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <linux/module.h>
|
#include <linux/module.h>
|
||||||
@ -118,12 +117,12 @@ static int bfs_write_inode(struct inode *inode, struct writeback_control *wbc)
|
|||||||
{
|
{
|
||||||
struct bfs_sb_info *info = BFS_SB(inode->i_sb);
|
struct bfs_sb_info *info = BFS_SB(inode->i_sb);
|
||||||
unsigned int ino = (u16)inode->i_ino;
|
unsigned int ino = (u16)inode->i_ino;
|
||||||
unsigned long i_sblock;
|
unsigned long i_sblock;
|
||||||
struct bfs_inode *di;
|
struct bfs_inode *di;
|
||||||
struct buffer_head *bh;
|
struct buffer_head *bh;
|
||||||
int err = 0;
|
int err = 0;
|
||||||
|
|
||||||
dprintf("ino=%08x\n", ino);
|
dprintf("ino=%08x\n", ino);
|
||||||
|
|
||||||
di = find_inode(inode->i_sb, ino, &bh);
|
di = find_inode(inode->i_sb, ino, &bh);
|
||||||
if (IS_ERR(di))
|
if (IS_ERR(di))
|
||||||
@ -144,7 +143,7 @@ static int bfs_write_inode(struct inode *inode, struct writeback_control *wbc)
|
|||||||
di->i_atime = cpu_to_le32(inode->i_atime.tv_sec);
|
di->i_atime = cpu_to_le32(inode->i_atime.tv_sec);
|
||||||
di->i_mtime = cpu_to_le32(inode->i_mtime.tv_sec);
|
di->i_mtime = cpu_to_le32(inode->i_mtime.tv_sec);
|
||||||
di->i_ctime = cpu_to_le32(inode->i_ctime.tv_sec);
|
di->i_ctime = cpu_to_le32(inode->i_ctime.tv_sec);
|
||||||
i_sblock = BFS_I(inode)->i_sblock;
|
i_sblock = BFS_I(inode)->i_sblock;
|
||||||
di->i_sblock = cpu_to_le32(i_sblock);
|
di->i_sblock = cpu_to_le32(i_sblock);
|
||||||
di->i_eblock = cpu_to_le32(BFS_I(inode)->i_eblock);
|
di->i_eblock = cpu_to_le32(BFS_I(inode)->i_eblock);
|
||||||
di->i_eoffset = cpu_to_le32(i_sblock * BFS_BSIZE + inode->i_size - 1);
|
di->i_eoffset = cpu_to_le32(i_sblock * BFS_BSIZE + inode->i_size - 1);
|
||||||
@ -188,13 +187,13 @@ static void bfs_evict_inode(struct inode *inode)
|
|||||||
mark_buffer_dirty(bh);
|
mark_buffer_dirty(bh);
|
||||||
brelse(bh);
|
brelse(bh);
|
||||||
|
|
||||||
if (bi->i_dsk_ino) {
|
if (bi->i_dsk_ino) {
|
||||||
if (bi->i_sblock)
|
if (bi->i_sblock)
|
||||||
info->si_freeb += bi->i_eblock + 1 - bi->i_sblock;
|
info->si_freeb += bi->i_eblock + 1 - bi->i_sblock;
|
||||||
info->si_freei++;
|
info->si_freei++;
|
||||||
clear_bit(ino, info->si_imap);
|
clear_bit(ino, info->si_imap);
|
||||||
bfs_dump_imap("delete_inode", s);
|
bfs_dump_imap("evict_inode", s);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If this was the last file, make the previous block
|
* If this was the last file, make the previous block
|
||||||
@ -214,7 +213,6 @@ static void bfs_put_super(struct super_block *s)
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
mutex_destroy(&info->bfs_lock);
|
mutex_destroy(&info->bfs_lock);
|
||||||
kfree(info->si_imap);
|
|
||||||
kfree(info);
|
kfree(info);
|
||||||
s->s_fs_info = NULL;
|
s->s_fs_info = NULL;
|
||||||
}
|
}
|
||||||
@ -311,8 +309,7 @@ void bfs_dump_imap(const char *prefix, struct super_block *s)
|
|||||||
else
|
else
|
||||||
strcat(tmpbuf, "0");
|
strcat(tmpbuf, "0");
|
||||||
}
|
}
|
||||||
printf("BFS-fs: %s: lasti=%08lx <%s>\n",
|
printf("%s: lasti=%08lx <%s>\n", prefix, BFS_SB(s)->si_lasti, tmpbuf);
|
||||||
prefix, BFS_SB(s)->si_lasti, tmpbuf);
|
|
||||||
free_page((unsigned long)tmpbuf);
|
free_page((unsigned long)tmpbuf);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
@ -322,7 +319,7 @@ static int bfs_fill_super(struct super_block *s, void *data, int silent)
|
|||||||
struct buffer_head *bh, *sbh;
|
struct buffer_head *bh, *sbh;
|
||||||
struct bfs_super_block *bfs_sb;
|
struct bfs_super_block *bfs_sb;
|
||||||
struct inode *inode;
|
struct inode *inode;
|
||||||
unsigned i, imap_len;
|
unsigned i;
|
||||||
struct bfs_sb_info *info;
|
struct bfs_sb_info *info;
|
||||||
int ret = -EINVAL;
|
int ret = -EINVAL;
|
||||||
unsigned long i_sblock, i_eblock, i_eoff, s_size;
|
unsigned long i_sblock, i_eblock, i_eoff, s_size;
|
||||||
@ -341,8 +338,7 @@ static int bfs_fill_super(struct super_block *s, void *data, int silent)
|
|||||||
bfs_sb = (struct bfs_super_block *)sbh->b_data;
|
bfs_sb = (struct bfs_super_block *)sbh->b_data;
|
||||||
if (le32_to_cpu(bfs_sb->s_magic) != BFS_MAGIC) {
|
if (le32_to_cpu(bfs_sb->s_magic) != BFS_MAGIC) {
|
||||||
if (!silent)
|
if (!silent)
|
||||||
printf("No BFS filesystem on %s (magic=%08x)\n",
|
printf("No BFS filesystem on %s (magic=%08x)\n", s->s_id, le32_to_cpu(bfs_sb->s_magic));
|
||||||
s->s_id, le32_to_cpu(bfs_sb->s_magic));
|
|
||||||
goto out1;
|
goto out1;
|
||||||
}
|
}
|
||||||
if (BFS_UNCLEAN(bfs_sb, s) && !silent)
|
if (BFS_UNCLEAN(bfs_sb, s) && !silent)
|
||||||
@ -351,18 +347,16 @@ static int bfs_fill_super(struct super_block *s, void *data, int silent)
|
|||||||
s->s_magic = BFS_MAGIC;
|
s->s_magic = BFS_MAGIC;
|
||||||
|
|
||||||
if (le32_to_cpu(bfs_sb->s_start) > le32_to_cpu(bfs_sb->s_end) ||
|
if (le32_to_cpu(bfs_sb->s_start) > le32_to_cpu(bfs_sb->s_end) ||
|
||||||
le32_to_cpu(bfs_sb->s_start) < BFS_BSIZE) {
|
le32_to_cpu(bfs_sb->s_start) < sizeof(struct bfs_super_block) + sizeof(struct bfs_dirent)) {
|
||||||
printf("Superblock is corrupted\n");
|
printf("Superblock is corrupted on %s\n", s->s_id);
|
||||||
goto out1;
|
goto out1;
|
||||||
}
|
}
|
||||||
|
|
||||||
info->si_lasti = (le32_to_cpu(bfs_sb->s_start) - BFS_BSIZE) /
|
info->si_lasti = (le32_to_cpu(bfs_sb->s_start) - BFS_BSIZE) / sizeof(struct bfs_inode) + BFS_ROOT_INO - 1;
|
||||||
sizeof(struct bfs_inode)
|
if (info->si_lasti == BFS_MAX_LASTI)
|
||||||
+ BFS_ROOT_INO - 1;
|
printf("WARNING: filesystem %s was created with 512 inodes, the real maximum is 511, mounting anyway\n", s->s_id);
|
||||||
imap_len = (info->si_lasti / 8) + 1;
|
else if (info->si_lasti > BFS_MAX_LASTI) {
|
||||||
info->si_imap = kzalloc(imap_len, GFP_KERNEL | __GFP_NOWARN);
|
printf("Impossible last inode number %lu > %d on %s\n", info->si_lasti, BFS_MAX_LASTI, s->s_id);
|
||||||
if (!info->si_imap) {
|
|
||||||
printf("Cannot allocate %u bytes\n", imap_len);
|
|
||||||
goto out1;
|
goto out1;
|
||||||
}
|
}
|
||||||
for (i = 0; i < BFS_ROOT_INO; i++)
|
for (i = 0; i < BFS_ROOT_INO; i++)
|
||||||
@ -372,26 +366,25 @@ static int bfs_fill_super(struct super_block *s, void *data, int silent)
|
|||||||
inode = bfs_iget(s, BFS_ROOT_INO);
|
inode = bfs_iget(s, BFS_ROOT_INO);
|
||||||
if (IS_ERR(inode)) {
|
if (IS_ERR(inode)) {
|
||||||
ret = PTR_ERR(inode);
|
ret = PTR_ERR(inode);
|
||||||
goto out2;
|
goto out1;
|
||||||
}
|
}
|
||||||
s->s_root = d_make_root(inode);
|
s->s_root = d_make_root(inode);
|
||||||
if (!s->s_root) {
|
if (!s->s_root) {
|
||||||
ret = -ENOMEM;
|
ret = -ENOMEM;
|
||||||
goto out2;
|
goto out1;
|
||||||
}
|
}
|
||||||
|
|
||||||
info->si_blocks = (le32_to_cpu(bfs_sb->s_end) + 1) >> BFS_BSIZE_BITS;
|
info->si_blocks = (le32_to_cpu(bfs_sb->s_end) + 1) >> BFS_BSIZE_BITS;
|
||||||
info->si_freeb = (le32_to_cpu(bfs_sb->s_end) + 1
|
info->si_freeb = (le32_to_cpu(bfs_sb->s_end) + 1 - le32_to_cpu(bfs_sb->s_start)) >> BFS_BSIZE_BITS;
|
||||||
- le32_to_cpu(bfs_sb->s_start)) >> BFS_BSIZE_BITS;
|
|
||||||
info->si_freei = 0;
|
info->si_freei = 0;
|
||||||
info->si_lf_eblk = 0;
|
info->si_lf_eblk = 0;
|
||||||
|
|
||||||
/* can we read the last block? */
|
/* can we read the last block? */
|
||||||
bh = sb_bread(s, info->si_blocks - 1);
|
bh = sb_bread(s, info->si_blocks - 1);
|
||||||
if (!bh) {
|
if (!bh) {
|
||||||
printf("Last block not available: %lu\n", info->si_blocks - 1);
|
printf("Last block not available on %s: %lu\n", s->s_id, info->si_blocks - 1);
|
||||||
ret = -EIO;
|
ret = -EIO;
|
||||||
goto out3;
|
goto out2;
|
||||||
}
|
}
|
||||||
brelse(bh);
|
brelse(bh);
|
||||||
|
|
||||||
@ -425,11 +418,11 @@ static int bfs_fill_super(struct super_block *s, void *data, int silent)
|
|||||||
(i_eoff != le32_to_cpu(-1) && i_eoff > s_size) ||
|
(i_eoff != le32_to_cpu(-1) && i_eoff > s_size) ||
|
||||||
i_sblock * BFS_BSIZE > i_eoff) {
|
i_sblock * BFS_BSIZE > i_eoff) {
|
||||||
|
|
||||||
printf("Inode 0x%08x corrupted\n", i);
|
printf("Inode 0x%08x corrupted on %s\n", i, s->s_id);
|
||||||
|
|
||||||
brelse(bh);
|
brelse(bh);
|
||||||
ret = -EIO;
|
ret = -EIO;
|
||||||
goto out3;
|
goto out2;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!di->i_ino) {
|
if (!di->i_ino) {
|
||||||
@ -445,14 +438,12 @@ static int bfs_fill_super(struct super_block *s, void *data, int silent)
|
|||||||
}
|
}
|
||||||
brelse(bh);
|
brelse(bh);
|
||||||
brelse(sbh);
|
brelse(sbh);
|
||||||
bfs_dump_imap("read_super", s);
|
bfs_dump_imap("fill_super", s);
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
out3:
|
out2:
|
||||||
dput(s->s_root);
|
dput(s->s_root);
|
||||||
s->s_root = NULL;
|
s->s_root = NULL;
|
||||||
out2:
|
|
||||||
kfree(info->si_imap);
|
|
||||||
out1:
|
out1:
|
||||||
brelse(sbh);
|
brelse(sbh);
|
||||||
out:
|
out:
|
||||||
@ -482,7 +473,7 @@ static int __init init_bfs_fs(void)
|
|||||||
int err = init_inodecache();
|
int err = init_inodecache();
|
||||||
if (err)
|
if (err)
|
||||||
goto out1;
|
goto out1;
|
||||||
err = register_filesystem(&bfs_fs_type);
|
err = register_filesystem(&bfs_fs_type);
|
||||||
if (err)
|
if (err)
|
||||||
goto out;
|
goto out;
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -42,10 +42,14 @@ static int load_script(struct linux_binprm *bprm)
|
|||||||
fput(bprm->file);
|
fput(bprm->file);
|
||||||
bprm->file = NULL;
|
bprm->file = NULL;
|
||||||
|
|
||||||
bprm->buf[BINPRM_BUF_SIZE - 1] = '\0';
|
for (cp = bprm->buf+2;; cp++) {
|
||||||
if ((cp = strchr(bprm->buf, '\n')) == NULL)
|
if (cp >= bprm->buf + BINPRM_BUF_SIZE)
|
||||||
cp = bprm->buf+BINPRM_BUF_SIZE-1;
|
return -ENOEXEC;
|
||||||
|
if (!*cp || (*cp == '\n'))
|
||||||
|
break;
|
||||||
|
}
|
||||||
*cp = '\0';
|
*cp = '\0';
|
||||||
|
|
||||||
while (cp > bprm->buf) {
|
while (cp > bprm->buf) {
|
||||||
cp--;
|
cp--;
|
||||||
if ((*cp == ' ') || (*cp == '\t'))
|
if ((*cp == ' ') || (*cp == '\t'))
|
||||||
|
@ -4103,8 +4103,7 @@ int extent_readpages(struct address_space *mapping, struct list_head *pages,
|
|||||||
|
|
||||||
while (!list_empty(pages)) {
|
while (!list_empty(pages)) {
|
||||||
for (nr = 0; nr < ARRAY_SIZE(pagepool) && !list_empty(pages);) {
|
for (nr = 0; nr < ARRAY_SIZE(pagepool) && !list_empty(pages);) {
|
||||||
struct page *page = list_entry(pages->prev,
|
struct page *page = lru_to_page(pages);
|
||||||
struct page, lru);
|
|
||||||
|
|
||||||
prefetchw(&page->flags);
|
prefetchw(&page->flags);
|
||||||
list_del(&page->lru);
|
list_del(&page->lru);
|
||||||
|
@ -2366,7 +2366,7 @@ static int cont_expand_zero(struct file *file, struct address_space *mapping,
|
|||||||
|
|
||||||
balance_dirty_pages_ratelimited(mapping);
|
balance_dirty_pages_ratelimited(mapping);
|
||||||
|
|
||||||
if (unlikely(fatal_signal_pending(current))) {
|
if (fatal_signal_pending(current)) {
|
||||||
err = -EINTR;
|
err = -EINTR;
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
@ -306,7 +306,7 @@ static int start_read(struct inode *inode, struct ceph_rw_context *rw_ctx,
|
|||||||
struct ceph_osd_client *osdc =
|
struct ceph_osd_client *osdc =
|
||||||
&ceph_inode_to_client(inode)->client->osdc;
|
&ceph_inode_to_client(inode)->client->osdc;
|
||||||
struct ceph_inode_info *ci = ceph_inode(inode);
|
struct ceph_inode_info *ci = ceph_inode(inode);
|
||||||
struct page *page = list_entry(page_list->prev, struct page, lru);
|
struct page *page = lru_to_page(page_list);
|
||||||
struct ceph_vino vino;
|
struct ceph_vino vino;
|
||||||
struct ceph_osd_request *req;
|
struct ceph_osd_request *req;
|
||||||
u64 off;
|
u64 off;
|
||||||
@ -333,8 +333,7 @@ static int start_read(struct inode *inode, struct ceph_rw_context *rw_ctx,
|
|||||||
if (got)
|
if (got)
|
||||||
ceph_put_cap_refs(ci, got);
|
ceph_put_cap_refs(ci, got);
|
||||||
while (!list_empty(page_list)) {
|
while (!list_empty(page_list)) {
|
||||||
page = list_entry(page_list->prev,
|
page = lru_to_page(page_list);
|
||||||
struct page, lru);
|
|
||||||
list_del(&page->lru);
|
list_del(&page->lru);
|
||||||
put_page(page);
|
put_page(page);
|
||||||
}
|
}
|
||||||
|
@ -33,6 +33,7 @@
|
|||||||
#include <linux/mount.h>
|
#include <linux/mount.h>
|
||||||
#include <linux/slab.h>
|
#include <linux/slab.h>
|
||||||
#include <linux/swap.h>
|
#include <linux/swap.h>
|
||||||
|
#include <linux/mm.h>
|
||||||
#include <asm/div64.h>
|
#include <asm/div64.h>
|
||||||
#include "cifsfs.h"
|
#include "cifsfs.h"
|
||||||
#include "cifspdu.h"
|
#include "cifspdu.h"
|
||||||
@ -3964,7 +3965,7 @@ readpages_get_pages(struct address_space *mapping, struct list_head *page_list,
|
|||||||
|
|
||||||
INIT_LIST_HEAD(tmplist);
|
INIT_LIST_HEAD(tmplist);
|
||||||
|
|
||||||
page = list_entry(page_list->prev, struct page, lru);
|
page = lru_to_page(page_list);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Lock the page and put it in the cache. Since no one else
|
* Lock the page and put it in the cache. Since no one else
|
||||||
|
220
fs/eventpoll.c
220
fs/eventpoll.c
@ -381,7 +381,8 @@ static void ep_nested_calls_init(struct nested_calls *ncalls)
|
|||||||
*/
|
*/
|
||||||
static inline int ep_events_available(struct eventpoll *ep)
|
static inline int ep_events_available(struct eventpoll *ep)
|
||||||
{
|
{
|
||||||
return !list_empty(&ep->rdllist) || ep->ovflist != EP_UNACTIVE_PTR;
|
return !list_empty_careful(&ep->rdllist) ||
|
||||||
|
READ_ONCE(ep->ovflist) != EP_UNACTIVE_PTR;
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef CONFIG_NET_RX_BUSY_POLL
|
#ifdef CONFIG_NET_RX_BUSY_POLL
|
||||||
@ -471,7 +472,6 @@ static inline void ep_set_busy_poll_napi_id(struct epitem *epi)
|
|||||||
* no re-entered.
|
* no re-entered.
|
||||||
*
|
*
|
||||||
* @ncalls: Pointer to the nested_calls structure to be used for this call.
|
* @ncalls: Pointer to the nested_calls structure to be used for this call.
|
||||||
* @max_nests: Maximum number of allowed nesting calls.
|
|
||||||
* @nproc: Nested call core function pointer.
|
* @nproc: Nested call core function pointer.
|
||||||
* @priv: Opaque data to be passed to the @nproc callback.
|
* @priv: Opaque data to be passed to the @nproc callback.
|
||||||
* @cookie: Cookie to be used to identify this nested call.
|
* @cookie: Cookie to be used to identify this nested call.
|
||||||
@ -480,7 +480,7 @@ static inline void ep_set_busy_poll_napi_id(struct epitem *epi)
|
|||||||
* Returns: Returns the code returned by the @nproc callback, or -1 if
|
* Returns: Returns the code returned by the @nproc callback, or -1 if
|
||||||
* the maximum recursion limit has been exceeded.
|
* the maximum recursion limit has been exceeded.
|
||||||
*/
|
*/
|
||||||
static int ep_call_nested(struct nested_calls *ncalls, int max_nests,
|
static int ep_call_nested(struct nested_calls *ncalls,
|
||||||
int (*nproc)(void *, void *, int), void *priv,
|
int (*nproc)(void *, void *, int), void *priv,
|
||||||
void *cookie, void *ctx)
|
void *cookie, void *ctx)
|
||||||
{
|
{
|
||||||
@ -499,7 +499,7 @@ static int ep_call_nested(struct nested_calls *ncalls, int max_nests,
|
|||||||
*/
|
*/
|
||||||
list_for_each_entry(tncur, lsthead, llink) {
|
list_for_each_entry(tncur, lsthead, llink) {
|
||||||
if (tncur->ctx == ctx &&
|
if (tncur->ctx == ctx &&
|
||||||
(tncur->cookie == cookie || ++call_nests > max_nests)) {
|
(tncur->cookie == cookie || ++call_nests > EP_MAX_NESTS)) {
|
||||||
/*
|
/*
|
||||||
* Ops ... loop detected or maximum nest level reached.
|
* Ops ... loop detected or maximum nest level reached.
|
||||||
* We abort this wake by breaking the cycle itself.
|
* We abort this wake by breaking the cycle itself.
|
||||||
@ -573,7 +573,7 @@ static void ep_poll_safewake(wait_queue_head_t *wq)
|
|||||||
{
|
{
|
||||||
int this_cpu = get_cpu();
|
int this_cpu = get_cpu();
|
||||||
|
|
||||||
ep_call_nested(&poll_safewake_ncalls, EP_MAX_NESTS,
|
ep_call_nested(&poll_safewake_ncalls,
|
||||||
ep_poll_wakeup_proc, NULL, wq, (void *) (long) this_cpu);
|
ep_poll_wakeup_proc, NULL, wq, (void *) (long) this_cpu);
|
||||||
|
|
||||||
put_cpu();
|
put_cpu();
|
||||||
@ -699,7 +699,7 @@ static __poll_t ep_scan_ready_list(struct eventpoll *ep,
|
|||||||
*/
|
*/
|
||||||
spin_lock_irq(&ep->wq.lock);
|
spin_lock_irq(&ep->wq.lock);
|
||||||
list_splice_init(&ep->rdllist, &txlist);
|
list_splice_init(&ep->rdllist, &txlist);
|
||||||
ep->ovflist = NULL;
|
WRITE_ONCE(ep->ovflist, NULL);
|
||||||
spin_unlock_irq(&ep->wq.lock);
|
spin_unlock_irq(&ep->wq.lock);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -713,7 +713,7 @@ static __poll_t ep_scan_ready_list(struct eventpoll *ep,
|
|||||||
* other events might have been queued by the poll callback.
|
* other events might have been queued by the poll callback.
|
||||||
* We re-insert them inside the main ready-list here.
|
* We re-insert them inside the main ready-list here.
|
||||||
*/
|
*/
|
||||||
for (nepi = ep->ovflist; (epi = nepi) != NULL;
|
for (nepi = READ_ONCE(ep->ovflist); (epi = nepi) != NULL;
|
||||||
nepi = epi->next, epi->next = EP_UNACTIVE_PTR) {
|
nepi = epi->next, epi->next = EP_UNACTIVE_PTR) {
|
||||||
/*
|
/*
|
||||||
* We need to check if the item is already in the list.
|
* We need to check if the item is already in the list.
|
||||||
@ -731,7 +731,7 @@ static __poll_t ep_scan_ready_list(struct eventpoll *ep,
|
|||||||
* releasing the lock, events will be queued in the normal way inside
|
* releasing the lock, events will be queued in the normal way inside
|
||||||
* ep->rdllist.
|
* ep->rdllist.
|
||||||
*/
|
*/
|
||||||
ep->ovflist = EP_UNACTIVE_PTR;
|
WRITE_ONCE(ep->ovflist, EP_UNACTIVE_PTR);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Quickly re-inject items left on "txlist".
|
* Quickly re-inject items left on "txlist".
|
||||||
@ -1154,10 +1154,10 @@ static int ep_poll_callback(wait_queue_entry_t *wait, unsigned mode, int sync, v
|
|||||||
* semantics). All the events that happen during that period of time are
|
* semantics). All the events that happen during that period of time are
|
||||||
* chained in ep->ovflist and requeued later on.
|
* chained in ep->ovflist and requeued later on.
|
||||||
*/
|
*/
|
||||||
if (unlikely(ep->ovflist != EP_UNACTIVE_PTR)) {
|
if (READ_ONCE(ep->ovflist) != EP_UNACTIVE_PTR) {
|
||||||
if (epi->next == EP_UNACTIVE_PTR) {
|
if (epi->next == EP_UNACTIVE_PTR) {
|
||||||
epi->next = ep->ovflist;
|
epi->next = READ_ONCE(ep->ovflist);
|
||||||
ep->ovflist = epi;
|
WRITE_ONCE(ep->ovflist, epi);
|
||||||
if (epi->ws) {
|
if (epi->ws) {
|
||||||
/*
|
/*
|
||||||
* Activate ep->ws since epi->ws may get
|
* Activate ep->ws since epi->ws may get
|
||||||
@ -1333,7 +1333,6 @@ static int reverse_path_check_proc(void *priv, void *cookie, int call_nests)
|
|||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
error = ep_call_nested(&poll_loop_ncalls,
|
error = ep_call_nested(&poll_loop_ncalls,
|
||||||
EP_MAX_NESTS,
|
|
||||||
reverse_path_check_proc,
|
reverse_path_check_proc,
|
||||||
child_file, child_file,
|
child_file, child_file,
|
||||||
current);
|
current);
|
||||||
@ -1367,7 +1366,7 @@ static int reverse_path_check(void)
|
|||||||
/* let's call this for all tfiles */
|
/* let's call this for all tfiles */
|
||||||
list_for_each_entry(current_file, &tfile_check_list, f_tfile_llink) {
|
list_for_each_entry(current_file, &tfile_check_list, f_tfile_llink) {
|
||||||
path_count_init();
|
path_count_init();
|
||||||
error = ep_call_nested(&poll_loop_ncalls, EP_MAX_NESTS,
|
error = ep_call_nested(&poll_loop_ncalls,
|
||||||
reverse_path_check_proc, current_file,
|
reverse_path_check_proc, current_file,
|
||||||
current_file, current);
|
current_file, current);
|
||||||
if (error)
|
if (error)
|
||||||
@ -1626,21 +1625,24 @@ static __poll_t ep_send_events_proc(struct eventpoll *ep, struct list_head *head
|
|||||||
{
|
{
|
||||||
struct ep_send_events_data *esed = priv;
|
struct ep_send_events_data *esed = priv;
|
||||||
__poll_t revents;
|
__poll_t revents;
|
||||||
struct epitem *epi;
|
struct epitem *epi, *tmp;
|
||||||
struct epoll_event __user *uevent;
|
struct epoll_event __user *uevent = esed->events;
|
||||||
struct wakeup_source *ws;
|
struct wakeup_source *ws;
|
||||||
poll_table pt;
|
poll_table pt;
|
||||||
|
|
||||||
init_poll_funcptr(&pt, NULL);
|
init_poll_funcptr(&pt, NULL);
|
||||||
|
esed->res = 0;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* We can loop without lock because we are passed a task private list.
|
* We can loop without lock because we are passed a task private list.
|
||||||
* Items cannot vanish during the loop because ep_scan_ready_list() is
|
* Items cannot vanish during the loop because ep_scan_ready_list() is
|
||||||
* holding "mtx" during this call.
|
* holding "mtx" during this call.
|
||||||
*/
|
*/
|
||||||
for (esed->res = 0, uevent = esed->events;
|
lockdep_assert_held(&ep->mtx);
|
||||||
!list_empty(head) && esed->res < esed->maxevents;) {
|
|
||||||
epi = list_first_entry(head, struct epitem, rdllink);
|
list_for_each_entry_safe(epi, tmp, head, rdllink) {
|
||||||
|
if (esed->res >= esed->maxevents)
|
||||||
|
break;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Activate ep->ws before deactivating epi->ws to prevent
|
* Activate ep->ws before deactivating epi->ws to prevent
|
||||||
@ -1660,42 +1662,42 @@ static __poll_t ep_send_events_proc(struct eventpoll *ep, struct list_head *head
|
|||||||
|
|
||||||
list_del_init(&epi->rdllink);
|
list_del_init(&epi->rdllink);
|
||||||
|
|
||||||
revents = ep_item_poll(epi, &pt, 1);
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If the event mask intersect the caller-requested one,
|
* If the event mask intersect the caller-requested one,
|
||||||
* deliver the event to userspace. Again, ep_scan_ready_list()
|
* deliver the event to userspace. Again, ep_scan_ready_list()
|
||||||
* is holding "mtx", so no operations coming from userspace
|
* is holding ep->mtx, so no operations coming from userspace
|
||||||
* can change the item.
|
* can change the item.
|
||||||
*/
|
*/
|
||||||
if (revents) {
|
revents = ep_item_poll(epi, &pt, 1);
|
||||||
if (__put_user(revents, &uevent->events) ||
|
if (!revents)
|
||||||
__put_user(epi->event.data, &uevent->data)) {
|
continue;
|
||||||
list_add(&epi->rdllink, head);
|
|
||||||
ep_pm_stay_awake(epi);
|
if (__put_user(revents, &uevent->events) ||
|
||||||
if (!esed->res)
|
__put_user(epi->event.data, &uevent->data)) {
|
||||||
esed->res = -EFAULT;
|
list_add(&epi->rdllink, head);
|
||||||
return 0;
|
ep_pm_stay_awake(epi);
|
||||||
}
|
if (!esed->res)
|
||||||
esed->res++;
|
esed->res = -EFAULT;
|
||||||
uevent++;
|
return 0;
|
||||||
if (epi->event.events & EPOLLONESHOT)
|
}
|
||||||
epi->event.events &= EP_PRIVATE_BITS;
|
esed->res++;
|
||||||
else if (!(epi->event.events & EPOLLET)) {
|
uevent++;
|
||||||
/*
|
if (epi->event.events & EPOLLONESHOT)
|
||||||
* If this file has been added with Level
|
epi->event.events &= EP_PRIVATE_BITS;
|
||||||
* Trigger mode, we need to insert back inside
|
else if (!(epi->event.events & EPOLLET)) {
|
||||||
* the ready list, so that the next call to
|
/*
|
||||||
* epoll_wait() will check again the events
|
* If this file has been added with Level
|
||||||
* availability. At this point, no one can insert
|
* Trigger mode, we need to insert back inside
|
||||||
* into ep->rdllist besides us. The epoll_ctl()
|
* the ready list, so that the next call to
|
||||||
* callers are locked out by
|
* epoll_wait() will check again the events
|
||||||
* ep_scan_ready_list() holding "mtx" and the
|
* availability. At this point, no one can insert
|
||||||
* poll callback will queue them in ep->ovflist.
|
* into ep->rdllist besides us. The epoll_ctl()
|
||||||
*/
|
* callers are locked out by
|
||||||
list_add_tail(&epi->rdllink, &ep->rdllist);
|
* ep_scan_ready_list() holding "mtx" and the
|
||||||
ep_pm_stay_awake(epi);
|
* poll callback will queue them in ep->ovflist.
|
||||||
}
|
*/
|
||||||
|
list_add_tail(&epi->rdllink, &ep->rdllist);
|
||||||
|
ep_pm_stay_awake(epi);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1747,6 +1749,7 @@ static int ep_poll(struct eventpoll *ep, struct epoll_event __user *events,
|
|||||||
{
|
{
|
||||||
int res = 0, eavail, timed_out = 0;
|
int res = 0, eavail, timed_out = 0;
|
||||||
u64 slack = 0;
|
u64 slack = 0;
|
||||||
|
bool waiter = false;
|
||||||
wait_queue_entry_t wait;
|
wait_queue_entry_t wait;
|
||||||
ktime_t expires, *to = NULL;
|
ktime_t expires, *to = NULL;
|
||||||
|
|
||||||
@ -1761,11 +1764,18 @@ static int ep_poll(struct eventpoll *ep, struct epoll_event __user *events,
|
|||||||
} else if (timeout == 0) {
|
} else if (timeout == 0) {
|
||||||
/*
|
/*
|
||||||
* Avoid the unnecessary trip to the wait queue loop, if the
|
* Avoid the unnecessary trip to the wait queue loop, if the
|
||||||
* caller specified a non blocking operation.
|
* caller specified a non blocking operation. We still need
|
||||||
|
* lock because we could race and not see an epi being added
|
||||||
|
* to the ready list while in irq callback. Thus incorrectly
|
||||||
|
* returning 0 back to userspace.
|
||||||
*/
|
*/
|
||||||
timed_out = 1;
|
timed_out = 1;
|
||||||
|
|
||||||
spin_lock_irq(&ep->wq.lock);
|
spin_lock_irq(&ep->wq.lock);
|
||||||
goto check_events;
|
eavail = ep_events_available(ep);
|
||||||
|
spin_unlock_irq(&ep->wq.lock);
|
||||||
|
|
||||||
|
goto send_events;
|
||||||
}
|
}
|
||||||
|
|
||||||
fetch_events:
|
fetch_events:
|
||||||
@ -1773,64 +1783,66 @@ fetch_events:
|
|||||||
if (!ep_events_available(ep))
|
if (!ep_events_available(ep))
|
||||||
ep_busy_loop(ep, timed_out);
|
ep_busy_loop(ep, timed_out);
|
||||||
|
|
||||||
spin_lock_irq(&ep->wq.lock);
|
eavail = ep_events_available(ep);
|
||||||
|
if (eavail)
|
||||||
|
goto send_events;
|
||||||
|
|
||||||
if (!ep_events_available(ep)) {
|
/*
|
||||||
/*
|
* Busy poll timed out. Drop NAPI ID for now, we can add
|
||||||
* Busy poll timed out. Drop NAPI ID for now, we can add
|
* it back in when we have moved a socket with a valid NAPI
|
||||||
* it back in when we have moved a socket with a valid NAPI
|
* ID onto the ready list.
|
||||||
* ID onto the ready list.
|
*/
|
||||||
*/
|
ep_reset_busy_poll_napi_id(ep);
|
||||||
ep_reset_busy_poll_napi_id(ep);
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* We don't have any available event to return to the caller.
|
* We don't have any available event to return to the caller. We need
|
||||||
* We need to sleep here, and we will be wake up by
|
* to sleep here, and we will be woken by ep_poll_callback() when events
|
||||||
* ep_poll_callback() when events will become available.
|
* become available.
|
||||||
*/
|
*/
|
||||||
|
if (!waiter) {
|
||||||
|
waiter = true;
|
||||||
init_waitqueue_entry(&wait, current);
|
init_waitqueue_entry(&wait, current);
|
||||||
|
|
||||||
|
spin_lock_irq(&ep->wq.lock);
|
||||||
__add_wait_queue_exclusive(&ep->wq, &wait);
|
__add_wait_queue_exclusive(&ep->wq, &wait);
|
||||||
|
spin_unlock_irq(&ep->wq.lock);
|
||||||
|
}
|
||||||
|
|
||||||
for (;;) {
|
for (;;) {
|
||||||
/*
|
/*
|
||||||
* We don't want to sleep if the ep_poll_callback() sends us
|
* We don't want to sleep if the ep_poll_callback() sends us
|
||||||
* a wakeup in between. That's why we set the task state
|
* a wakeup in between. That's why we set the task state
|
||||||
* to TASK_INTERRUPTIBLE before doing the checks.
|
* to TASK_INTERRUPTIBLE before doing the checks.
|
||||||
*/
|
*/
|
||||||
set_current_state(TASK_INTERRUPTIBLE);
|
set_current_state(TASK_INTERRUPTIBLE);
|
||||||
/*
|
/*
|
||||||
* Always short-circuit for fatal signals to allow
|
* Always short-circuit for fatal signals to allow
|
||||||
* threads to make a timely exit without the chance of
|
* threads to make a timely exit without the chance of
|
||||||
* finding more events available and fetching
|
* finding more events available and fetching
|
||||||
* repeatedly.
|
* repeatedly.
|
||||||
*/
|
*/
|
||||||
if (fatal_signal_pending(current)) {
|
if (fatal_signal_pending(current)) {
|
||||||
res = -EINTR;
|
res = -EINTR;
|
||||||
break;
|
break;
|
||||||
}
|
|
||||||
if (ep_events_available(ep) || timed_out)
|
|
||||||
break;
|
|
||||||
if (signal_pending(current)) {
|
|
||||||
res = -EINTR;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
spin_unlock_irq(&ep->wq.lock);
|
|
||||||
if (!schedule_hrtimeout_range(to, slack, HRTIMER_MODE_ABS))
|
|
||||||
timed_out = 1;
|
|
||||||
|
|
||||||
spin_lock_irq(&ep->wq.lock);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
__remove_wait_queue(&ep->wq, &wait);
|
eavail = ep_events_available(ep);
|
||||||
__set_current_state(TASK_RUNNING);
|
if (eavail)
|
||||||
|
break;
|
||||||
|
if (signal_pending(current)) {
|
||||||
|
res = -EINTR;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!schedule_hrtimeout_range(to, slack, HRTIMER_MODE_ABS)) {
|
||||||
|
timed_out = 1;
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
check_events:
|
|
||||||
/* Is it worth to try to dig for events ? */
|
|
||||||
eavail = ep_events_available(ep);
|
|
||||||
|
|
||||||
spin_unlock_irq(&ep->wq.lock);
|
__set_current_state(TASK_RUNNING);
|
||||||
|
|
||||||
|
send_events:
|
||||||
/*
|
/*
|
||||||
* Try to transfer events to user space. In case we get 0 events and
|
* Try to transfer events to user space. In case we get 0 events and
|
||||||
* there's still timeout left over, we go trying again in search of
|
* there's still timeout left over, we go trying again in search of
|
||||||
@ -1840,6 +1852,12 @@ check_events:
|
|||||||
!(res = ep_send_events(ep, events, maxevents)) && !timed_out)
|
!(res = ep_send_events(ep, events, maxevents)) && !timed_out)
|
||||||
goto fetch_events;
|
goto fetch_events;
|
||||||
|
|
||||||
|
if (waiter) {
|
||||||
|
spin_lock_irq(&ep->wq.lock);
|
||||||
|
__remove_wait_queue(&ep->wq, &wait);
|
||||||
|
spin_unlock_irq(&ep->wq.lock);
|
||||||
|
}
|
||||||
|
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1876,7 +1894,7 @@ static int ep_loop_check_proc(void *priv, void *cookie, int call_nests)
|
|||||||
ep_tovisit = epi->ffd.file->private_data;
|
ep_tovisit = epi->ffd.file->private_data;
|
||||||
if (ep_tovisit->visited)
|
if (ep_tovisit->visited)
|
||||||
continue;
|
continue;
|
||||||
error = ep_call_nested(&poll_loop_ncalls, EP_MAX_NESTS,
|
error = ep_call_nested(&poll_loop_ncalls,
|
||||||
ep_loop_check_proc, epi->ffd.file,
|
ep_loop_check_proc, epi->ffd.file,
|
||||||
ep_tovisit, current);
|
ep_tovisit, current);
|
||||||
if (error != 0)
|
if (error != 0)
|
||||||
@ -1916,7 +1934,7 @@ static int ep_loop_check(struct eventpoll *ep, struct file *file)
|
|||||||
int ret;
|
int ret;
|
||||||
struct eventpoll *ep_cur, *ep_next;
|
struct eventpoll *ep_cur, *ep_next;
|
||||||
|
|
||||||
ret = ep_call_nested(&poll_loop_ncalls, EP_MAX_NESTS,
|
ret = ep_call_nested(&poll_loop_ncalls,
|
||||||
ep_loop_check_proc, file, ep, current);
|
ep_loop_check_proc, file, ep, current);
|
||||||
/* clear visited list */
|
/* clear visited list */
|
||||||
list_for_each_entry_safe(ep_cur, ep_next, &visited_list,
|
list_for_each_entry_safe(ep_cur, ep_next, &visited_list,
|
||||||
|
109
fs/exec.c
109
fs/exec.c
@ -218,55 +218,10 @@ static struct page *get_arg_page(struct linux_binprm *bprm, unsigned long pos,
|
|||||||
if (ret <= 0)
|
if (ret <= 0)
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
if (write) {
|
if (write)
|
||||||
unsigned long size = bprm->vma->vm_end - bprm->vma->vm_start;
|
acct_arg_size(bprm, vma_pages(bprm->vma));
|
||||||
unsigned long ptr_size, limit;
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Since the stack will hold pointers to the strings, we
|
|
||||||
* must account for them as well.
|
|
||||||
*
|
|
||||||
* The size calculation is the entire vma while each arg page is
|
|
||||||
* built, so each time we get here it's calculating how far it
|
|
||||||
* is currently (rather than each call being just the newly
|
|
||||||
* added size from the arg page). As a result, we need to
|
|
||||||
* always add the entire size of the pointers, so that on the
|
|
||||||
* last call to get_arg_page() we'll actually have the entire
|
|
||||||
* correct size.
|
|
||||||
*/
|
|
||||||
ptr_size = (bprm->argc + bprm->envc) * sizeof(void *);
|
|
||||||
if (ptr_size > ULONG_MAX - size)
|
|
||||||
goto fail;
|
|
||||||
size += ptr_size;
|
|
||||||
|
|
||||||
acct_arg_size(bprm, size / PAGE_SIZE);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* We've historically supported up to 32 pages (ARG_MAX)
|
|
||||||
* of argument strings even with small stacks
|
|
||||||
*/
|
|
||||||
if (size <= ARG_MAX)
|
|
||||||
return page;
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Limit to 1/4 of the max stack size or 3/4 of _STK_LIM
|
|
||||||
* (whichever is smaller) for the argv+env strings.
|
|
||||||
* This ensures that:
|
|
||||||
* - the remaining binfmt code will not run out of stack space,
|
|
||||||
* - the program will have a reasonable amount of stack left
|
|
||||||
* to work from.
|
|
||||||
*/
|
|
||||||
limit = _STK_LIM / 4 * 3;
|
|
||||||
limit = min(limit, bprm->rlim_stack.rlim_cur / 4);
|
|
||||||
if (size > limit)
|
|
||||||
goto fail;
|
|
||||||
}
|
|
||||||
|
|
||||||
return page;
|
return page;
|
||||||
|
|
||||||
fail:
|
|
||||||
put_page(page);
|
|
||||||
return NULL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void put_arg_page(struct page *page)
|
static void put_arg_page(struct page *page)
|
||||||
@ -492,6 +447,50 @@ static int count(struct user_arg_ptr argv, int max)
|
|||||||
return i;
|
return i;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int prepare_arg_pages(struct linux_binprm *bprm,
|
||||||
|
struct user_arg_ptr argv, struct user_arg_ptr envp)
|
||||||
|
{
|
||||||
|
unsigned long limit, ptr_size;
|
||||||
|
|
||||||
|
bprm->argc = count(argv, MAX_ARG_STRINGS);
|
||||||
|
if (bprm->argc < 0)
|
||||||
|
return bprm->argc;
|
||||||
|
|
||||||
|
bprm->envc = count(envp, MAX_ARG_STRINGS);
|
||||||
|
if (bprm->envc < 0)
|
||||||
|
return bprm->envc;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Limit to 1/4 of the max stack size or 3/4 of _STK_LIM
|
||||||
|
* (whichever is smaller) for the argv+env strings.
|
||||||
|
* This ensures that:
|
||||||
|
* - the remaining binfmt code will not run out of stack space,
|
||||||
|
* - the program will have a reasonable amount of stack left
|
||||||
|
* to work from.
|
||||||
|
*/
|
||||||
|
limit = _STK_LIM / 4 * 3;
|
||||||
|
limit = min(limit, bprm->rlim_stack.rlim_cur / 4);
|
||||||
|
/*
|
||||||
|
* We've historically supported up to 32 pages (ARG_MAX)
|
||||||
|
* of argument strings even with small stacks
|
||||||
|
*/
|
||||||
|
limit = max_t(unsigned long, limit, ARG_MAX);
|
||||||
|
/*
|
||||||
|
* We must account for the size of all the argv and envp pointers to
|
||||||
|
* the argv and envp strings, since they will also take up space in
|
||||||
|
* the stack. They aren't stored until much later when we can't
|
||||||
|
* signal to the parent that the child has run out of stack space.
|
||||||
|
* Instead, calculate it here so it's possible to fail gracefully.
|
||||||
|
*/
|
||||||
|
ptr_size = (bprm->argc + bprm->envc) * sizeof(void *);
|
||||||
|
if (limit <= ptr_size)
|
||||||
|
return -E2BIG;
|
||||||
|
limit -= ptr_size;
|
||||||
|
|
||||||
|
bprm->argmin = bprm->p - limit;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* 'copy_strings()' copies argument/environment strings from the old
|
* 'copy_strings()' copies argument/environment strings from the old
|
||||||
* processes's memory to the new process's stack. The call to get_user_pages()
|
* processes's memory to the new process's stack. The call to get_user_pages()
|
||||||
@ -527,6 +526,10 @@ static int copy_strings(int argc, struct user_arg_ptr argv,
|
|||||||
pos = bprm->p;
|
pos = bprm->p;
|
||||||
str += len;
|
str += len;
|
||||||
bprm->p -= len;
|
bprm->p -= len;
|
||||||
|
#ifdef CONFIG_MMU
|
||||||
|
if (bprm->p < bprm->argmin)
|
||||||
|
goto out;
|
||||||
|
#endif
|
||||||
|
|
||||||
while (len > 0) {
|
while (len > 0) {
|
||||||
int offset, bytes_to_copy;
|
int offset, bytes_to_copy;
|
||||||
@ -1084,7 +1087,7 @@ static int de_thread(struct task_struct *tsk)
|
|||||||
__set_current_state(TASK_KILLABLE);
|
__set_current_state(TASK_KILLABLE);
|
||||||
spin_unlock_irq(lock);
|
spin_unlock_irq(lock);
|
||||||
schedule();
|
schedule();
|
||||||
if (unlikely(__fatal_signal_pending(tsk)))
|
if (__fatal_signal_pending(tsk))
|
||||||
goto killed;
|
goto killed;
|
||||||
spin_lock_irq(lock);
|
spin_lock_irq(lock);
|
||||||
}
|
}
|
||||||
@ -1112,7 +1115,7 @@ static int de_thread(struct task_struct *tsk)
|
|||||||
write_unlock_irq(&tasklist_lock);
|
write_unlock_irq(&tasklist_lock);
|
||||||
cgroup_threadgroup_change_end(tsk);
|
cgroup_threadgroup_change_end(tsk);
|
||||||
schedule();
|
schedule();
|
||||||
if (unlikely(__fatal_signal_pending(tsk)))
|
if (__fatal_signal_pending(tsk))
|
||||||
goto killed;
|
goto killed;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1789,12 +1792,8 @@ static int __do_execve_file(int fd, struct filename *filename,
|
|||||||
if (retval)
|
if (retval)
|
||||||
goto out_unmark;
|
goto out_unmark;
|
||||||
|
|
||||||
bprm->argc = count(argv, MAX_ARG_STRINGS);
|
retval = prepare_arg_pages(bprm, argv, envp);
|
||||||
if ((retval = bprm->argc) < 0)
|
if (retval < 0)
|
||||||
goto out;
|
|
||||||
|
|
||||||
bprm->envc = count(envp, MAX_ARG_STRINGS);
|
|
||||||
if ((retval = bprm->envc) < 0)
|
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
retval = prepare_binprm(bprm);
|
retval = prepare_binprm(bprm);
|
||||||
|
@ -128,7 +128,7 @@ int ext4_mpage_readpages(struct address_space *mapping,
|
|||||||
|
|
||||||
prefetchw(&page->flags);
|
prefetchw(&page->flags);
|
||||||
if (pages) {
|
if (pages) {
|
||||||
page = list_entry(pages->prev, struct page, lru);
|
page = lru_to_page(pages);
|
||||||
list_del(&page->lru);
|
list_del(&page->lru);
|
||||||
if (add_to_page_cache_lru(page, mapping, page->index,
|
if (add_to_page_cache_lru(page, mapping, page->index,
|
||||||
readahead_gfp_mask(mapping)))
|
readahead_gfp_mask(mapping)))
|
||||||
|
@ -363,7 +363,7 @@ int fat_bmap(struct inode *inode, sector_t sector, sector_t *phys,
|
|||||||
|
|
||||||
*phys = 0;
|
*phys = 0;
|
||||||
*mapped_blocks = 0;
|
*mapped_blocks = 0;
|
||||||
if ((sbi->fat_bits != 32) && (inode->i_ino == MSDOS_ROOT_INO)) {
|
if (!is_fat32(sbi) && (inode->i_ino == MSDOS_ROOT_INO)) {
|
||||||
if (sector < (sbi->dir_entries >> sbi->dir_per_block_bits)) {
|
if (sector < (sbi->dir_entries >> sbi->dir_per_block_bits)) {
|
||||||
*phys = sector + sbi->dir_start;
|
*phys = sector + sbi->dir_start;
|
||||||
*mapped_blocks = 1;
|
*mapped_blocks = 1;
|
||||||
|
@ -57,7 +57,7 @@ static inline void fat_dir_readahead(struct inode *dir, sector_t iblock,
|
|||||||
if ((iblock & (sbi->sec_per_clus - 1)) || sbi->sec_per_clus == 1)
|
if ((iblock & (sbi->sec_per_clus - 1)) || sbi->sec_per_clus == 1)
|
||||||
return;
|
return;
|
||||||
/* root dir of FAT12/FAT16 */
|
/* root dir of FAT12/FAT16 */
|
||||||
if ((sbi->fat_bits != 32) && (dir->i_ino == MSDOS_ROOT_INO))
|
if (!is_fat32(sbi) && (dir->i_ino == MSDOS_ROOT_INO))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
bh = sb_find_get_block(sb, phys);
|
bh = sb_find_get_block(sb, phys);
|
||||||
@ -1313,7 +1313,7 @@ int fat_add_entries(struct inode *dir, void *slots, int nr_slots,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (dir->i_ino == MSDOS_ROOT_INO) {
|
if (dir->i_ino == MSDOS_ROOT_INO) {
|
||||||
if (sbi->fat_bits != 32)
|
if (!is_fat32(sbi))
|
||||||
goto error;
|
goto error;
|
||||||
} else if (MSDOS_I(dir)->i_start == 0) {
|
} else if (MSDOS_I(dir)->i_start == 0) {
|
||||||
fat_msg(sb, KERN_ERR, "Corrupted directory (i_pos %lld)",
|
fat_msg(sb, KERN_ERR, "Corrupted directory (i_pos %lld)",
|
||||||
|
30
fs/fat/fat.h
30
fs/fat/fat.h
@ -142,6 +142,34 @@ static inline struct msdos_sb_info *MSDOS_SB(struct super_block *sb)
|
|||||||
return sb->s_fs_info;
|
return sb->s_fs_info;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Functions that determine the variant of the FAT file system (i.e.,
|
||||||
|
* whether this is FAT12, FAT16 or FAT32.
|
||||||
|
*/
|
||||||
|
static inline bool is_fat12(const struct msdos_sb_info *sbi)
|
||||||
|
{
|
||||||
|
return sbi->fat_bits == 12;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline bool is_fat16(const struct msdos_sb_info *sbi)
|
||||||
|
{
|
||||||
|
return sbi->fat_bits == 16;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline bool is_fat32(const struct msdos_sb_info *sbi)
|
||||||
|
{
|
||||||
|
return sbi->fat_bits == 32;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Maximum number of clusters */
|
||||||
|
static inline u32 max_fat(struct super_block *sb)
|
||||||
|
{
|
||||||
|
struct msdos_sb_info *sbi = MSDOS_SB(sb);
|
||||||
|
|
||||||
|
return is_fat32(sbi) ? MAX_FAT32 :
|
||||||
|
is_fat16(sbi) ? MAX_FAT16 : MAX_FAT12;
|
||||||
|
}
|
||||||
|
|
||||||
static inline struct msdos_inode_info *MSDOS_I(struct inode *inode)
|
static inline struct msdos_inode_info *MSDOS_I(struct inode *inode)
|
||||||
{
|
{
|
||||||
return container_of(inode, struct msdos_inode_info, vfs_inode);
|
return container_of(inode, struct msdos_inode_info, vfs_inode);
|
||||||
@ -257,7 +285,7 @@ static inline int fat_get_start(const struct msdos_sb_info *sbi,
|
|||||||
const struct msdos_dir_entry *de)
|
const struct msdos_dir_entry *de)
|
||||||
{
|
{
|
||||||
int cluster = le16_to_cpu(de->start);
|
int cluster = le16_to_cpu(de->start);
|
||||||
if (sbi->fat_bits == 32)
|
if (is_fat32(sbi))
|
||||||
cluster |= (le16_to_cpu(de->starthi) << 16);
|
cluster |= (le16_to_cpu(de->starthi) << 16);
|
||||||
return cluster;
|
return cluster;
|
||||||
}
|
}
|
||||||
|
@ -290,19 +290,17 @@ void fat_ent_access_init(struct super_block *sb)
|
|||||||
|
|
||||||
mutex_init(&sbi->fat_lock);
|
mutex_init(&sbi->fat_lock);
|
||||||
|
|
||||||
switch (sbi->fat_bits) {
|
if (is_fat32(sbi)) {
|
||||||
case 32:
|
|
||||||
sbi->fatent_shift = 2;
|
sbi->fatent_shift = 2;
|
||||||
sbi->fatent_ops = &fat32_ops;
|
sbi->fatent_ops = &fat32_ops;
|
||||||
break;
|
} else if (is_fat16(sbi)) {
|
||||||
case 16:
|
|
||||||
sbi->fatent_shift = 1;
|
sbi->fatent_shift = 1;
|
||||||
sbi->fatent_ops = &fat16_ops;
|
sbi->fatent_ops = &fat16_ops;
|
||||||
break;
|
} else if (is_fat12(sbi)) {
|
||||||
case 12:
|
|
||||||
sbi->fatent_shift = -1;
|
sbi->fatent_shift = -1;
|
||||||
sbi->fatent_ops = &fat12_ops;
|
sbi->fatent_ops = &fat12_ops;
|
||||||
break;
|
} else {
|
||||||
|
fat_fs_error(sb, "invalid FAT variant, %u bits", sbi->fat_bits);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -310,7 +308,7 @@ static void mark_fsinfo_dirty(struct super_block *sb)
|
|||||||
{
|
{
|
||||||
struct msdos_sb_info *sbi = MSDOS_SB(sb);
|
struct msdos_sb_info *sbi = MSDOS_SB(sb);
|
||||||
|
|
||||||
if (sb_rdonly(sb) || sbi->fat_bits != 32)
|
if (sb_rdonly(sb) || !is_fat32(sbi))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
__mark_inode_dirty(sbi->fsinfo_inode, I_DIRTY_SYNC);
|
__mark_inode_dirty(sbi->fsinfo_inode, I_DIRTY_SYNC);
|
||||||
@ -327,7 +325,7 @@ static inline int fat_ent_update_ptr(struct super_block *sb,
|
|||||||
/* Is this fatent's blocks including this entry? */
|
/* Is this fatent's blocks including this entry? */
|
||||||
if (!fatent->nr_bhs || bhs[0]->b_blocknr != blocknr)
|
if (!fatent->nr_bhs || bhs[0]->b_blocknr != blocknr)
|
||||||
return 0;
|
return 0;
|
||||||
if (sbi->fat_bits == 12) {
|
if (is_fat12(sbi)) {
|
||||||
if ((offset + 1) < sb->s_blocksize) {
|
if ((offset + 1) < sb->s_blocksize) {
|
||||||
/* This entry is on bhs[0]. */
|
/* This entry is on bhs[0]. */
|
||||||
if (fatent->nr_bhs == 2) {
|
if (fatent->nr_bhs == 2) {
|
||||||
|
@ -686,7 +686,7 @@ static void fat_set_state(struct super_block *sb,
|
|||||||
|
|
||||||
b = (struct fat_boot_sector *) bh->b_data;
|
b = (struct fat_boot_sector *) bh->b_data;
|
||||||
|
|
||||||
if (sbi->fat_bits == 32) {
|
if (is_fat32(sbi)) {
|
||||||
if (set)
|
if (set)
|
||||||
b->fat32.state |= FAT_STATE_DIRTY;
|
b->fat32.state |= FAT_STATE_DIRTY;
|
||||||
else
|
else
|
||||||
@ -1396,7 +1396,7 @@ static int fat_read_root(struct inode *inode)
|
|||||||
inode->i_mode = fat_make_mode(sbi, ATTR_DIR, S_IRWXUGO);
|
inode->i_mode = fat_make_mode(sbi, ATTR_DIR, S_IRWXUGO);
|
||||||
inode->i_op = sbi->dir_ops;
|
inode->i_op = sbi->dir_ops;
|
||||||
inode->i_fop = &fat_dir_operations;
|
inode->i_fop = &fat_dir_operations;
|
||||||
if (sbi->fat_bits == 32) {
|
if (is_fat32(sbi)) {
|
||||||
MSDOS_I(inode)->i_start = sbi->root_cluster;
|
MSDOS_I(inode)->i_start = sbi->root_cluster;
|
||||||
error = fat_calc_dir_size(inode);
|
error = fat_calc_dir_size(inode);
|
||||||
if (error < 0)
|
if (error < 0)
|
||||||
@ -1423,7 +1423,7 @@ static unsigned long calc_fat_clusters(struct super_block *sb)
|
|||||||
struct msdos_sb_info *sbi = MSDOS_SB(sb);
|
struct msdos_sb_info *sbi = MSDOS_SB(sb);
|
||||||
|
|
||||||
/* Divide first to avoid overflow */
|
/* Divide first to avoid overflow */
|
||||||
if (sbi->fat_bits != 12) {
|
if (!is_fat12(sbi)) {
|
||||||
unsigned long ent_per_sec = sb->s_blocksize * 8 / sbi->fat_bits;
|
unsigned long ent_per_sec = sb->s_blocksize * 8 / sbi->fat_bits;
|
||||||
return ent_per_sec * sbi->fat_length;
|
return ent_per_sec * sbi->fat_length;
|
||||||
}
|
}
|
||||||
@ -1743,7 +1743,7 @@ int fat_fill_super(struct super_block *sb, void *data, int silent, int isvfat,
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* interpret volume ID as a little endian 32 bit integer */
|
/* interpret volume ID as a little endian 32 bit integer */
|
||||||
if (sbi->fat_bits == 32)
|
if (is_fat32(sbi))
|
||||||
sbi->vol_id = bpb.fat32_vol_id;
|
sbi->vol_id = bpb.fat32_vol_id;
|
||||||
else /* fat 16 or 12 */
|
else /* fat 16 or 12 */
|
||||||
sbi->vol_id = bpb.fat16_vol_id;
|
sbi->vol_id = bpb.fat16_vol_id;
|
||||||
@ -1769,11 +1769,11 @@ int fat_fill_super(struct super_block *sb, void *data, int silent, int isvfat,
|
|||||||
|
|
||||||
total_clusters = (total_sectors - sbi->data_start) / sbi->sec_per_clus;
|
total_clusters = (total_sectors - sbi->data_start) / sbi->sec_per_clus;
|
||||||
|
|
||||||
if (sbi->fat_bits != 32)
|
if (!is_fat32(sbi))
|
||||||
sbi->fat_bits = (total_clusters > MAX_FAT12) ? 16 : 12;
|
sbi->fat_bits = (total_clusters > MAX_FAT12) ? 16 : 12;
|
||||||
|
|
||||||
/* some OSes set FAT_STATE_DIRTY and clean it on unmount. */
|
/* some OSes set FAT_STATE_DIRTY and clean it on unmount. */
|
||||||
if (sbi->fat_bits == 32)
|
if (is_fat32(sbi))
|
||||||
sbi->dirty = bpb.fat32_state & FAT_STATE_DIRTY;
|
sbi->dirty = bpb.fat32_state & FAT_STATE_DIRTY;
|
||||||
else /* fat 16 or 12 */
|
else /* fat 16 or 12 */
|
||||||
sbi->dirty = bpb.fat16_state & FAT_STATE_DIRTY;
|
sbi->dirty = bpb.fat16_state & FAT_STATE_DIRTY;
|
||||||
@ -1781,7 +1781,7 @@ int fat_fill_super(struct super_block *sb, void *data, int silent, int isvfat,
|
|||||||
/* check that FAT table does not overflow */
|
/* check that FAT table does not overflow */
|
||||||
fat_clusters = calc_fat_clusters(sb);
|
fat_clusters = calc_fat_clusters(sb);
|
||||||
total_clusters = min(total_clusters, fat_clusters - FAT_START_ENT);
|
total_clusters = min(total_clusters, fat_clusters - FAT_START_ENT);
|
||||||
if (total_clusters > MAX_FAT(sb)) {
|
if (total_clusters > max_fat(sb)) {
|
||||||
if (!silent)
|
if (!silent)
|
||||||
fat_msg(sb, KERN_ERR, "count of clusters too big (%u)",
|
fat_msg(sb, KERN_ERR, "count of clusters too big (%u)",
|
||||||
total_clusters);
|
total_clusters);
|
||||||
@ -1803,11 +1803,15 @@ int fat_fill_super(struct super_block *sb, void *data, int silent, int isvfat,
|
|||||||
fat_ent_access_init(sb);
|
fat_ent_access_init(sb);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* The low byte of FAT's first entry must have same value with
|
* The low byte of the first FAT entry must have the same value as
|
||||||
* media-field. But in real world, too many devices is
|
* the media field of the boot sector. But in real world, too many
|
||||||
* writing wrong value. So, removed that validity check.
|
* devices are writing wrong values. So, removed that validity check.
|
||||||
*
|
*
|
||||||
* if (FAT_FIRST_ENT(sb, media) != first)
|
* The removed check compared the first FAT entry to a value dependent
|
||||||
|
* on the media field like this:
|
||||||
|
* == (0x0F00 | media), for FAT12
|
||||||
|
* == (0XFF00 | media), for FAT16
|
||||||
|
* == (0x0FFFFF | media), for FAT32
|
||||||
*/
|
*/
|
||||||
|
|
||||||
error = -EINVAL;
|
error = -EINVAL;
|
||||||
|
@ -64,7 +64,7 @@ int fat_clusters_flush(struct super_block *sb)
|
|||||||
struct buffer_head *bh;
|
struct buffer_head *bh;
|
||||||
struct fat_boot_fsinfo *fsinfo;
|
struct fat_boot_fsinfo *fsinfo;
|
||||||
|
|
||||||
if (sbi->fat_bits != 32)
|
if (!is_fat32(sbi))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
bh = sb_bread(sb, sbi->fsinfo_sector);
|
bh = sb_bread(sb, sbi->fsinfo_sector);
|
||||||
|
@ -565,6 +565,7 @@ const struct inode_operations hfsplus_dir_inode_operations = {
|
|||||||
.symlink = hfsplus_symlink,
|
.symlink = hfsplus_symlink,
|
||||||
.mknod = hfsplus_mknod,
|
.mknod = hfsplus_mknod,
|
||||||
.rename = hfsplus_rename,
|
.rename = hfsplus_rename,
|
||||||
|
.getattr = hfsplus_getattr,
|
||||||
.listxattr = hfsplus_listxattr,
|
.listxattr = hfsplus_listxattr,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -488,6 +488,8 @@ void hfsplus_inode_write_fork(struct inode *inode,
|
|||||||
struct hfsplus_fork_raw *fork);
|
struct hfsplus_fork_raw *fork);
|
||||||
int hfsplus_cat_read_inode(struct inode *inode, struct hfs_find_data *fd);
|
int hfsplus_cat_read_inode(struct inode *inode, struct hfs_find_data *fd);
|
||||||
int hfsplus_cat_write_inode(struct inode *inode);
|
int hfsplus_cat_write_inode(struct inode *inode);
|
||||||
|
int hfsplus_getattr(const struct path *path, struct kstat *stat,
|
||||||
|
u32 request_mask, unsigned int query_flags);
|
||||||
int hfsplus_file_fsync(struct file *file, loff_t start, loff_t end,
|
int hfsplus_file_fsync(struct file *file, loff_t start, loff_t end,
|
||||||
int datasync);
|
int datasync);
|
||||||
|
|
||||||
|
@ -270,6 +270,26 @@ static int hfsplus_setattr(struct dentry *dentry, struct iattr *attr)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int hfsplus_getattr(const struct path *path, struct kstat *stat,
|
||||||
|
u32 request_mask, unsigned int query_flags)
|
||||||
|
{
|
||||||
|
struct inode *inode = d_inode(path->dentry);
|
||||||
|
struct hfsplus_inode_info *hip = HFSPLUS_I(inode);
|
||||||
|
|
||||||
|
if (inode->i_flags & S_APPEND)
|
||||||
|
stat->attributes |= STATX_ATTR_APPEND;
|
||||||
|
if (inode->i_flags & S_IMMUTABLE)
|
||||||
|
stat->attributes |= STATX_ATTR_IMMUTABLE;
|
||||||
|
if (hip->userflags & HFSPLUS_FLG_NODUMP)
|
||||||
|
stat->attributes |= STATX_ATTR_NODUMP;
|
||||||
|
|
||||||
|
stat->attributes_mask |= STATX_ATTR_APPEND | STATX_ATTR_IMMUTABLE |
|
||||||
|
STATX_ATTR_NODUMP;
|
||||||
|
|
||||||
|
generic_fillattr(inode, stat);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
int hfsplus_file_fsync(struct file *file, loff_t start, loff_t end,
|
int hfsplus_file_fsync(struct file *file, loff_t start, loff_t end,
|
||||||
int datasync)
|
int datasync)
|
||||||
{
|
{
|
||||||
@ -329,6 +349,7 @@ int hfsplus_file_fsync(struct file *file, loff_t start, loff_t end,
|
|||||||
|
|
||||||
static const struct inode_operations hfsplus_file_inode_operations = {
|
static const struct inode_operations hfsplus_file_inode_operations = {
|
||||||
.setattr = hfsplus_setattr,
|
.setattr = hfsplus_setattr,
|
||||||
|
.getattr = hfsplus_getattr,
|
||||||
.listxattr = hfsplus_listxattr,
|
.listxattr = hfsplus_listxattr,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -30,6 +30,7 @@
|
|||||||
#include <linux/quotaops.h>
|
#include <linux/quotaops.h>
|
||||||
#include <linux/blkdev.h>
|
#include <linux/blkdev.h>
|
||||||
#include <linux/uio.h>
|
#include <linux/uio.h>
|
||||||
|
#include <linux/mm.h>
|
||||||
|
|
||||||
#include <cluster/masklog.h>
|
#include <cluster/masklog.h>
|
||||||
|
|
||||||
@ -397,7 +398,7 @@ static int ocfs2_readpages(struct file *filp, struct address_space *mapping,
|
|||||||
* Check whether a remote node truncated this file - we just
|
* Check whether a remote node truncated this file - we just
|
||||||
* drop out in that case as it's not worth handling here.
|
* drop out in that case as it's not worth handling here.
|
||||||
*/
|
*/
|
||||||
last = list_entry(pages->prev, struct page, lru);
|
last = lru_to_page(pages);
|
||||||
start = (loff_t)last->index << PAGE_SHIFT;
|
start = (loff_t)last->index << PAGE_SHIFT;
|
||||||
if (start >= i_size_read(inode))
|
if (start >= i_size_read(inode))
|
||||||
goto out_unlock;
|
goto out_unlock;
|
||||||
|
@ -77,7 +77,7 @@ static int orangefs_readpages(struct file *file,
|
|||||||
for (page_idx = 0; page_idx < nr_pages; page_idx++) {
|
for (page_idx = 0; page_idx < nr_pages; page_idx++) {
|
||||||
struct page *page;
|
struct page *page;
|
||||||
|
|
||||||
page = list_entry(pages->prev, struct page, lru);
|
page = lru_to_page(pages);
|
||||||
list_del(&page->lru);
|
list_del(&page->lru);
|
||||||
if (!add_to_page_cache(page,
|
if (!add_to_page_cache(page,
|
||||||
mapping,
|
mapping,
|
||||||
|
@ -105,7 +105,7 @@ static int wait_for_free(struct slot_map *m)
|
|||||||
left = t;
|
left = t;
|
||||||
else
|
else
|
||||||
left = t + (left - n);
|
left = t + (left - n);
|
||||||
if (unlikely(signal_pending(current)))
|
if (signal_pending(current))
|
||||||
left = -EINTR;
|
left = -EINTR;
|
||||||
} while (left > 0);
|
} while (left > 0);
|
||||||
|
|
||||||
|
@ -581,8 +581,10 @@ static int proc_pid_limits(struct seq_file *m, struct pid_namespace *ns,
|
|||||||
/*
|
/*
|
||||||
* print the file header
|
* print the file header
|
||||||
*/
|
*/
|
||||||
seq_printf(m, "%-25s %-20s %-20s %-10s\n",
|
seq_puts(m, "Limit "
|
||||||
"Limit", "Soft Limit", "Hard Limit", "Units");
|
"Soft Limit "
|
||||||
|
"Hard Limit "
|
||||||
|
"Units \n");
|
||||||
|
|
||||||
for (i = 0; i < RLIM_NLIMITS; i++) {
|
for (i = 0; i < RLIM_NLIMITS; i++) {
|
||||||
if (rlim[i].rlim_cur == RLIM_INFINITY)
|
if (rlim[i].rlim_cur == RLIM_INFINITY)
|
||||||
@ -2356,10 +2358,13 @@ static ssize_t timerslack_ns_write(struct file *file, const char __user *buf,
|
|||||||
return -ESRCH;
|
return -ESRCH;
|
||||||
|
|
||||||
if (p != current) {
|
if (p != current) {
|
||||||
if (!capable(CAP_SYS_NICE)) {
|
rcu_read_lock();
|
||||||
|
if (!ns_capable(__task_cred(p)->user_ns, CAP_SYS_NICE)) {
|
||||||
|
rcu_read_unlock();
|
||||||
count = -EPERM;
|
count = -EPERM;
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
rcu_read_unlock();
|
||||||
|
|
||||||
err = security_task_setscheduler(p);
|
err = security_task_setscheduler(p);
|
||||||
if (err) {
|
if (err) {
|
||||||
@ -2392,11 +2397,14 @@ static int timerslack_ns_show(struct seq_file *m, void *v)
|
|||||||
return -ESRCH;
|
return -ESRCH;
|
||||||
|
|
||||||
if (p != current) {
|
if (p != current) {
|
||||||
|
rcu_read_lock();
|
||||||
if (!capable(CAP_SYS_NICE)) {
|
if (!ns_capable(__task_cred(p)->user_ns, CAP_SYS_NICE)) {
|
||||||
|
rcu_read_unlock();
|
||||||
err = -EPERM;
|
err = -EPERM;
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
rcu_read_unlock();
|
||||||
|
|
||||||
err = security_task_getscheduler(p);
|
err = security_task_getscheduler(p);
|
||||||
if (err)
|
if (err)
|
||||||
goto out;
|
goto out;
|
||||||
|
@ -59,7 +59,6 @@ static struct kmem_cache *pde_opener_cache __ro_after_init;
|
|||||||
static struct inode *proc_alloc_inode(struct super_block *sb)
|
static struct inode *proc_alloc_inode(struct super_block *sb)
|
||||||
{
|
{
|
||||||
struct proc_inode *ei;
|
struct proc_inode *ei;
|
||||||
struct inode *inode;
|
|
||||||
|
|
||||||
ei = kmem_cache_alloc(proc_inode_cachep, GFP_KERNEL);
|
ei = kmem_cache_alloc(proc_inode_cachep, GFP_KERNEL);
|
||||||
if (!ei)
|
if (!ei)
|
||||||
@ -71,8 +70,7 @@ static struct inode *proc_alloc_inode(struct super_block *sb)
|
|||||||
ei->sysctl = NULL;
|
ei->sysctl = NULL;
|
||||||
ei->sysctl_entry = NULL;
|
ei->sysctl_entry = NULL;
|
||||||
ei->ns_ops = NULL;
|
ei->ns_ops = NULL;
|
||||||
inode = &ei->vfs_inode;
|
return &ei->vfs_inode;
|
||||||
return inode;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void proc_i_callback(struct rcu_head *head)
|
static void proc_i_callback(struct rcu_head *head)
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
#include <linux/dcache.h>
|
#include <linux/dcache.h>
|
||||||
|
#include "internal.h"
|
||||||
|
|
||||||
unsigned name_to_int(const struct qstr *qstr)
|
unsigned name_to_int(const struct qstr *qstr)
|
||||||
{
|
{
|
||||||
|
@ -9,7 +9,7 @@
|
|||||||
* This is defined the same way as ffs.
|
* This is defined the same way as ffs.
|
||||||
* Note fls(0) = 0, fls(1) = 1, fls(0x80000000) = 32.
|
* Note fls(0) = 0, fls(1) = 1, fls(0x80000000) = 32.
|
||||||
*/
|
*/
|
||||||
static __always_inline int fls(int x)
|
static __always_inline int fls(unsigned int x)
|
||||||
{
|
{
|
||||||
return x ? sizeof(x) * 8 - __builtin_clz(x) : 0;
|
return x ? sizeof(x) * 8 - __builtin_clz(x) : 0;
|
||||||
}
|
}
|
||||||
|
@ -10,7 +10,7 @@
|
|||||||
* Note fls(0) = 0, fls(1) = 1, fls(0x80000000) = 32.
|
* Note fls(0) = 0, fls(1) = 1, fls(0x80000000) = 32.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
static __always_inline int fls(int x)
|
static __always_inline int fls(unsigned int x)
|
||||||
{
|
{
|
||||||
int r = 32;
|
int r = 32;
|
||||||
|
|
||||||
|
@ -25,6 +25,7 @@ struct linux_binprm {
|
|||||||
#endif
|
#endif
|
||||||
struct mm_struct *mm;
|
struct mm_struct *mm;
|
||||||
unsigned long p; /* current top of mem */
|
unsigned long p; /* current top of mem */
|
||||||
|
unsigned long argmin; /* rlimit marker for copy_strings() */
|
||||||
unsigned int
|
unsigned int
|
||||||
/*
|
/*
|
||||||
* True after the bprm_set_creds hook has been called once
|
* True after the bprm_set_creds hook has been called once
|
||||||
|
@ -5,21 +5,8 @@
|
|||||||
#include <linux/compiler.h>
|
#include <linux/compiler.h>
|
||||||
|
|
||||||
#ifdef __CHECKER__
|
#ifdef __CHECKER__
|
||||||
#define __BUILD_BUG_ON_NOT_POWER_OF_2(n) (0)
|
|
||||||
#define BUILD_BUG_ON_NOT_POWER_OF_2(n) (0)
|
|
||||||
#define BUILD_BUG_ON_ZERO(e) (0)
|
#define BUILD_BUG_ON_ZERO(e) (0)
|
||||||
#define BUILD_BUG_ON_INVALID(e) (0)
|
|
||||||
#define BUILD_BUG_ON_MSG(cond, msg) (0)
|
|
||||||
#define BUILD_BUG_ON(condition) (0)
|
|
||||||
#define BUILD_BUG() (0)
|
|
||||||
#else /* __CHECKER__ */
|
#else /* __CHECKER__ */
|
||||||
|
|
||||||
/* Force a compilation error if a constant expression is not a power of 2 */
|
|
||||||
#define __BUILD_BUG_ON_NOT_POWER_OF_2(n) \
|
|
||||||
BUILD_BUG_ON(((n) & ((n) - 1)) != 0)
|
|
||||||
#define BUILD_BUG_ON_NOT_POWER_OF_2(n) \
|
|
||||||
BUILD_BUG_ON((n) == 0 || (((n) & ((n) - 1)) != 0))
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Force a compilation error if condition is true, but also produce a
|
* Force a compilation error if condition is true, but also produce a
|
||||||
* result (of value 0 and type size_t), so the expression can be used
|
* result (of value 0 and type size_t), so the expression can be used
|
||||||
@ -27,6 +14,13 @@
|
|||||||
* aren't permitted).
|
* aren't permitted).
|
||||||
*/
|
*/
|
||||||
#define BUILD_BUG_ON_ZERO(e) (sizeof(struct { int:(-!!(e)); }))
|
#define BUILD_BUG_ON_ZERO(e) (sizeof(struct { int:(-!!(e)); }))
|
||||||
|
#endif /* __CHECKER__ */
|
||||||
|
|
||||||
|
/* Force a compilation error if a constant expression is not a power of 2 */
|
||||||
|
#define __BUILD_BUG_ON_NOT_POWER_OF_2(n) \
|
||||||
|
BUILD_BUG_ON(((n) & ((n) - 1)) != 0)
|
||||||
|
#define BUILD_BUG_ON_NOT_POWER_OF_2(n) \
|
||||||
|
BUILD_BUG_ON((n) == 0 || (((n) & ((n) - 1)) != 0))
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* BUILD_BUG_ON_INVALID() permits the compiler to check the validity of the
|
* BUILD_BUG_ON_INVALID() permits the compiler to check the validity of the
|
||||||
@ -51,23 +45,9 @@
|
|||||||
* If you have some code which relies on certain constants being equal, or
|
* If you have some code which relies on certain constants being equal, or
|
||||||
* some other compile-time-evaluated condition, you should use BUILD_BUG_ON to
|
* some other compile-time-evaluated condition, you should use BUILD_BUG_ON to
|
||||||
* detect if someone changes it.
|
* detect if someone changes it.
|
||||||
*
|
|
||||||
* The implementation uses gcc's reluctance to create a negative array, but gcc
|
|
||||||
* (as of 4.4) only emits that error for obvious cases (e.g. not arguments to
|
|
||||||
* inline functions). Luckily, in 4.3 they added the "error" function
|
|
||||||
* attribute just for this type of case. Thus, we use a negative sized array
|
|
||||||
* (should always create an error on gcc versions older than 4.4) and then call
|
|
||||||
* an undefined function with the error attribute (should always create an
|
|
||||||
* error on gcc 4.3 and later). If for some reason, neither creates a
|
|
||||||
* compile-time error, we'll still have a link-time error, which is harder to
|
|
||||||
* track down.
|
|
||||||
*/
|
*/
|
||||||
#ifndef __OPTIMIZE__
|
|
||||||
#define BUILD_BUG_ON(condition) ((void)sizeof(char[1 - 2*!!(condition)]))
|
|
||||||
#else
|
|
||||||
#define BUILD_BUG_ON(condition) \
|
#define BUILD_BUG_ON(condition) \
|
||||||
BUILD_BUG_ON_MSG(condition, "BUILD_BUG_ON failed: " #condition)
|
BUILD_BUG_ON_MSG(condition, "BUILD_BUG_ON failed: " #condition)
|
||||||
#endif
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* BUILD_BUG - break compile if used.
|
* BUILD_BUG - break compile if used.
|
||||||
@ -78,6 +58,4 @@
|
|||||||
*/
|
*/
|
||||||
#define BUILD_BUG() BUILD_BUG_ON_MSG(1, "BUILD_BUG failed")
|
#define BUILD_BUG() BUILD_BUG_ON_MSG(1, "BUILD_BUG failed")
|
||||||
|
|
||||||
#endif /* __CHECKER__ */
|
|
||||||
|
|
||||||
#endif /* _LINUX_BUILD_BUG_H */
|
#endif /* _LINUX_BUILD_BUG_H */
|
||||||
|
@ -51,7 +51,8 @@ typedef unsigned long (*genpool_algo_t)(unsigned long *map,
|
|||||||
unsigned long size,
|
unsigned long size,
|
||||||
unsigned long start,
|
unsigned long start,
|
||||||
unsigned int nr,
|
unsigned int nr,
|
||||||
void *data, struct gen_pool *pool);
|
void *data, struct gen_pool *pool,
|
||||||
|
unsigned long start_addr);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* General purpose special memory pool descriptor.
|
* General purpose special memory pool descriptor.
|
||||||
@ -131,24 +132,24 @@ extern void gen_pool_set_algo(struct gen_pool *pool, genpool_algo_t algo,
|
|||||||
|
|
||||||
extern unsigned long gen_pool_first_fit(unsigned long *map, unsigned long size,
|
extern unsigned long gen_pool_first_fit(unsigned long *map, unsigned long size,
|
||||||
unsigned long start, unsigned int nr, void *data,
|
unsigned long start, unsigned int nr, void *data,
|
||||||
struct gen_pool *pool);
|
struct gen_pool *pool, unsigned long start_addr);
|
||||||
|
|
||||||
extern unsigned long gen_pool_fixed_alloc(unsigned long *map,
|
extern unsigned long gen_pool_fixed_alloc(unsigned long *map,
|
||||||
unsigned long size, unsigned long start, unsigned int nr,
|
unsigned long size, unsigned long start, unsigned int nr,
|
||||||
void *data, struct gen_pool *pool);
|
void *data, struct gen_pool *pool, unsigned long start_addr);
|
||||||
|
|
||||||
extern unsigned long gen_pool_first_fit_align(unsigned long *map,
|
extern unsigned long gen_pool_first_fit_align(unsigned long *map,
|
||||||
unsigned long size, unsigned long start, unsigned int nr,
|
unsigned long size, unsigned long start, unsigned int nr,
|
||||||
void *data, struct gen_pool *pool);
|
void *data, struct gen_pool *pool, unsigned long start_addr);
|
||||||
|
|
||||||
|
|
||||||
extern unsigned long gen_pool_first_fit_order_align(unsigned long *map,
|
extern unsigned long gen_pool_first_fit_order_align(unsigned long *map,
|
||||||
unsigned long size, unsigned long start, unsigned int nr,
|
unsigned long size, unsigned long start, unsigned int nr,
|
||||||
void *data, struct gen_pool *pool);
|
void *data, struct gen_pool *pool, unsigned long start_addr);
|
||||||
|
|
||||||
extern unsigned long gen_pool_best_fit(unsigned long *map, unsigned long size,
|
extern unsigned long gen_pool_best_fit(unsigned long *map, unsigned long size,
|
||||||
unsigned long start, unsigned int nr, void *data,
|
unsigned long start, unsigned int nr, void *data,
|
||||||
struct gen_pool *pool);
|
struct gen_pool *pool, unsigned long start_addr);
|
||||||
|
|
||||||
|
|
||||||
extern struct gen_pool *devm_gen_pool_create(struct device *dev,
|
extern struct gen_pool *devm_gen_pool_create(struct device *dev,
|
||||||
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user