- The series "resource: A couple of cleanups" from Andy Shevchenko

performs some cleanups in the resource management code.
 
 - The series "Improve the copy of task comm" from Yafang Shao addresses
   possible race-induced overflows in the management of task_struct.comm[].
 
 - The series "Remove unnecessary header includes from
   {tools/}lib/list_sort.c" from Kuan-Wei Chiu adds some cleanups and a
   small fix to the list_sort library code and to its selftest.
 
 - The series "Enhance min heap API with non-inline functions and
   optimizations" also from Kuan-Wei Chiu optimizes and cleans up the
   min_heap library code.
 
 - The series "nilfs2: Finish folio conversion" from Ryusuke Konishi
   finishes off nilfs2's folioification.
 
 - The series "add detect count for hung tasks" from Lance Yang adds more
   userspace visibility into the hung-task detector's activity.
 
 - Apart from that, singelton patches in many places - please see the
   individual changelogs for details.
 -----BEGIN PGP SIGNATURE-----
 
 iHUEABYIAB0WIQTTMBEPP41GrTpTJgfdBJ7gKXxAjgUCZ0L6lQAKCRDdBJ7gKXxA
 jmEIAPwMSglNPKRIOgzOvHh8MUJW1Dy8iKJ2kWCO3f6QTUIM2AEA+PazZbUd/g2m
 Ii8igH0UBibIgva7MrCyJedDI1O23AA=
 =8BIU
 -----END PGP SIGNATURE-----

Merge tag 'mm-nonmm-stable-2024-11-24-02-05' of git://git.kernel.org/pub/scm/linux/kernel/git/akpm/mm

Pull non-MM updates from Andrew Morton:

 - The series "resource: A couple of cleanups" from Andy Shevchenko
   performs some cleanups in the resource management code

 - The series "Improve the copy of task comm" from Yafang Shao addresses
   possible race-induced overflows in the management of
   task_struct.comm[]

 - The series "Remove unnecessary header includes from
   {tools/}lib/list_sort.c" from Kuan-Wei Chiu adds some cleanups and a
   small fix to the list_sort library code and to its selftest

 - The series "Enhance min heap API with non-inline functions and
   optimizations" also from Kuan-Wei Chiu optimizes and cleans up the
   min_heap library code

 - The series "nilfs2: Finish folio conversion" from Ryusuke Konishi
   finishes off nilfs2's folioification

 - The series "add detect count for hung tasks" from Lance Yang adds
   more userspace visibility into the hung-task detector's activity

 - Apart from that, singelton patches in many places - please see the
   individual changelogs for details

* tag 'mm-nonmm-stable-2024-11-24-02-05' of git://git.kernel.org/pub/scm/linux/kernel/git/akpm/mm: (71 commits)
  gdb: lx-symbols: do not error out on monolithic build
  kernel/reboot: replace sprintf() with sysfs_emit()
  lib: util_macros_kunit: add kunit test for util_macros.h
  util_macros.h: fix/rework find_closest() macros
  Improve consistency of '#error' directive messages
  ocfs2: fix uninitialized value in ocfs2_file_read_iter()
  hung_task: add docs for hung_task_detect_count
  hung_task: add detect count for hung tasks
  dma-buf: use atomic64_inc_return() in dma_buf_getfile()
  fs/proc/kcore.c: fix coccinelle reported ERROR instances
  resource: avoid unnecessary resource tree walking in __region_intersects()
  ocfs2: remove unused errmsg function and table
  ocfs2: cluster: fix a typo
  lib/scatterlist: use sg_phys() helper
  checkpatch: always parse orig_commit in fixes tag
  nilfs2: convert metadata aops from writepage to writepages
  nilfs2: convert nilfs_recovery_copy_block() to take a folio
  nilfs2: convert nilfs_page_count_clean_buffers() to take a folio
  nilfs2: remove nilfs_writepage
  nilfs2: convert checkpoint file to be folio-based
  ...
This commit is contained in:
Linus Torvalds 2024-11-25 16:09:48 -08:00
commit f5f4745a7f
102 changed files with 1952 additions and 896 deletions

View File

@ -401,6 +401,15 @@ The upper bound on the number of tasks that are checked.
This file shows up if ``CONFIG_DETECT_HUNG_TASK`` is enabled. This file shows up if ``CONFIG_DETECT_HUNG_TASK`` is enabled.
hung_task_detect_count
======================
Indicates the total number of tasks that have been detected as hung since
the system boot.
This file shows up if ``CONFIG_DETECT_HUNG_TASK`` is enabled.
hung_task_timeout_secs hung_task_timeout_secs
====================== ======================

View File

@ -52,6 +52,7 @@ Library functionality that is used throughout the kernel.
wrappers/atomic_bitops wrappers/atomic_bitops
floating-point floating-point
union_find union_find
min_heap
Low level entry and exit Low level entry and exit
======================== ========================

View File

@ -0,0 +1,300 @@
.. SPDX-License-Identifier: GPL-2.0
============
Min Heap API
============
Introduction
============
The Min Heap API provides a set of functions and macros for managing min-heaps
in the Linux kernel. A min-heap is a binary tree structure where the value of
each node is less than or equal to the values of its children, ensuring that
the smallest element is always at the root.
This document provides a guide to the Min Heap API, detailing how to define and
use min-heaps. Users should not directly call functions with **__min_heap_*()**
prefixes, but should instead use the provided macro wrappers.
In addition to the standard version of the functions, the API also includes a
set of inline versions for performance-critical scenarios. These inline
functions have the same names as their non-inline counterparts but include an
**_inline** suffix. For example, **__min_heap_init_inline** and its
corresponding macro wrapper **min_heap_init_inline**. The inline versions allow
custom comparison and swap functions to be called directly, rather than through
indirect function calls. This can significantly reduce overhead, especially
when CONFIG_MITIGATION_RETPOLINE is enabled, as indirect function calls become
more expensive. As with the non-inline versions, it is important to use the
macro wrappers for inline functions instead of directly calling the functions
themselves.
Data Structures
===============
Min-Heap Definition
-------------------
The core data structure for representing a min-heap is defined using the
**MIN_HEAP_PREALLOCATED** and **DEFINE_MIN_HEAP** macros. These macros allow
you to define a min-heap with a preallocated buffer or dynamically allocated
memory.
Example:
.. code-block:: c
#define MIN_HEAP_PREALLOCATED(_type, _name, _nr)
struct _name {
int nr; /* Number of elements in the heap */
int size; /* Maximum number of elements that can be held */
_type *data; /* Pointer to the heap data */
_type preallocated[_nr]; /* Static preallocated array */
}
#define DEFINE_MIN_HEAP(_type, _name) MIN_HEAP_PREALLOCATED(_type, _name, 0)
A typical heap structure will include a counter for the number of elements
(`nr`), the maximum capacity of the heap (`size`), and a pointer to an array of
elements (`data`). Optionally, you can specify a static array for preallocated
heap storage using **MIN_HEAP_PREALLOCATED**.
Min Heap Callbacks
------------------
The **struct min_heap_callbacks** provides customization options for ordering
elements in the heap and swapping them. It contains two function pointers:
.. code-block:: c
struct min_heap_callbacks {
bool (*less)(const void *lhs, const void *rhs, void *args);
void (*swp)(void *lhs, void *rhs, void *args);
};
- **less** is the comparison function used to establish the order of elements.
- **swp** is a function for swapping elements in the heap. If swp is set to
NULL, the default swap function will be used, which swaps the elements based on their size
Macro Wrappers
==============
The following macro wrappers are provided for interacting with the heap in a
user-friendly manner. Each macro corresponds to a function that operates on the
heap, and they abstract away direct calls to internal functions.
Each macro accepts various parameters that are detailed below.
Heap Initialization
--------------------
.. code-block:: c
min_heap_init(heap, data, size);
- **heap**: A pointer to the min-heap structure to be initialized.
- **data**: A pointer to the buffer where the heap elements will be stored. If
`NULL`, the preallocated buffer within the heap structure will be used.
- **size**: The maximum number of elements the heap can hold.
This macro initializes the heap, setting its initial state. If `data` is
`NULL`, the preallocated memory inside the heap structure will be used for
storage. Otherwise, the user-provided buffer is used. The operation is **O(1)**.
**Inline Version:** min_heap_init_inline(heap, data, size)
Accessing the Top Element
-------------------------
.. code-block:: c
element = min_heap_peek(heap);
- **heap**: A pointer to the min-heap from which to retrieve the smallest
element.
This macro returns a pointer to the smallest element (the root) of the heap, or
`NULL` if the heap is empty. The operation is **O(1)**.
**Inline Version:** min_heap_peek_inline(heap)
Heap Insertion
--------------
.. code-block:: c
success = min_heap_push(heap, element, callbacks, args);
- **heap**: A pointer to the min-heap into which the element should be inserted.
- **element**: A pointer to the element to be inserted into the heap.
- **callbacks**: A pointer to a `struct min_heap_callbacks` providing the
`less` and `swp` functions.
- **args**: Optional arguments passed to the `less` and `swp` functions.
This macro inserts an element into the heap. It returns `true` if the insertion
was successful and `false` if the heap is full. The operation is **O(log n)**.
**Inline Version:** min_heap_push_inline(heap, element, callbacks, args)
Heap Removal
------------
.. code-block:: c
success = min_heap_pop(heap, callbacks, args);
- **heap**: A pointer to the min-heap from which to remove the smallest element.
- **callbacks**: A pointer to a `struct min_heap_callbacks` providing the
`less` and `swp` functions.
- **args**: Optional arguments passed to the `less` and `swp` functions.
This macro removes the smallest element (the root) from the heap. It returns
`true` if the element was successfully removed, or `false` if the heap is
empty. The operation is **O(log n)**.
**Inline Version:** min_heap_pop_inline(heap, callbacks, args)
Heap Maintenance
----------------
You can use the following macros to maintain the heap's structure:
.. code-block:: c
min_heap_sift_down(heap, pos, callbacks, args);
- **heap**: A pointer to the min-heap.
- **pos**: The index from which to start sifting down.
- **callbacks**: A pointer to a `struct min_heap_callbacks` providing the
`less` and `swp` functions.
- **args**: Optional arguments passed to the `less` and `swp` functions.
This macro restores the heap property by moving the element at the specified
index (`pos`) down the heap until it is in the correct position. The operation
is **O(log n)**.
**Inline Version:** min_heap_sift_down_inline(heap, pos, callbacks, args)
.. code-block:: c
min_heap_sift_up(heap, idx, callbacks, args);
- **heap**: A pointer to the min-heap.
- **idx**: The index of the element to sift up.
- **callbacks**: A pointer to a `struct min_heap_callbacks` providing the
`less` and `swp` functions.
- **args**: Optional arguments passed to the `less` and `swp` functions.
This macro restores the heap property by moving the element at the specified
index (`idx`) up the heap. The operation is **O(log n)**.
**Inline Version:** min_heap_sift_up_inline(heap, idx, callbacks, args)
.. code-block:: c
min_heapify_all(heap, callbacks, args);
- **heap**: A pointer to the min-heap.
- **callbacks**: A pointer to a `struct min_heap_callbacks` providing the
`less` and `swp` functions.
- **args**: Optional arguments passed to the `less` and `swp` functions.
This macro ensures that the entire heap satisfies the heap property. It is
called when the heap is built from scratch or after many modifications. The
operation is **O(n)**.
**Inline Version:** min_heapify_all_inline(heap, callbacks, args)
Removing Specific Elements
--------------------------
.. code-block:: c
success = min_heap_del(heap, idx, callbacks, args);
- **heap**: A pointer to the min-heap.
- **idx**: The index of the element to delete.
- **callbacks**: A pointer to a `struct min_heap_callbacks` providing the
`less` and `swp` functions.
- **args**: Optional arguments passed to the `less` and `swp` functions.
This macro removes an element at the specified index (`idx`) from the heap and
restores the heap property. The operation is **O(log n)**.
**Inline Version:** min_heap_del_inline(heap, idx, callbacks, args)
Other Utilities
===============
- **min_heap_full(heap)**: Checks whether the heap is full.
Complexity: **O(1)**.
.. code-block:: c
bool full = min_heap_full(heap);
- `heap`: A pointer to the min-heap to check.
This macro returns `true` if the heap is full, otherwise `false`.
**Inline Version:** min_heap_full_inline(heap)
- **min_heap_empty(heap)**: Checks whether the heap is empty.
Complexity: **O(1)**.
.. code-block:: c
bool empty = min_heap_empty(heap);
- `heap`: A pointer to the min-heap to check.
This macro returns `true` if the heap is empty, otherwise `false`.
**Inline Version:** min_heap_empty_inline(heap)
Example Usage
=============
An example usage of the min-heap API would involve defining a heap structure,
initializing it, and inserting and removing elements as needed.
.. code-block:: c
#include <linux/min_heap.h>
int my_less_function(const void *lhs, const void *rhs, void *args) {
return (*(int *)lhs < *(int *)rhs);
}
struct min_heap_callbacks heap_cb = {
.less = my_less_function, /* Comparison function for heap order */
.swp = NULL, /* Use default swap function */
};
void example_usage(void) {
/* Pre-populate the buffer with elements */
int buffer[5] = {5, 2, 8, 1, 3};
/* Declare a min-heap */
DEFINE_MIN_HEAP(int, my_heap);
/* Initialize the heap with preallocated buffer and size */
min_heap_init(&my_heap, buffer, 5);
/* Build the heap using min_heapify_all */
my_heap.nr = 5; /* Set the number of elements in the heap */
min_heapify_all(&my_heap, &heap_cb, NULL);
/* Peek at the top element (should be 1 in this case) */
int *top = min_heap_peek(&my_heap);
pr_info("Top element: %d\n", *top);
/* Pop the top element (1) and get the new top (2) */
min_heap_pop(&my_heap, &heap_cb, NULL);
top = min_heap_peek(&my_heap);
pr_info("New top element: %d\n", *top);
/* Insert a new element (0) and recheck the top */
int new_element = 0;
min_heap_push(&my_heap, &new_element, &heap_cb, NULL);
top = min_heap_peek(&my_heap);
pr_info("Top element after insertion: %d\n", *top);
}

View File

@ -15585,6 +15585,15 @@ F: arch/arm/boot/dts/marvell/armada-xp-crs326-24g-2s.dts
F: arch/arm/boot/dts/marvell/armada-xp-crs328-4c-20s-4s-bit.dts F: arch/arm/boot/dts/marvell/armada-xp-crs328-4c-20s-4s-bit.dts
F: arch/arm/boot/dts/marvell/armada-xp-crs328-4c-20s-4s.dts F: arch/arm/boot/dts/marvell/armada-xp-crs328-4c-20s-4s.dts
MIN HEAP
M: Kuan-Wei Chiu <visitorckw@gmail.com>
L: linux-kernel@vger.kernel.org
S: Maintained
F: Documentation/core-api/min_heap.rst
F: include/linux/min_heap.h
F: lib/min_heap.c
F: lib/test_min_heap.c
MIPI CCS, SMIA AND SMIA++ IMAGE SENSOR DRIVER MIPI CCS, SMIA AND SMIA++ IMAGE SENSOR DRIVER
M: Sakari Ailus <sakari.ailus@linux.intel.com> M: Sakari Ailus <sakari.ailus@linux.intel.com>
L: linux-media@vger.kernel.org L: linux-media@vger.kernel.org

View File

@ -3,7 +3,7 @@
#define _ALPHA_SPINLOCK_TYPES_H #define _ALPHA_SPINLOCK_TYPES_H
#ifndef __LINUX_SPINLOCK_TYPES_RAW_H #ifndef __LINUX_SPINLOCK_TYPES_RAW_H
# error "please don't include this file directly" # error "Please do not include this file directly."
#endif #endif
typedef struct { typedef struct {

View File

@ -3,7 +3,7 @@
#define __ASM_SPINLOCK_TYPES_H #define __ASM_SPINLOCK_TYPES_H
#ifndef __LINUX_SPINLOCK_TYPES_RAW_H #ifndef __LINUX_SPINLOCK_TYPES_RAW_H
# error "please don't include this file directly" # error "Please do not include this file directly."
#endif #endif
#define TICKET_SHIFT 16 #define TICKET_SHIFT 16

View File

@ -6,7 +6,7 @@
#define __ASM_SPINLOCK_TYPES_H #define __ASM_SPINLOCK_TYPES_H
#if !defined(__LINUX_SPINLOCK_TYPES_RAW_H) && !defined(__ASM_SPINLOCK_H) #if !defined(__LINUX_SPINLOCK_TYPES_RAW_H) && !defined(__ASM_SPINLOCK_H)
# error "please don't include this file directly" # error "Please do not include this file directly."
#endif #endif
#include <asm-generic/qspinlock_types.h> #include <asm-generic/qspinlock_types.h>

View File

@ -9,7 +9,7 @@
#define _ASM_SPINLOCK_TYPES_H #define _ASM_SPINLOCK_TYPES_H
#ifndef __LINUX_SPINLOCK_TYPES_RAW_H #ifndef __LINUX_SPINLOCK_TYPES_RAW_H
# error "please don't include this file directly" # error "Please do not include this file directly."
#endif #endif
typedef struct { typedef struct {

View File

@ -3,7 +3,7 @@
#define _ASM_POWERPC_SIMPLE_SPINLOCK_TYPES_H #define _ASM_POWERPC_SIMPLE_SPINLOCK_TYPES_H
#ifndef __LINUX_SPINLOCK_TYPES_RAW_H #ifndef __LINUX_SPINLOCK_TYPES_RAW_H
# error "please don't include this file directly" # error "Please do not include this file directly."
#endif #endif
typedef struct { typedef struct {

View File

@ -3,7 +3,7 @@
#define _ASM_POWERPC_SPINLOCK_TYPES_H #define _ASM_POWERPC_SPINLOCK_TYPES_H
#ifndef __LINUX_SPINLOCK_TYPES_RAW_H #ifndef __LINUX_SPINLOCK_TYPES_RAW_H
# error "please don't include this file directly" # error "Please do not include this file directly."
#endif #endif
#ifdef CONFIG_PPC_QUEUED_SPINLOCKS #ifdef CONFIG_PPC_QUEUED_SPINLOCKS

View File

@ -3,7 +3,7 @@
#define __ASM_SPINLOCK_TYPES_H #define __ASM_SPINLOCK_TYPES_H
#ifndef __LINUX_SPINLOCK_TYPES_RAW_H #ifndef __LINUX_SPINLOCK_TYPES_RAW_H
# error "please don't include this file directly" # error "Please do not include this file directly."
#endif #endif
typedef struct { typedef struct {

View File

@ -3,7 +3,7 @@
#define __ASM_SH_SPINLOCK_TYPES_H #define __ASM_SH_SPINLOCK_TYPES_H
#ifndef __LINUX_SPINLOCK_TYPES_RAW_H #ifndef __LINUX_SPINLOCK_TYPES_RAW_H
# error "please don't include this file directly" # error "Please do not include this file directly."
#endif #endif
typedef struct { typedef struct {

View File

@ -3,7 +3,7 @@
#define __ASM_SPINLOCK_TYPES_H #define __ASM_SPINLOCK_TYPES_H
#if !defined(__LINUX_SPINLOCK_TYPES_RAW_H) && !defined(__ASM_SPINLOCK_H) #if !defined(__LINUX_SPINLOCK_TYPES_RAW_H) && !defined(__ASM_SPINLOCK_H)
# error "please don't include this file directly" # error "Please do not include this file directly."
#endif #endif
#include <asm-generic/qspinlock_types.h> #include <asm-generic/qspinlock_types.h>

View File

@ -870,7 +870,7 @@ int drm_framebuffer_init(struct drm_device *dev, struct drm_framebuffer *fb,
INIT_LIST_HEAD(&fb->filp_head); INIT_LIST_HEAD(&fb->filp_head);
fb->funcs = funcs; fb->funcs = funcs;
strcpy(fb->comm, current->comm); strscpy(fb->comm, current->comm);
ret = __drm_mode_object_add(dev, &fb->base, DRM_MODE_OBJECT_FB, ret = __drm_mode_object_add(dev, &fb->base, DRM_MODE_OBJECT_FB,
false, drm_framebuffer_free); false, drm_framebuffer_free);

View File

@ -1104,7 +1104,7 @@ i915_vma_coredump_create(const struct intel_gt *gt,
} }
INIT_LIST_HEAD(&dst->page_list); INIT_LIST_HEAD(&dst->page_list);
strcpy(dst->name, name); strscpy(dst->name, name);
dst->next = NULL; dst->next = NULL;
dst->gtt_offset = vma_res->start; dst->gtt_offset = vma_res->start;
@ -1404,7 +1404,7 @@ static bool record_context(struct i915_gem_context_coredump *e,
rcu_read_lock(); rcu_read_lock();
task = pid_task(ctx->pid, PIDTYPE_PID); task = pid_task(ctx->pid, PIDTYPE_PID);
if (task) { if (task) {
strcpy(e->comm, task->comm); strscpy(e->comm, task->comm);
e->pid = task->pid; e->pid = task->pid;
} }
rcu_read_unlock(); rcu_read_unlock();
@ -1450,7 +1450,7 @@ capture_vma_snapshot(struct intel_engine_capture_vma *next,
return next; return next;
} }
strcpy(c->name, name); strscpy(c->name, name);
c->vma_res = i915_vma_resource_get(vma_res); c->vma_res = i915_vma_resource_get(vma_res);
c->next = next; c->next = next;

View File

@ -5,6 +5,7 @@ config BCACHE
select BLOCK_HOLDER_DEPRECATED if SYSFS select BLOCK_HOLDER_DEPRECATED if SYSFS
select CRC64 select CRC64
select CLOSURES select CLOSURES
select MIN_HEAP
help help
Allows a block device to be used as cache for other devices; uses Allows a block device to be used as cache for other devices; uses
a btree for indexing and the layout is optimized for SSDs. a btree for indexing and the layout is optimized for SSDs.

View File

@ -189,23 +189,16 @@ static inline bool new_bucket_min_cmp(const void *l, const void *r, void *args)
return new_bucket_prio(ca, *lhs) < new_bucket_prio(ca, *rhs); return new_bucket_prio(ca, *lhs) < new_bucket_prio(ca, *rhs);
} }
static inline void new_bucket_swap(void *l, void *r, void __always_unused *args)
{
struct bucket **lhs = l, **rhs = r;
swap(*lhs, *rhs);
}
static void invalidate_buckets_lru(struct cache *ca) static void invalidate_buckets_lru(struct cache *ca)
{ {
struct bucket *b; struct bucket *b;
const struct min_heap_callbacks bucket_max_cmp_callback = { const struct min_heap_callbacks bucket_max_cmp_callback = {
.less = new_bucket_max_cmp, .less = new_bucket_max_cmp,
.swp = new_bucket_swap, .swp = NULL,
}; };
const struct min_heap_callbacks bucket_min_cmp_callback = { const struct min_heap_callbacks bucket_min_cmp_callback = {
.less = new_bucket_min_cmp, .less = new_bucket_min_cmp,
.swp = new_bucket_swap, .swp = NULL,
}; };
ca->heap.nr = 0; ca->heap.nr = 0;

View File

@ -1093,14 +1093,6 @@ static inline bool new_btree_iter_cmp(const void *l, const void *r, void __alway
return bkey_cmp(_l->k, _r->k) <= 0; return bkey_cmp(_l->k, _r->k) <= 0;
} }
static inline void new_btree_iter_swap(void *iter1, void *iter2, void __always_unused *args)
{
struct btree_iter_set *_iter1 = iter1;
struct btree_iter_set *_iter2 = iter2;
swap(*_iter1, *_iter2);
}
static inline bool btree_iter_end(struct btree_iter *iter) static inline bool btree_iter_end(struct btree_iter *iter)
{ {
return !iter->heap.nr; return !iter->heap.nr;
@ -1111,7 +1103,7 @@ void bch_btree_iter_push(struct btree_iter *iter, struct bkey *k,
{ {
const struct min_heap_callbacks callbacks = { const struct min_heap_callbacks callbacks = {
.less = new_btree_iter_cmp, .less = new_btree_iter_cmp,
.swp = new_btree_iter_swap, .swp = NULL,
}; };
if (k != end) if (k != end)
@ -1157,7 +1149,7 @@ static inline struct bkey *__bch_btree_iter_next(struct btree_iter *iter,
struct bkey *ret = NULL; struct bkey *ret = NULL;
const struct min_heap_callbacks callbacks = { const struct min_heap_callbacks callbacks = {
.less = cmp, .less = cmp,
.swp = new_btree_iter_swap, .swp = NULL,
}; };
if (!btree_iter_end(iter)) { if (!btree_iter_end(iter)) {
@ -1231,7 +1223,7 @@ static void btree_mergesort(struct btree_keys *b, struct bset *out,
: bch_ptr_invalid; : bch_ptr_invalid;
const struct min_heap_callbacks callbacks = { const struct min_heap_callbacks callbacks = {
.less = b->ops->sort_cmp, .less = b->ops->sort_cmp,
.swp = new_btree_iter_swap, .swp = NULL,
}; };
/* Heapify the iterator, using our comparison function */ /* Heapify the iterator, using our comparison function */

View File

@ -266,20 +266,12 @@ static bool new_bch_extent_sort_cmp(const void *l, const void *r, void __always_
return !(c ? c > 0 : _l->k < _r->k); return !(c ? c > 0 : _l->k < _r->k);
} }
static inline void new_btree_iter_swap(void *iter1, void *iter2, void __always_unused *args)
{
struct btree_iter_set *_iter1 = iter1;
struct btree_iter_set *_iter2 = iter2;
swap(*_iter1, *_iter2);
}
static struct bkey *bch_extent_sort_fixup(struct btree_iter *iter, static struct bkey *bch_extent_sort_fixup(struct btree_iter *iter,
struct bkey *tmp) struct bkey *tmp)
{ {
const struct min_heap_callbacks callbacks = { const struct min_heap_callbacks callbacks = {
.less = new_bch_extent_sort_cmp, .less = new_bch_extent_sort_cmp,
.swp = new_btree_iter_swap, .swp = NULL,
}; };
while (iter->heap.nr > 1) { while (iter->heap.nr > 1) {
struct btree_iter_set *top = iter->heap.data, *i = top + 1; struct btree_iter_set *top = iter->heap.data, *i = top + 1;

View File

@ -190,14 +190,6 @@ static bool new_bucket_cmp(const void *l, const void *r, void __always_unused *a
return GC_SECTORS_USED(*_l) >= GC_SECTORS_USED(*_r); return GC_SECTORS_USED(*_l) >= GC_SECTORS_USED(*_r);
} }
static void new_bucket_swap(void *l, void *r, void __always_unused *args)
{
struct bucket **_l = l;
struct bucket **_r = r;
swap(*_l, *_r);
}
static unsigned int bucket_heap_top(struct cache *ca) static unsigned int bucket_heap_top(struct cache *ca)
{ {
struct bucket *b; struct bucket *b;
@ -212,7 +204,7 @@ void bch_moving_gc(struct cache_set *c)
unsigned long sectors_to_move, reserve_sectors; unsigned long sectors_to_move, reserve_sectors;
const struct min_heap_callbacks callbacks = { const struct min_heap_callbacks callbacks = {
.less = new_bucket_cmp, .less = new_bucket_cmp,
.swp = new_bucket_swap, .swp = NULL,
}; };
if (!c->copy_gc_enabled) if (!c->copy_gc_enabled)

View File

@ -7,6 +7,7 @@ config DM_VDO
select DM_BUFIO select DM_BUFIO
select LZ4_COMPRESS select LZ4_COMPRESS
select LZ4_DECOMPRESS select LZ4_DECOMPRESS
select MIN_HEAP
help help
This device mapper target presents a block device with This device mapper target presents a block device with
deduplication, compression and thin-provisioning. deduplication, compression and thin-provisioning.

View File

@ -166,7 +166,7 @@ static void swap_mappings(void *item1, void *item2, void __always_unused *args)
static const struct min_heap_callbacks repair_min_heap = { static const struct min_heap_callbacks repair_min_heap = {
.less = mapping_is_less_than, .less = mapping_is_less_than,
.swp = swap_mappings, .swp = NULL,
}; };
static struct numbered_block_mapping *sort_next_heap_element(struct repair_completion *repair) static struct numbered_block_mapping *sort_next_heap_element(struct repair_completion *repair)

View File

@ -3301,17 +3301,9 @@ static bool slab_status_is_less_than(const void *item1, const void *item2,
return info1->slab_number < info2->slab_number; return info1->slab_number < info2->slab_number;
} }
static void swap_slab_statuses(void *item1, void *item2, void __always_unused *args)
{
struct slab_status *info1 = item1;
struct slab_status *info2 = item2;
swap(*info1, *info2);
}
static const struct min_heap_callbacks slab_status_min_heap = { static const struct min_heap_callbacks slab_status_min_heap = {
.less = slab_status_is_less_than, .less = slab_status_is_less_than,
.swp = swap_slab_statuses, .swp = NULL,
}; };
/* Inform the slab actor that a action has finished on some slab; used by apply_to_slabs(). */ /* Inform the slab actor that a action has finished on some slab; used by apply_to_slabs(). */

View File

@ -24,6 +24,7 @@ config BCACHEFS_FS
select XXHASH select XXHASH
select SRCU select SRCU
select SYMBOLIC_ERRNAME select SYMBOLIC_ERRNAME
select MIN_HEAP
help help
The bcachefs filesystem - a modern, copy on write filesystem, with The bcachefs filesystem - a modern, copy on write filesystem, with
support for multiple devices, compression, checksumming, etc. support for multiple devices, compression, checksumming, etc.

View File

@ -14,21 +14,13 @@ static inline bool io_timer_cmp(const void *l, const void *r, void __always_unus
return (*_l)->expire < (*_r)->expire; return (*_l)->expire < (*_r)->expire;
} }
static inline void io_timer_swp(void *l, void *r, void __always_unused *args) static const struct min_heap_callbacks callbacks = {
{ .less = io_timer_cmp,
struct io_timer **_l = (struct io_timer **)l; .swp = NULL,
struct io_timer **_r = (struct io_timer **)r; };
swap(*_l, *_r);
}
void bch2_io_timer_add(struct io_clock *clock, struct io_timer *timer) void bch2_io_timer_add(struct io_clock *clock, struct io_timer *timer)
{ {
const struct min_heap_callbacks callbacks = {
.less = io_timer_cmp,
.swp = io_timer_swp,
};
spin_lock(&clock->timer_lock); spin_lock(&clock->timer_lock);
if (time_after_eq64((u64) atomic64_read(&clock->now), timer->expire)) { if (time_after_eq64((u64) atomic64_read(&clock->now), timer->expire)) {
@ -48,11 +40,6 @@ out:
void bch2_io_timer_del(struct io_clock *clock, struct io_timer *timer) void bch2_io_timer_del(struct io_clock *clock, struct io_timer *timer)
{ {
const struct min_heap_callbacks callbacks = {
.less = io_timer_cmp,
.swp = io_timer_swp,
};
spin_lock(&clock->timer_lock); spin_lock(&clock->timer_lock);
for (size_t i = 0; i < clock->timers.nr; i++) for (size_t i = 0; i < clock->timers.nr; i++)
@ -142,10 +129,6 @@ void bch2_kthread_io_clock_wait(struct io_clock *clock,
static struct io_timer *get_expired_timer(struct io_clock *clock, u64 now) static struct io_timer *get_expired_timer(struct io_clock *clock, u64 now)
{ {
struct io_timer *ret = NULL; struct io_timer *ret = NULL;
const struct min_heap_callbacks callbacks = {
.less = io_timer_cmp,
.swp = io_timer_swp,
};
if (clock->timers.nr && if (clock->timers.nr &&
time_after_eq64(now, clock->timers.data[0]->expire)) { time_after_eq64(now, clock->timers.data[0]->expire)) {

View File

@ -1057,6 +1057,11 @@ static inline void ec_stripes_heap_swap(void *l, void *r, void *h)
ec_stripes_heap_set_backpointer(_h, j); ec_stripes_heap_set_backpointer(_h, j);
} }
static const struct min_heap_callbacks callbacks = {
.less = ec_stripes_heap_cmp,
.swp = ec_stripes_heap_swap,
};
static void heap_verify_backpointer(struct bch_fs *c, size_t idx) static void heap_verify_backpointer(struct bch_fs *c, size_t idx)
{ {
ec_stripes_heap *h = &c->ec_stripes_heap; ec_stripes_heap *h = &c->ec_stripes_heap;
@ -1069,11 +1074,6 @@ static void heap_verify_backpointer(struct bch_fs *c, size_t idx)
void bch2_stripes_heap_del(struct bch_fs *c, void bch2_stripes_heap_del(struct bch_fs *c,
struct stripe *m, size_t idx) struct stripe *m, size_t idx)
{ {
const struct min_heap_callbacks callbacks = {
.less = ec_stripes_heap_cmp,
.swp = ec_stripes_heap_swap,
};
mutex_lock(&c->ec_stripes_heap_lock); mutex_lock(&c->ec_stripes_heap_lock);
heap_verify_backpointer(c, idx); heap_verify_backpointer(c, idx);
@ -1084,11 +1084,6 @@ void bch2_stripes_heap_del(struct bch_fs *c,
void bch2_stripes_heap_insert(struct bch_fs *c, void bch2_stripes_heap_insert(struct bch_fs *c,
struct stripe *m, size_t idx) struct stripe *m, size_t idx)
{ {
const struct min_heap_callbacks callbacks = {
.less = ec_stripes_heap_cmp,
.swp = ec_stripes_heap_swap,
};
mutex_lock(&c->ec_stripes_heap_lock); mutex_lock(&c->ec_stripes_heap_lock);
BUG_ON(min_heap_full(&c->ec_stripes_heap)); BUG_ON(min_heap_full(&c->ec_stripes_heap));
@ -1107,10 +1102,6 @@ void bch2_stripes_heap_insert(struct bch_fs *c,
void bch2_stripes_heap_update(struct bch_fs *c, void bch2_stripes_heap_update(struct bch_fs *c,
struct stripe *m, size_t idx) struct stripe *m, size_t idx)
{ {
const struct min_heap_callbacks callbacks = {
.less = ec_stripes_heap_cmp,
.swp = ec_stripes_heap_swap,
};
ec_stripes_heap *h = &c->ec_stripes_heap; ec_stripes_heap *h = &c->ec_stripes_heap;
bool do_deletes; bool do_deletes;
size_t i; size_t i;

View File

@ -1189,16 +1189,6 @@ static int unshare_sighand(struct task_struct *me)
return 0; return 0;
} }
char *__get_task_comm(char *buf, size_t buf_size, struct task_struct *tsk)
{
task_lock(tsk);
/* Always NUL terminated and zero-padded */
strscpy_pad(buf, tsk->comm, buf_size);
task_unlock(tsk);
return buf;
}
EXPORT_SYMBOL_GPL(__get_task_comm);
/* /*
* These functions flushes out all traces of the currently running executable * These functions flushes out all traces of the currently running executable
* so that a new one can be started * so that a new one can be started

View File

@ -177,12 +177,14 @@ nilfs_palloc_entry_blkoff(const struct inode *inode, __u64 nr)
* nilfs_palloc_desc_block_init - initialize buffer of a group descriptor block * nilfs_palloc_desc_block_init - initialize buffer of a group descriptor block
* @inode: inode of metadata file * @inode: inode of metadata file
* @bh: buffer head of the buffer to be initialized * @bh: buffer head of the buffer to be initialized
* @kaddr: kernel address mapped for the page including the buffer * @from: kernel address mapped for a chunk of the block
*
* This function does not yet support the case where block size > PAGE_SIZE.
*/ */
static void nilfs_palloc_desc_block_init(struct inode *inode, static void nilfs_palloc_desc_block_init(struct inode *inode,
struct buffer_head *bh, void *kaddr) struct buffer_head *bh, void *from)
{ {
struct nilfs_palloc_group_desc *desc = kaddr + bh_offset(bh); struct nilfs_palloc_group_desc *desc = from;
unsigned long n = nilfs_palloc_groups_per_desc_block(inode); unsigned long n = nilfs_palloc_groups_per_desc_block(inode);
__le32 nfrees; __le32 nfrees;
@ -337,38 +339,55 @@ static int nilfs_palloc_delete_entry_block(struct inode *inode, __u64 nr)
} }
/** /**
* nilfs_palloc_block_get_group_desc - get kernel address of a group descriptor * nilfs_palloc_group_desc_offset - calculate the byte offset of a group
* descriptor in the folio containing it
* @inode: inode of metadata file using this allocator * @inode: inode of metadata file using this allocator
* @group: group number * @group: group number
* @bh: buffer head of the buffer storing the group descriptor block * @bh: buffer head of the group descriptor block
* @kaddr: kernel address mapped for the page including the buffer *
* Return: Byte offset in the folio of the group descriptor for @group.
*/ */
static struct nilfs_palloc_group_desc * static size_t nilfs_palloc_group_desc_offset(const struct inode *inode,
nilfs_palloc_block_get_group_desc(const struct inode *inode,
unsigned long group, unsigned long group,
const struct buffer_head *bh, void *kaddr) const struct buffer_head *bh)
{ {
return (struct nilfs_palloc_group_desc *)(kaddr + bh_offset(bh)) + return offset_in_folio(bh->b_folio, bh->b_data) +
group % nilfs_palloc_groups_per_desc_block(inode); sizeof(struct nilfs_palloc_group_desc) *
(group % nilfs_palloc_groups_per_desc_block(inode));
} }
/** /**
* nilfs_palloc_block_get_entry - get kernel address of an entry * nilfs_palloc_bitmap_offset - calculate the byte offset of a bitmap block
* in the folio containing it
* @bh: buffer head of the bitmap block
*
* Return: Byte offset in the folio of the bitmap block for @bh.
*/
static size_t nilfs_palloc_bitmap_offset(const struct buffer_head *bh)
{
return offset_in_folio(bh->b_folio, bh->b_data);
}
/**
* nilfs_palloc_entry_offset - calculate the byte offset of an entry in the
* folio containing it
* @inode: inode of metadata file using this allocator * @inode: inode of metadata file using this allocator
* @nr: serial number of the entry (e.g. inode number) * @nr: serial number of the entry (e.g. inode number)
* @bh: buffer head of the buffer storing the entry block * @bh: buffer head of the entry block
* @kaddr: kernel address mapped for the page including the buffer *
* Return: Byte offset in the folio of the entry @nr.
*/ */
void *nilfs_palloc_block_get_entry(const struct inode *inode, __u64 nr, size_t nilfs_palloc_entry_offset(const struct inode *inode, __u64 nr,
const struct buffer_head *bh, void *kaddr) const struct buffer_head *bh)
{ {
unsigned long entry_offset, group_offset; unsigned long entry_index_in_group, entry_index_in_block;
nilfs_palloc_group(inode, nr, &group_offset); nilfs_palloc_group(inode, nr, &entry_index_in_group);
entry_offset = group_offset % NILFS_MDT(inode)->mi_entries_per_block; entry_index_in_block = entry_index_in_group %
NILFS_MDT(inode)->mi_entries_per_block;
return kaddr + bh_offset(bh) + return offset_in_folio(bh->b_folio, bh->b_data) +
entry_offset * NILFS_MDT(inode)->mi_entry_size; entry_index_in_block * NILFS_MDT(inode)->mi_entry_size;
} }
/** /**
@ -506,7 +525,7 @@ int nilfs_palloc_prepare_alloc_entry(struct inode *inode,
struct buffer_head *desc_bh, *bitmap_bh; struct buffer_head *desc_bh, *bitmap_bh;
struct nilfs_palloc_group_desc *desc; struct nilfs_palloc_group_desc *desc;
unsigned char *bitmap; unsigned char *bitmap;
void *desc_kaddr, *bitmap_kaddr; size_t doff, boff;
unsigned long group, maxgroup, ngroups; unsigned long group, maxgroup, ngroups;
unsigned long group_offset, maxgroup_offset; unsigned long group_offset, maxgroup_offset;
unsigned long n, entries_per_group; unsigned long n, entries_per_group;
@ -529,17 +548,17 @@ int nilfs_palloc_prepare_alloc_entry(struct inode *inode,
ret = nilfs_palloc_get_desc_block(inode, group, 1, &desc_bh); ret = nilfs_palloc_get_desc_block(inode, group, 1, &desc_bh);
if (ret < 0) if (ret < 0)
return ret; return ret;
desc_kaddr = kmap_local_page(desc_bh->b_page);
desc = nilfs_palloc_block_get_group_desc( doff = nilfs_palloc_group_desc_offset(inode, group, desc_bh);
inode, group, desc_bh, desc_kaddr); desc = kmap_local_folio(desc_bh->b_folio, doff);
n = nilfs_palloc_rest_groups_in_desc_block(inode, group, n = nilfs_palloc_rest_groups_in_desc_block(inode, group,
maxgroup); maxgroup);
for (j = 0; j < n; j++, desc++, group++, group_offset = 0) { for (j = 0; j < n; j++, group++, group_offset = 0) {
lock = nilfs_mdt_bgl_lock(inode, group); lock = nilfs_mdt_bgl_lock(inode, group);
if (nilfs_palloc_group_desc_nfrees(desc, lock) == 0) if (nilfs_palloc_group_desc_nfrees(&desc[j], lock) == 0)
continue; continue;
kunmap_local(desc_kaddr); kunmap_local(desc);
ret = nilfs_palloc_get_bitmap_block(inode, group, 1, ret = nilfs_palloc_get_bitmap_block(inode, group, 1,
&bitmap_bh); &bitmap_bh);
if (unlikely(ret < 0)) { if (unlikely(ret < 0)) {
@ -547,12 +566,14 @@ int nilfs_palloc_prepare_alloc_entry(struct inode *inode,
return ret; return ret;
} }
desc_kaddr = kmap_local_page(desc_bh->b_page); /*
desc = nilfs_palloc_block_get_group_desc( * Re-kmap the folio containing the first (and
inode, group, desc_bh, desc_kaddr); * subsequent) group descriptors.
*/
desc = kmap_local_folio(desc_bh->b_folio, doff);
bitmap_kaddr = kmap_local_page(bitmap_bh->b_page); boff = nilfs_palloc_bitmap_offset(bitmap_bh);
bitmap = bitmap_kaddr + bh_offset(bitmap_bh); bitmap = kmap_local_folio(bitmap_bh->b_folio, boff);
pos = nilfs_palloc_find_available_slot( pos = nilfs_palloc_find_available_slot(
bitmap, group_offset, entries_per_group, lock, bitmap, group_offset, entries_per_group, lock,
wrap); wrap);
@ -562,14 +583,14 @@ int nilfs_palloc_prepare_alloc_entry(struct inode *inode,
* beginning, the wrap flag only has an effect on the * beginning, the wrap flag only has an effect on the
* first search. * first search.
*/ */
kunmap_local(bitmap_kaddr); kunmap_local(bitmap);
if (pos >= 0) if (pos >= 0)
goto found; goto found;
brelse(bitmap_bh); brelse(bitmap_bh);
} }
kunmap_local(desc_kaddr); kunmap_local(desc);
brelse(desc_bh); brelse(desc_bh);
} }
@ -578,9 +599,9 @@ int nilfs_palloc_prepare_alloc_entry(struct inode *inode,
found: found:
/* found a free entry */ /* found a free entry */
nilfs_palloc_group_desc_add_entries(desc, lock, -1); nilfs_palloc_group_desc_add_entries(&desc[j], lock, -1);
req->pr_entry_nr = entries_per_group * group + pos; req->pr_entry_nr = entries_per_group * group + pos;
kunmap_local(desc_kaddr); kunmap_local(desc);
req->pr_desc_bh = desc_bh; req->pr_desc_bh = desc_bh;
req->pr_bitmap_bh = bitmap_bh; req->pr_bitmap_bh = bitmap_bh;
@ -611,18 +632,18 @@ void nilfs_palloc_commit_alloc_entry(struct inode *inode,
void nilfs_palloc_commit_free_entry(struct inode *inode, void nilfs_palloc_commit_free_entry(struct inode *inode,
struct nilfs_palloc_req *req) struct nilfs_palloc_req *req)
{ {
struct nilfs_palloc_group_desc *desc;
unsigned long group, group_offset; unsigned long group, group_offset;
size_t doff, boff;
struct nilfs_palloc_group_desc *desc;
unsigned char *bitmap; unsigned char *bitmap;
void *desc_kaddr, *bitmap_kaddr;
spinlock_t *lock; spinlock_t *lock;
group = nilfs_palloc_group(inode, req->pr_entry_nr, &group_offset); group = nilfs_palloc_group(inode, req->pr_entry_nr, &group_offset);
desc_kaddr = kmap_local_page(req->pr_desc_bh->b_page); doff = nilfs_palloc_group_desc_offset(inode, group, req->pr_desc_bh);
desc = nilfs_palloc_block_get_group_desc(inode, group, desc = kmap_local_folio(req->pr_desc_bh->b_folio, doff);
req->pr_desc_bh, desc_kaddr);
bitmap_kaddr = kmap_local_page(req->pr_bitmap_bh->b_page); boff = nilfs_palloc_bitmap_offset(req->pr_bitmap_bh);
bitmap = bitmap_kaddr + bh_offset(req->pr_bitmap_bh); bitmap = kmap_local_folio(req->pr_bitmap_bh->b_folio, boff);
lock = nilfs_mdt_bgl_lock(inode, group); lock = nilfs_mdt_bgl_lock(inode, group);
if (!nilfs_clear_bit_atomic(lock, group_offset, bitmap)) if (!nilfs_clear_bit_atomic(lock, group_offset, bitmap))
@ -633,8 +654,8 @@ void nilfs_palloc_commit_free_entry(struct inode *inode,
else else
nilfs_palloc_group_desc_add_entries(desc, lock, 1); nilfs_palloc_group_desc_add_entries(desc, lock, 1);
kunmap_local(bitmap_kaddr); kunmap_local(bitmap);
kunmap_local(desc_kaddr); kunmap_local(desc);
mark_buffer_dirty(req->pr_desc_bh); mark_buffer_dirty(req->pr_desc_bh);
mark_buffer_dirty(req->pr_bitmap_bh); mark_buffer_dirty(req->pr_bitmap_bh);
@ -653,17 +674,17 @@ void nilfs_palloc_abort_alloc_entry(struct inode *inode,
struct nilfs_palloc_req *req) struct nilfs_palloc_req *req)
{ {
struct nilfs_palloc_group_desc *desc; struct nilfs_palloc_group_desc *desc;
void *desc_kaddr, *bitmap_kaddr; size_t doff, boff;
unsigned char *bitmap; unsigned char *bitmap;
unsigned long group, group_offset; unsigned long group, group_offset;
spinlock_t *lock; spinlock_t *lock;
group = nilfs_palloc_group(inode, req->pr_entry_nr, &group_offset); group = nilfs_palloc_group(inode, req->pr_entry_nr, &group_offset);
desc_kaddr = kmap_local_page(req->pr_desc_bh->b_page); doff = nilfs_palloc_group_desc_offset(inode, group, req->pr_desc_bh);
desc = nilfs_palloc_block_get_group_desc(inode, group, desc = kmap_local_folio(req->pr_desc_bh->b_folio, doff);
req->pr_desc_bh, desc_kaddr);
bitmap_kaddr = kmap_local_page(req->pr_bitmap_bh->b_page); boff = nilfs_palloc_bitmap_offset(req->pr_bitmap_bh);
bitmap = bitmap_kaddr + bh_offset(req->pr_bitmap_bh); bitmap = kmap_local_folio(req->pr_bitmap_bh->b_folio, boff);
lock = nilfs_mdt_bgl_lock(inode, group); lock = nilfs_mdt_bgl_lock(inode, group);
if (!nilfs_clear_bit_atomic(lock, group_offset, bitmap)) if (!nilfs_clear_bit_atomic(lock, group_offset, bitmap))
@ -674,8 +695,8 @@ void nilfs_palloc_abort_alloc_entry(struct inode *inode,
else else
nilfs_palloc_group_desc_add_entries(desc, lock, 1); nilfs_palloc_group_desc_add_entries(desc, lock, 1);
kunmap_local(bitmap_kaddr); kunmap_local(bitmap);
kunmap_local(desc_kaddr); kunmap_local(desc);
brelse(req->pr_bitmap_bh); brelse(req->pr_bitmap_bh);
brelse(req->pr_desc_bh); brelse(req->pr_desc_bh);
@ -739,7 +760,7 @@ int nilfs_palloc_freev(struct inode *inode, __u64 *entry_nrs, size_t nitems)
struct buffer_head *desc_bh, *bitmap_bh; struct buffer_head *desc_bh, *bitmap_bh;
struct nilfs_palloc_group_desc *desc; struct nilfs_palloc_group_desc *desc;
unsigned char *bitmap; unsigned char *bitmap;
void *desc_kaddr, *bitmap_kaddr; size_t doff, boff;
unsigned long group, group_offset; unsigned long group, group_offset;
__u64 group_min_nr, last_nrs[8]; __u64 group_min_nr, last_nrs[8];
const unsigned long epg = nilfs_palloc_entries_per_group(inode); const unsigned long epg = nilfs_palloc_entries_per_group(inode);
@ -767,8 +788,8 @@ int nilfs_palloc_freev(struct inode *inode, __u64 *entry_nrs, size_t nitems)
/* Get the first entry number of the group */ /* Get the first entry number of the group */
group_min_nr = (__u64)group * epg; group_min_nr = (__u64)group * epg;
bitmap_kaddr = kmap_local_page(bitmap_bh->b_page); boff = nilfs_palloc_bitmap_offset(bitmap_bh);
bitmap = bitmap_kaddr + bh_offset(bitmap_bh); bitmap = kmap_local_folio(bitmap_bh->b_folio, boff);
lock = nilfs_mdt_bgl_lock(inode, group); lock = nilfs_mdt_bgl_lock(inode, group);
j = i; j = i;
@ -813,7 +834,7 @@ int nilfs_palloc_freev(struct inode *inode, __u64 *entry_nrs, size_t nitems)
entry_start = rounddown(group_offset, epb); entry_start = rounddown(group_offset, epb);
} while (true); } while (true);
kunmap_local(bitmap_kaddr); kunmap_local(bitmap);
mark_buffer_dirty(bitmap_bh); mark_buffer_dirty(bitmap_bh);
brelse(bitmap_bh); brelse(bitmap_bh);
@ -827,11 +848,10 @@ int nilfs_palloc_freev(struct inode *inode, __u64 *entry_nrs, size_t nitems)
inode->i_ino); inode->i_ino);
} }
desc_kaddr = kmap_local_page(desc_bh->b_page); doff = nilfs_palloc_group_desc_offset(inode, group, desc_bh);
desc = nilfs_palloc_block_get_group_desc( desc = kmap_local_folio(desc_bh->b_folio, doff);
inode, group, desc_bh, desc_kaddr);
nfree = nilfs_palloc_group_desc_add_entries(desc, lock, n); nfree = nilfs_palloc_group_desc_add_entries(desc, lock, n);
kunmap_local(desc_kaddr); kunmap_local(desc);
mark_buffer_dirty(desc_bh); mark_buffer_dirty(desc_bh);
nilfs_mdt_mark_dirty(inode); nilfs_mdt_mark_dirty(inode);
brelse(desc_bh); brelse(desc_bh);

View File

@ -31,8 +31,8 @@ nilfs_palloc_entries_per_group(const struct inode *inode)
int nilfs_palloc_init_blockgroup(struct inode *, unsigned int); int nilfs_palloc_init_blockgroup(struct inode *, unsigned int);
int nilfs_palloc_get_entry_block(struct inode *, __u64, int, int nilfs_palloc_get_entry_block(struct inode *, __u64, int,
struct buffer_head **); struct buffer_head **);
void *nilfs_palloc_block_get_entry(const struct inode *, __u64, size_t nilfs_palloc_entry_offset(const struct inode *inode, __u64 nr,
const struct buffer_head *, void *); const struct buffer_head *bh);
int nilfs_palloc_count_max_entries(struct inode *, u64, u64 *); int nilfs_palloc_count_max_entries(struct inode *, u64, u64 *);

View File

@ -68,54 +68,41 @@ static inline int nilfs_cpfile_is_in_first(const struct inode *cpfile,
static unsigned int static unsigned int
nilfs_cpfile_block_add_valid_checkpoints(const struct inode *cpfile, nilfs_cpfile_block_add_valid_checkpoints(const struct inode *cpfile,
struct buffer_head *bh, struct buffer_head *bh,
void *kaddr,
unsigned int n) unsigned int n)
{ {
struct nilfs_checkpoint *cp = kaddr + bh_offset(bh); struct nilfs_checkpoint *cp;
unsigned int count; unsigned int count;
cp = kmap_local_folio(bh->b_folio,
offset_in_folio(bh->b_folio, bh->b_data));
count = le32_to_cpu(cp->cp_checkpoints_count) + n; count = le32_to_cpu(cp->cp_checkpoints_count) + n;
cp->cp_checkpoints_count = cpu_to_le32(count); cp->cp_checkpoints_count = cpu_to_le32(count);
kunmap_local(cp);
return count; return count;
} }
static unsigned int static unsigned int
nilfs_cpfile_block_sub_valid_checkpoints(const struct inode *cpfile, nilfs_cpfile_block_sub_valid_checkpoints(const struct inode *cpfile,
struct buffer_head *bh, struct buffer_head *bh,
void *kaddr,
unsigned int n) unsigned int n)
{ {
struct nilfs_checkpoint *cp = kaddr + bh_offset(bh); struct nilfs_checkpoint *cp;
unsigned int count; unsigned int count;
cp = kmap_local_folio(bh->b_folio,
offset_in_folio(bh->b_folio, bh->b_data));
WARN_ON(le32_to_cpu(cp->cp_checkpoints_count) < n); WARN_ON(le32_to_cpu(cp->cp_checkpoints_count) < n);
count = le32_to_cpu(cp->cp_checkpoints_count) - n; count = le32_to_cpu(cp->cp_checkpoints_count) - n;
cp->cp_checkpoints_count = cpu_to_le32(count); cp->cp_checkpoints_count = cpu_to_le32(count);
kunmap_local(cp);
return count; return count;
} }
static inline struct nilfs_cpfile_header *
nilfs_cpfile_block_get_header(const struct inode *cpfile,
struct buffer_head *bh,
void *kaddr)
{
return kaddr + bh_offset(bh);
}
static struct nilfs_checkpoint *
nilfs_cpfile_block_get_checkpoint(const struct inode *cpfile, __u64 cno,
struct buffer_head *bh,
void *kaddr)
{
return kaddr + bh_offset(bh) + nilfs_cpfile_get_offset(cpfile, cno) *
NILFS_MDT(cpfile)->mi_entry_size;
}
static void nilfs_cpfile_block_init(struct inode *cpfile, static void nilfs_cpfile_block_init(struct inode *cpfile,
struct buffer_head *bh, struct buffer_head *bh,
void *kaddr) void *from)
{ {
struct nilfs_checkpoint *cp = kaddr + bh_offset(bh); struct nilfs_checkpoint *cp = from;
size_t cpsz = NILFS_MDT(cpfile)->mi_entry_size; size_t cpsz = NILFS_MDT(cpfile)->mi_entry_size;
int n = nilfs_cpfile_checkpoints_per_block(cpfile); int n = nilfs_cpfile_checkpoints_per_block(cpfile);
@ -125,6 +112,54 @@ static void nilfs_cpfile_block_init(struct inode *cpfile,
} }
} }
/**
* nilfs_cpfile_checkpoint_offset - calculate the byte offset of a checkpoint
* entry in the folio containing it
* @cpfile: checkpoint file inode
* @cno: checkpoint number
* @bh: buffer head of block containing checkpoint indexed by @cno
*
* Return: Byte offset in the folio of the checkpoint specified by @cno.
*/
static size_t nilfs_cpfile_checkpoint_offset(const struct inode *cpfile,
__u64 cno,
struct buffer_head *bh)
{
return offset_in_folio(bh->b_folio, bh->b_data) +
nilfs_cpfile_get_offset(cpfile, cno) *
NILFS_MDT(cpfile)->mi_entry_size;
}
/**
* nilfs_cpfile_cp_snapshot_list_offset - calculate the byte offset of a
* checkpoint snapshot list in the folio
* containing it
* @cpfile: checkpoint file inode
* @cno: checkpoint number
* @bh: buffer head of block containing checkpoint indexed by @cno
*
* Return: Byte offset in the folio of the checkpoint snapshot list specified
* by @cno.
*/
static size_t nilfs_cpfile_cp_snapshot_list_offset(const struct inode *cpfile,
__u64 cno,
struct buffer_head *bh)
{
return nilfs_cpfile_checkpoint_offset(cpfile, cno, bh) +
offsetof(struct nilfs_checkpoint, cp_snapshot_list);
}
/**
* nilfs_cpfile_ch_snapshot_list_offset - calculate the byte offset of the
* snapshot list in the header
*
* Return: Byte offset in the folio of the checkpoint snapshot list
*/
static size_t nilfs_cpfile_ch_snapshot_list_offset(void)
{
return offsetof(struct nilfs_cpfile_header, ch_snapshot_list);
}
static int nilfs_cpfile_get_header_block(struct inode *cpfile, static int nilfs_cpfile_get_header_block(struct inode *cpfile,
struct buffer_head **bhp) struct buffer_head **bhp)
{ {
@ -214,7 +249,7 @@ int nilfs_cpfile_read_checkpoint(struct inode *cpfile, __u64 cno,
{ {
struct buffer_head *cp_bh; struct buffer_head *cp_bh;
struct nilfs_checkpoint *cp; struct nilfs_checkpoint *cp;
void *kaddr; size_t offset;
int ret; int ret;
if (cno < 1 || cno > nilfs_mdt_cno(cpfile)) if (cno < 1 || cno > nilfs_mdt_cno(cpfile))
@ -228,8 +263,8 @@ int nilfs_cpfile_read_checkpoint(struct inode *cpfile, __u64 cno,
goto out_sem; goto out_sem;
} }
kaddr = kmap_local_page(cp_bh->b_page); offset = nilfs_cpfile_checkpoint_offset(cpfile, cno, cp_bh);
cp = nilfs_cpfile_block_get_checkpoint(cpfile, cno, cp_bh, kaddr); cp = kmap_local_folio(cp_bh->b_folio, offset);
if (nilfs_checkpoint_invalid(cp)) { if (nilfs_checkpoint_invalid(cp)) {
ret = -EINVAL; ret = -EINVAL;
goto put_cp; goto put_cp;
@ -254,7 +289,7 @@ int nilfs_cpfile_read_checkpoint(struct inode *cpfile, __u64 cno,
root->ifile = ifile; root->ifile = ifile;
put_cp: put_cp:
kunmap_local(kaddr); kunmap_local(cp);
brelse(cp_bh); brelse(cp_bh);
out_sem: out_sem:
up_read(&NILFS_MDT(cpfile)->mi_sem); up_read(&NILFS_MDT(cpfile)->mi_sem);
@ -282,7 +317,7 @@ int nilfs_cpfile_create_checkpoint(struct inode *cpfile, __u64 cno)
struct buffer_head *header_bh, *cp_bh; struct buffer_head *header_bh, *cp_bh;
struct nilfs_cpfile_header *header; struct nilfs_cpfile_header *header;
struct nilfs_checkpoint *cp; struct nilfs_checkpoint *cp;
void *kaddr; size_t offset;
int ret; int ret;
if (WARN_ON_ONCE(cno < 1)) if (WARN_ON_ONCE(cno < 1))
@ -297,24 +332,22 @@ int nilfs_cpfile_create_checkpoint(struct inode *cpfile, __u64 cno)
if (unlikely(ret < 0)) if (unlikely(ret < 0))
goto out_header; goto out_header;
kaddr = kmap_local_page(cp_bh->b_page); offset = nilfs_cpfile_checkpoint_offset(cpfile, cno, cp_bh);
cp = nilfs_cpfile_block_get_checkpoint(cpfile, cno, cp_bh, kaddr); cp = kmap_local_folio(cp_bh->b_folio, offset);
if (nilfs_checkpoint_invalid(cp)) { if (nilfs_checkpoint_invalid(cp)) {
/* a newly-created checkpoint */ /* a newly-created checkpoint */
nilfs_checkpoint_clear_invalid(cp); nilfs_checkpoint_clear_invalid(cp);
kunmap_local(cp);
if (!nilfs_cpfile_is_in_first(cpfile, cno)) if (!nilfs_cpfile_is_in_first(cpfile, cno))
nilfs_cpfile_block_add_valid_checkpoints(cpfile, cp_bh, nilfs_cpfile_block_add_valid_checkpoints(cpfile, cp_bh,
kaddr, 1); 1);
kunmap_local(kaddr);
kaddr = kmap_local_page(header_bh->b_page); header = kmap_local_folio(header_bh->b_folio, 0);
header = nilfs_cpfile_block_get_header(cpfile, header_bh,
kaddr);
le64_add_cpu(&header->ch_ncheckpoints, 1); le64_add_cpu(&header->ch_ncheckpoints, 1);
kunmap_local(kaddr); kunmap_local(header);
mark_buffer_dirty(header_bh); mark_buffer_dirty(header_bh);
} else { } else {
kunmap_local(kaddr); kunmap_local(cp);
} }
/* Force the buffer and the inode to become dirty */ /* Force the buffer and the inode to become dirty */
@ -353,7 +386,7 @@ int nilfs_cpfile_finalize_checkpoint(struct inode *cpfile, __u64 cno,
{ {
struct buffer_head *cp_bh; struct buffer_head *cp_bh;
struct nilfs_checkpoint *cp; struct nilfs_checkpoint *cp;
void *kaddr; size_t offset;
int ret; int ret;
if (WARN_ON_ONCE(cno < 1)) if (WARN_ON_ONCE(cno < 1))
@ -367,10 +400,10 @@ int nilfs_cpfile_finalize_checkpoint(struct inode *cpfile, __u64 cno,
goto out_sem; goto out_sem;
} }
kaddr = kmap_local_page(cp_bh->b_page); offset = nilfs_cpfile_checkpoint_offset(cpfile, cno, cp_bh);
cp = nilfs_cpfile_block_get_checkpoint(cpfile, cno, cp_bh, kaddr); cp = kmap_local_folio(cp_bh->b_folio, offset);
if (unlikely(nilfs_checkpoint_invalid(cp))) { if (unlikely(nilfs_checkpoint_invalid(cp))) {
kunmap_local(kaddr); kunmap_local(cp);
brelse(cp_bh); brelse(cp_bh);
goto error; goto error;
} }
@ -391,7 +424,7 @@ int nilfs_cpfile_finalize_checkpoint(struct inode *cpfile, __u64 cno,
nilfs_write_inode_common(root->ifile, &cp->cp_ifile_inode); nilfs_write_inode_common(root->ifile, &cp->cp_ifile_inode);
nilfs_bmap_write(NILFS_I(root->ifile)->i_bmap, &cp->cp_ifile_inode); nilfs_bmap_write(NILFS_I(root->ifile)->i_bmap, &cp->cp_ifile_inode);
kunmap_local(kaddr); kunmap_local(cp);
brelse(cp_bh); brelse(cp_bh);
out_sem: out_sem:
up_write(&NILFS_MDT(cpfile)->mi_sem); up_write(&NILFS_MDT(cpfile)->mi_sem);
@ -432,6 +465,7 @@ int nilfs_cpfile_delete_checkpoints(struct inode *cpfile,
struct nilfs_checkpoint *cp; struct nilfs_checkpoint *cp;
size_t cpsz = NILFS_MDT(cpfile)->mi_entry_size; size_t cpsz = NILFS_MDT(cpfile)->mi_entry_size;
__u64 cno; __u64 cno;
size_t offset;
void *kaddr; void *kaddr;
unsigned long tnicps; unsigned long tnicps;
int ret, ncps, nicps, nss, count, i; int ret, ncps, nicps, nss, count, i;
@ -462,9 +496,8 @@ int nilfs_cpfile_delete_checkpoints(struct inode *cpfile,
continue; continue;
} }
kaddr = kmap_local_page(cp_bh->b_page); offset = nilfs_cpfile_checkpoint_offset(cpfile, cno, cp_bh);
cp = nilfs_cpfile_block_get_checkpoint( cp = kaddr = kmap_local_folio(cp_bh->b_folio, offset);
cpfile, cno, cp_bh, kaddr);
nicps = 0; nicps = 0;
for (i = 0; i < ncps; i++, cp = (void *)cp + cpsz) { for (i = 0; i < ncps; i++, cp = (void *)cp + cpsz) {
if (nilfs_checkpoint_snapshot(cp)) { if (nilfs_checkpoint_snapshot(cp)) {
@ -474,43 +507,42 @@ int nilfs_cpfile_delete_checkpoints(struct inode *cpfile,
nicps++; nicps++;
} }
} }
if (nicps > 0) { kunmap_local(kaddr);
if (nicps <= 0) {
brelse(cp_bh);
continue;
}
tnicps += nicps; tnicps += nicps;
mark_buffer_dirty(cp_bh); mark_buffer_dirty(cp_bh);
nilfs_mdt_mark_dirty(cpfile); nilfs_mdt_mark_dirty(cpfile);
if (!nilfs_cpfile_is_in_first(cpfile, cno)) { if (nilfs_cpfile_is_in_first(cpfile, cno)) {
count =
nilfs_cpfile_block_sub_valid_checkpoints(
cpfile, cp_bh, kaddr, nicps);
if (count == 0) {
/* make hole */
kunmap_local(kaddr);
brelse(cp_bh); brelse(cp_bh);
ret =
nilfs_cpfile_delete_checkpoint_block(
cpfile, cno);
if (ret == 0)
continue; continue;
}
count = nilfs_cpfile_block_sub_valid_checkpoints(cpfile, cp_bh,
nicps);
brelse(cp_bh);
if (count)
continue;
/* Delete the block if there are no more valid checkpoints */
ret = nilfs_cpfile_delete_checkpoint_block(cpfile, cno);
if (unlikely(ret)) {
nilfs_err(cpfile->i_sb, nilfs_err(cpfile->i_sb,
"error %d deleting checkpoint block", "error %d deleting checkpoint block", ret);
ret);
break; break;
} }
} }
}
kunmap_local(kaddr);
brelse(cp_bh);
}
if (tnicps > 0) { if (tnicps > 0) {
kaddr = kmap_local_page(header_bh->b_page); header = kmap_local_folio(header_bh->b_folio, 0);
header = nilfs_cpfile_block_get_header(cpfile, header_bh,
kaddr);
le64_add_cpu(&header->ch_ncheckpoints, -(u64)tnicps); le64_add_cpu(&header->ch_ncheckpoints, -(u64)tnicps);
mark_buffer_dirty(header_bh); mark_buffer_dirty(header_bh);
nilfs_mdt_mark_dirty(cpfile); nilfs_mdt_mark_dirty(cpfile);
kunmap_local(kaddr); kunmap_local(header);
} }
brelse(header_bh); brelse(header_bh);
@ -544,6 +576,7 @@ static ssize_t nilfs_cpfile_do_get_cpinfo(struct inode *cpfile, __u64 *cnop,
struct buffer_head *bh; struct buffer_head *bh;
size_t cpsz = NILFS_MDT(cpfile)->mi_entry_size; size_t cpsz = NILFS_MDT(cpfile)->mi_entry_size;
__u64 cur_cno = nilfs_mdt_cno(cpfile), cno = *cnop; __u64 cur_cno = nilfs_mdt_cno(cpfile), cno = *cnop;
size_t offset;
void *kaddr; void *kaddr;
int n, ret; int n, ret;
int ncps, i; int ncps, i;
@ -562,8 +595,8 @@ static ssize_t nilfs_cpfile_do_get_cpinfo(struct inode *cpfile, __u64 *cnop,
} }
ncps = nilfs_cpfile_checkpoints_in_block(cpfile, cno, cur_cno); ncps = nilfs_cpfile_checkpoints_in_block(cpfile, cno, cur_cno);
kaddr = kmap_local_page(bh->b_page); offset = nilfs_cpfile_checkpoint_offset(cpfile, cno, bh);
cp = nilfs_cpfile_block_get_checkpoint(cpfile, cno, bh, kaddr); cp = kaddr = kmap_local_folio(bh->b_folio, offset);
for (i = 0; i < ncps && n < nci; i++, cp = (void *)cp + cpsz) { for (i = 0; i < ncps && n < nci; i++, cp = (void *)cp + cpsz) {
if (!nilfs_checkpoint_invalid(cp)) { if (!nilfs_checkpoint_invalid(cp)) {
nilfs_cpfile_checkpoint_to_cpinfo(cpfile, cp, nilfs_cpfile_checkpoint_to_cpinfo(cpfile, cp,
@ -597,7 +630,7 @@ static ssize_t nilfs_cpfile_do_get_ssinfo(struct inode *cpfile, __u64 *cnop,
struct nilfs_cpinfo *ci = buf; struct nilfs_cpinfo *ci = buf;
__u64 curr = *cnop, next; __u64 curr = *cnop, next;
unsigned long curr_blkoff, next_blkoff; unsigned long curr_blkoff, next_blkoff;
void *kaddr; size_t offset;
int n = 0, ret; int n = 0, ret;
down_read(&NILFS_MDT(cpfile)->mi_sem); down_read(&NILFS_MDT(cpfile)->mi_sem);
@ -606,10 +639,9 @@ static ssize_t nilfs_cpfile_do_get_ssinfo(struct inode *cpfile, __u64 *cnop,
ret = nilfs_cpfile_get_header_block(cpfile, &bh); ret = nilfs_cpfile_get_header_block(cpfile, &bh);
if (ret < 0) if (ret < 0)
goto out; goto out;
kaddr = kmap_local_page(bh->b_page); header = kmap_local_folio(bh->b_folio, 0);
header = nilfs_cpfile_block_get_header(cpfile, bh, kaddr);
curr = le64_to_cpu(header->ch_snapshot_list.ssl_next); curr = le64_to_cpu(header->ch_snapshot_list.ssl_next);
kunmap_local(kaddr); kunmap_local(header);
brelse(bh); brelse(bh);
if (curr == 0) { if (curr == 0) {
ret = 0; ret = 0;
@ -627,9 +659,9 @@ static ssize_t nilfs_cpfile_do_get_ssinfo(struct inode *cpfile, __u64 *cnop,
ret = 0; /* No snapshots (started from a hole block) */ ret = 0; /* No snapshots (started from a hole block) */
goto out; goto out;
} }
kaddr = kmap_local_page(bh->b_page); offset = nilfs_cpfile_checkpoint_offset(cpfile, curr, bh);
cp = kmap_local_folio(bh->b_folio, offset);
while (n < nci) { while (n < nci) {
cp = nilfs_cpfile_block_get_checkpoint(cpfile, curr, bh, kaddr);
curr = ~(__u64)0; /* Terminator */ curr = ~(__u64)0; /* Terminator */
if (unlikely(nilfs_checkpoint_invalid(cp) || if (unlikely(nilfs_checkpoint_invalid(cp) ||
!nilfs_checkpoint_snapshot(cp))) !nilfs_checkpoint_snapshot(cp)))
@ -641,9 +673,9 @@ static ssize_t nilfs_cpfile_do_get_ssinfo(struct inode *cpfile, __u64 *cnop,
if (next == 0) if (next == 0)
break; /* reach end of the snapshot list */ break; /* reach end of the snapshot list */
kunmap_local(cp);
next_blkoff = nilfs_cpfile_get_blkoff(cpfile, next); next_blkoff = nilfs_cpfile_get_blkoff(cpfile, next);
if (curr_blkoff != next_blkoff) { if (curr_blkoff != next_blkoff) {
kunmap_local(kaddr);
brelse(bh); brelse(bh);
ret = nilfs_cpfile_get_checkpoint_block(cpfile, next, ret = nilfs_cpfile_get_checkpoint_block(cpfile, next,
0, &bh); 0, &bh);
@ -651,12 +683,13 @@ static ssize_t nilfs_cpfile_do_get_ssinfo(struct inode *cpfile, __u64 *cnop,
WARN_ON(ret == -ENOENT); WARN_ON(ret == -ENOENT);
goto out; goto out;
} }
kaddr = kmap_local_page(bh->b_page);
} }
offset = nilfs_cpfile_checkpoint_offset(cpfile, next, bh);
cp = kmap_local_folio(bh->b_folio, offset);
curr = next; curr = next;
curr_blkoff = next_blkoff; curr_blkoff = next_blkoff;
} }
kunmap_local(kaddr); kunmap_local(cp);
brelse(bh); brelse(bh);
*cnop = curr; *cnop = curr;
ret = n; ret = n;
@ -733,26 +766,6 @@ int nilfs_cpfile_delete_checkpoint(struct inode *cpfile, __u64 cno)
return nilfs_cpfile_delete_checkpoints(cpfile, cno, cno + 1); return nilfs_cpfile_delete_checkpoints(cpfile, cno, cno + 1);
} }
static struct nilfs_snapshot_list *
nilfs_cpfile_block_get_snapshot_list(const struct inode *cpfile,
__u64 cno,
struct buffer_head *bh,
void *kaddr)
{
struct nilfs_cpfile_header *header;
struct nilfs_checkpoint *cp;
struct nilfs_snapshot_list *list;
if (cno != 0) {
cp = nilfs_cpfile_block_get_checkpoint(cpfile, cno, bh, kaddr);
list = &cp->cp_snapshot_list;
} else {
header = nilfs_cpfile_block_get_header(cpfile, bh, kaddr);
list = &header->ch_snapshot_list;
}
return list;
}
static int nilfs_cpfile_set_snapshot(struct inode *cpfile, __u64 cno) static int nilfs_cpfile_set_snapshot(struct inode *cpfile, __u64 cno)
{ {
struct buffer_head *header_bh, *curr_bh, *prev_bh, *cp_bh; struct buffer_head *header_bh, *curr_bh, *prev_bh, *cp_bh;
@ -761,94 +774,103 @@ static int nilfs_cpfile_set_snapshot(struct inode *cpfile, __u64 cno)
struct nilfs_snapshot_list *list; struct nilfs_snapshot_list *list;
__u64 curr, prev; __u64 curr, prev;
unsigned long curr_blkoff, prev_blkoff; unsigned long curr_blkoff, prev_blkoff;
void *kaddr; size_t offset, curr_list_offset, prev_list_offset;
int ret; int ret;
if (cno == 0) if (cno == 0)
return -ENOENT; /* checkpoint number 0 is invalid */ return -ENOENT; /* checkpoint number 0 is invalid */
down_write(&NILFS_MDT(cpfile)->mi_sem); down_write(&NILFS_MDT(cpfile)->mi_sem);
ret = nilfs_cpfile_get_header_block(cpfile, &header_bh);
if (unlikely(ret < 0))
goto out_sem;
ret = nilfs_cpfile_get_checkpoint_block(cpfile, cno, 0, &cp_bh); ret = nilfs_cpfile_get_checkpoint_block(cpfile, cno, 0, &cp_bh);
if (ret < 0) if (ret < 0)
goto out_sem; goto out_header;
kaddr = kmap_local_page(cp_bh->b_page);
cp = nilfs_cpfile_block_get_checkpoint(cpfile, cno, cp_bh, kaddr); offset = nilfs_cpfile_checkpoint_offset(cpfile, cno, cp_bh);
cp = kmap_local_folio(cp_bh->b_folio, offset);
if (nilfs_checkpoint_invalid(cp)) { if (nilfs_checkpoint_invalid(cp)) {
ret = -ENOENT; ret = -ENOENT;
kunmap_local(kaddr); kunmap_local(cp);
goto out_cp; goto out_cp;
} }
if (nilfs_checkpoint_snapshot(cp)) { if (nilfs_checkpoint_snapshot(cp)) {
ret = 0; ret = 0;
kunmap_local(kaddr); kunmap_local(cp);
goto out_cp; goto out_cp;
} }
kunmap_local(kaddr); kunmap_local(cp);
ret = nilfs_cpfile_get_header_block(cpfile, &header_bh); /*
if (ret < 0) * Find the last snapshot before the checkpoint being changed to
goto out_cp; * snapshot mode by going backwards through the snapshot list.
kaddr = kmap_local_page(header_bh->b_page); * Set "prev" to its checkpoint number, or 0 if not found.
header = nilfs_cpfile_block_get_header(cpfile, header_bh, kaddr); */
header = kmap_local_folio(header_bh->b_folio, 0);
list = &header->ch_snapshot_list; list = &header->ch_snapshot_list;
curr_bh = header_bh; curr_bh = header_bh;
get_bh(curr_bh); get_bh(curr_bh);
curr = 0; curr = 0;
curr_blkoff = 0; curr_blkoff = 0;
curr_list_offset = nilfs_cpfile_ch_snapshot_list_offset();
prev = le64_to_cpu(list->ssl_prev); prev = le64_to_cpu(list->ssl_prev);
while (prev > cno) { while (prev > cno) {
prev_blkoff = nilfs_cpfile_get_blkoff(cpfile, prev); prev_blkoff = nilfs_cpfile_get_blkoff(cpfile, prev);
curr = prev; curr = prev;
kunmap_local(list);
if (curr_blkoff != prev_blkoff) { if (curr_blkoff != prev_blkoff) {
kunmap_local(kaddr);
brelse(curr_bh); brelse(curr_bh);
ret = nilfs_cpfile_get_checkpoint_block(cpfile, curr, ret = nilfs_cpfile_get_checkpoint_block(cpfile, curr,
0, &curr_bh); 0, &curr_bh);
if (ret < 0) if (unlikely(ret < 0))
goto out_header; goto out_cp;
kaddr = kmap_local_page(curr_bh->b_page);
} }
curr_list_offset = nilfs_cpfile_cp_snapshot_list_offset(
cpfile, curr, curr_bh);
list = kmap_local_folio(curr_bh->b_folio, curr_list_offset);
curr_blkoff = prev_blkoff; curr_blkoff = prev_blkoff;
cp = nilfs_cpfile_block_get_checkpoint(
cpfile, curr, curr_bh, kaddr);
list = &cp->cp_snapshot_list;
prev = le64_to_cpu(list->ssl_prev); prev = le64_to_cpu(list->ssl_prev);
} }
kunmap_local(kaddr); kunmap_local(list);
if (prev != 0) { if (prev != 0) {
ret = nilfs_cpfile_get_checkpoint_block(cpfile, prev, 0, ret = nilfs_cpfile_get_checkpoint_block(cpfile, prev, 0,
&prev_bh); &prev_bh);
if (ret < 0) if (ret < 0)
goto out_curr; goto out_curr;
prev_list_offset = nilfs_cpfile_cp_snapshot_list_offset(
cpfile, prev, prev_bh);
} else { } else {
prev_bh = header_bh; prev_bh = header_bh;
get_bh(prev_bh); get_bh(prev_bh);
prev_list_offset = nilfs_cpfile_ch_snapshot_list_offset();
} }
kaddr = kmap_local_page(curr_bh->b_page); /* Update the list entry for the next snapshot */
list = nilfs_cpfile_block_get_snapshot_list( list = kmap_local_folio(curr_bh->b_folio, curr_list_offset);
cpfile, curr, curr_bh, kaddr);
list->ssl_prev = cpu_to_le64(cno); list->ssl_prev = cpu_to_le64(cno);
kunmap_local(kaddr); kunmap_local(list);
kaddr = kmap_local_page(cp_bh->b_page); /* Update the checkpoint being changed to a snapshot */
cp = nilfs_cpfile_block_get_checkpoint(cpfile, cno, cp_bh, kaddr); offset = nilfs_cpfile_checkpoint_offset(cpfile, cno, cp_bh);
cp = kmap_local_folio(cp_bh->b_folio, offset);
cp->cp_snapshot_list.ssl_next = cpu_to_le64(curr); cp->cp_snapshot_list.ssl_next = cpu_to_le64(curr);
cp->cp_snapshot_list.ssl_prev = cpu_to_le64(prev); cp->cp_snapshot_list.ssl_prev = cpu_to_le64(prev);
nilfs_checkpoint_set_snapshot(cp); nilfs_checkpoint_set_snapshot(cp);
kunmap_local(kaddr); kunmap_local(cp);
kaddr = kmap_local_page(prev_bh->b_page); /* Update the list entry for the previous snapshot */
list = nilfs_cpfile_block_get_snapshot_list( list = kmap_local_folio(prev_bh->b_folio, prev_list_offset);
cpfile, prev, prev_bh, kaddr);
list->ssl_next = cpu_to_le64(cno); list->ssl_next = cpu_to_le64(cno);
kunmap_local(kaddr); kunmap_local(list);
kaddr = kmap_local_page(header_bh->b_page); /* Update the statistics in the header */
header = nilfs_cpfile_block_get_header(cpfile, header_bh, kaddr); header = kmap_local_folio(header_bh->b_folio, 0);
le64_add_cpu(&header->ch_nsnapshots, 1); le64_add_cpu(&header->ch_nsnapshots, 1);
kunmap_local(kaddr); kunmap_local(header);
mark_buffer_dirty(prev_bh); mark_buffer_dirty(prev_bh);
mark_buffer_dirty(curr_bh); mark_buffer_dirty(curr_bh);
@ -861,12 +883,12 @@ static int nilfs_cpfile_set_snapshot(struct inode *cpfile, __u64 cno)
out_curr: out_curr:
brelse(curr_bh); brelse(curr_bh);
out_header:
brelse(header_bh);
out_cp: out_cp:
brelse(cp_bh); brelse(cp_bh);
out_header:
brelse(header_bh);
out_sem: out_sem:
up_write(&NILFS_MDT(cpfile)->mi_sem); up_write(&NILFS_MDT(cpfile)->mi_sem);
return ret; return ret;
@ -879,79 +901,87 @@ static int nilfs_cpfile_clear_snapshot(struct inode *cpfile, __u64 cno)
struct nilfs_checkpoint *cp; struct nilfs_checkpoint *cp;
struct nilfs_snapshot_list *list; struct nilfs_snapshot_list *list;
__u64 next, prev; __u64 next, prev;
void *kaddr; size_t offset, next_list_offset, prev_list_offset;
int ret; int ret;
if (cno == 0) if (cno == 0)
return -ENOENT; /* checkpoint number 0 is invalid */ return -ENOENT; /* checkpoint number 0 is invalid */
down_write(&NILFS_MDT(cpfile)->mi_sem); down_write(&NILFS_MDT(cpfile)->mi_sem);
ret = nilfs_cpfile_get_header_block(cpfile, &header_bh);
if (unlikely(ret < 0))
goto out_sem;
ret = nilfs_cpfile_get_checkpoint_block(cpfile, cno, 0, &cp_bh); ret = nilfs_cpfile_get_checkpoint_block(cpfile, cno, 0, &cp_bh);
if (ret < 0) if (ret < 0)
goto out_sem; goto out_header;
kaddr = kmap_local_page(cp_bh->b_page);
cp = nilfs_cpfile_block_get_checkpoint(cpfile, cno, cp_bh, kaddr); offset = nilfs_cpfile_checkpoint_offset(cpfile, cno, cp_bh);
cp = kmap_local_folio(cp_bh->b_folio, offset);
if (nilfs_checkpoint_invalid(cp)) { if (nilfs_checkpoint_invalid(cp)) {
ret = -ENOENT; ret = -ENOENT;
kunmap_local(kaddr); kunmap_local(cp);
goto out_cp; goto out_cp;
} }
if (!nilfs_checkpoint_snapshot(cp)) { if (!nilfs_checkpoint_snapshot(cp)) {
ret = 0; ret = 0;
kunmap_local(kaddr); kunmap_local(cp);
goto out_cp; goto out_cp;
} }
list = &cp->cp_snapshot_list; list = &cp->cp_snapshot_list;
next = le64_to_cpu(list->ssl_next); next = le64_to_cpu(list->ssl_next);
prev = le64_to_cpu(list->ssl_prev); prev = le64_to_cpu(list->ssl_prev);
kunmap_local(kaddr); kunmap_local(cp);
ret = nilfs_cpfile_get_header_block(cpfile, &header_bh);
if (ret < 0)
goto out_cp;
if (next != 0) { if (next != 0) {
ret = nilfs_cpfile_get_checkpoint_block(cpfile, next, 0, ret = nilfs_cpfile_get_checkpoint_block(cpfile, next, 0,
&next_bh); &next_bh);
if (ret < 0) if (ret < 0)
goto out_header; goto out_cp;
next_list_offset = nilfs_cpfile_cp_snapshot_list_offset(
cpfile, next, next_bh);
} else { } else {
next_bh = header_bh; next_bh = header_bh;
get_bh(next_bh); get_bh(next_bh);
next_list_offset = nilfs_cpfile_ch_snapshot_list_offset();
} }
if (prev != 0) { if (prev != 0) {
ret = nilfs_cpfile_get_checkpoint_block(cpfile, prev, 0, ret = nilfs_cpfile_get_checkpoint_block(cpfile, prev, 0,
&prev_bh); &prev_bh);
if (ret < 0) if (ret < 0)
goto out_next; goto out_next;
prev_list_offset = nilfs_cpfile_cp_snapshot_list_offset(
cpfile, prev, prev_bh);
} else { } else {
prev_bh = header_bh; prev_bh = header_bh;
get_bh(prev_bh); get_bh(prev_bh);
prev_list_offset = nilfs_cpfile_ch_snapshot_list_offset();
} }
kaddr = kmap_local_page(next_bh->b_page); /* Update the list entry for the next snapshot */
list = nilfs_cpfile_block_get_snapshot_list( list = kmap_local_folio(next_bh->b_folio, next_list_offset);
cpfile, next, next_bh, kaddr);
list->ssl_prev = cpu_to_le64(prev); list->ssl_prev = cpu_to_le64(prev);
kunmap_local(kaddr); kunmap_local(list);
kaddr = kmap_local_page(prev_bh->b_page); /* Update the list entry for the previous snapshot */
list = nilfs_cpfile_block_get_snapshot_list( list = kmap_local_folio(prev_bh->b_folio, prev_list_offset);
cpfile, prev, prev_bh, kaddr);
list->ssl_next = cpu_to_le64(next); list->ssl_next = cpu_to_le64(next);
kunmap_local(kaddr); kunmap_local(list);
kaddr = kmap_local_page(cp_bh->b_page); /* Update the snapshot being changed back to a plain checkpoint */
cp = nilfs_cpfile_block_get_checkpoint(cpfile, cno, cp_bh, kaddr); cp = kmap_local_folio(cp_bh->b_folio, offset);
cp->cp_snapshot_list.ssl_next = cpu_to_le64(0); cp->cp_snapshot_list.ssl_next = cpu_to_le64(0);
cp->cp_snapshot_list.ssl_prev = cpu_to_le64(0); cp->cp_snapshot_list.ssl_prev = cpu_to_le64(0);
nilfs_checkpoint_clear_snapshot(cp); nilfs_checkpoint_clear_snapshot(cp);
kunmap_local(kaddr); kunmap_local(cp);
kaddr = kmap_local_page(header_bh->b_page); /* Update the statistics in the header */
header = nilfs_cpfile_block_get_header(cpfile, header_bh, kaddr); header = kmap_local_folio(header_bh->b_folio, 0);
le64_add_cpu(&header->ch_nsnapshots, -1); le64_add_cpu(&header->ch_nsnapshots, -1);
kunmap_local(kaddr); kunmap_local(header);
mark_buffer_dirty(next_bh); mark_buffer_dirty(next_bh);
mark_buffer_dirty(prev_bh); mark_buffer_dirty(prev_bh);
@ -964,12 +994,12 @@ static int nilfs_cpfile_clear_snapshot(struct inode *cpfile, __u64 cno)
out_next: out_next:
brelse(next_bh); brelse(next_bh);
out_header:
brelse(header_bh);
out_cp: out_cp:
brelse(cp_bh); brelse(cp_bh);
out_header:
brelse(header_bh);
out_sem: out_sem:
up_write(&NILFS_MDT(cpfile)->mi_sem); up_write(&NILFS_MDT(cpfile)->mi_sem);
return ret; return ret;
@ -990,7 +1020,7 @@ int nilfs_cpfile_is_snapshot(struct inode *cpfile, __u64 cno)
{ {
struct buffer_head *bh; struct buffer_head *bh;
struct nilfs_checkpoint *cp; struct nilfs_checkpoint *cp;
void *kaddr; size_t offset;
int ret; int ret;
/* /*
@ -1004,13 +1034,14 @@ int nilfs_cpfile_is_snapshot(struct inode *cpfile, __u64 cno)
ret = nilfs_cpfile_get_checkpoint_block(cpfile, cno, 0, &bh); ret = nilfs_cpfile_get_checkpoint_block(cpfile, cno, 0, &bh);
if (ret < 0) if (ret < 0)
goto out; goto out;
kaddr = kmap_local_page(bh->b_page);
cp = nilfs_cpfile_block_get_checkpoint(cpfile, cno, bh, kaddr); offset = nilfs_cpfile_checkpoint_offset(cpfile, cno, bh);
cp = kmap_local_folio(bh->b_folio, offset);
if (nilfs_checkpoint_invalid(cp)) if (nilfs_checkpoint_invalid(cp))
ret = -ENOENT; ret = -ENOENT;
else else
ret = nilfs_checkpoint_snapshot(cp); ret = nilfs_checkpoint_snapshot(cp);
kunmap_local(kaddr); kunmap_local(cp);
brelse(bh); brelse(bh);
out: out:
@ -1079,7 +1110,6 @@ int nilfs_cpfile_get_stat(struct inode *cpfile, struct nilfs_cpstat *cpstat)
{ {
struct buffer_head *bh; struct buffer_head *bh;
struct nilfs_cpfile_header *header; struct nilfs_cpfile_header *header;
void *kaddr;
int ret; int ret;
down_read(&NILFS_MDT(cpfile)->mi_sem); down_read(&NILFS_MDT(cpfile)->mi_sem);
@ -1087,12 +1117,11 @@ int nilfs_cpfile_get_stat(struct inode *cpfile, struct nilfs_cpstat *cpstat)
ret = nilfs_cpfile_get_header_block(cpfile, &bh); ret = nilfs_cpfile_get_header_block(cpfile, &bh);
if (ret < 0) if (ret < 0)
goto out_sem; goto out_sem;
kaddr = kmap_local_page(bh->b_page); header = kmap_local_folio(bh->b_folio, 0);
header = nilfs_cpfile_block_get_header(cpfile, bh, kaddr);
cpstat->cs_cno = nilfs_mdt_cno(cpfile); cpstat->cs_cno = nilfs_mdt_cno(cpfile);
cpstat->cs_ncps = le64_to_cpu(header->ch_ncheckpoints); cpstat->cs_ncps = le64_to_cpu(header->ch_ncheckpoints);
cpstat->cs_nsss = le64_to_cpu(header->ch_nsnapshots); cpstat->cs_nsss = le64_to_cpu(header->ch_nsnapshots);
kunmap_local(kaddr); kunmap_local(header);
brelse(bh); brelse(bh);
out_sem: out_sem:

View File

@ -89,15 +89,15 @@ int nilfs_dat_prepare_alloc(struct inode *dat, struct nilfs_palloc_req *req)
void nilfs_dat_commit_alloc(struct inode *dat, struct nilfs_palloc_req *req) void nilfs_dat_commit_alloc(struct inode *dat, struct nilfs_palloc_req *req)
{ {
struct nilfs_dat_entry *entry; struct nilfs_dat_entry *entry;
void *kaddr; size_t offset;
kaddr = kmap_local_page(req->pr_entry_bh->b_page); offset = nilfs_palloc_entry_offset(dat, req->pr_entry_nr,
entry = nilfs_palloc_block_get_entry(dat, req->pr_entry_nr, req->pr_entry_bh);
req->pr_entry_bh, kaddr); entry = kmap_local_folio(req->pr_entry_bh->b_folio, offset);
entry->de_start = cpu_to_le64(NILFS_CNO_MIN); entry->de_start = cpu_to_le64(NILFS_CNO_MIN);
entry->de_end = cpu_to_le64(NILFS_CNO_MAX); entry->de_end = cpu_to_le64(NILFS_CNO_MAX);
entry->de_blocknr = cpu_to_le64(0); entry->de_blocknr = cpu_to_le64(0);
kunmap_local(kaddr); kunmap_local(entry);
nilfs_palloc_commit_alloc_entry(dat, req); nilfs_palloc_commit_alloc_entry(dat, req);
nilfs_dat_commit_entry(dat, req); nilfs_dat_commit_entry(dat, req);
@ -113,15 +113,15 @@ static void nilfs_dat_commit_free(struct inode *dat,
struct nilfs_palloc_req *req) struct nilfs_palloc_req *req)
{ {
struct nilfs_dat_entry *entry; struct nilfs_dat_entry *entry;
void *kaddr; size_t offset;
kaddr = kmap_local_page(req->pr_entry_bh->b_page); offset = nilfs_palloc_entry_offset(dat, req->pr_entry_nr,
entry = nilfs_palloc_block_get_entry(dat, req->pr_entry_nr, req->pr_entry_bh);
req->pr_entry_bh, kaddr); entry = kmap_local_folio(req->pr_entry_bh->b_folio, offset);
entry->de_start = cpu_to_le64(NILFS_CNO_MIN); entry->de_start = cpu_to_le64(NILFS_CNO_MIN);
entry->de_end = cpu_to_le64(NILFS_CNO_MIN); entry->de_end = cpu_to_le64(NILFS_CNO_MIN);
entry->de_blocknr = cpu_to_le64(0); entry->de_blocknr = cpu_to_le64(0);
kunmap_local(kaddr); kunmap_local(entry);
nilfs_dat_commit_entry(dat, req); nilfs_dat_commit_entry(dat, req);
@ -143,14 +143,14 @@ void nilfs_dat_commit_start(struct inode *dat, struct nilfs_palloc_req *req,
sector_t blocknr) sector_t blocknr)
{ {
struct nilfs_dat_entry *entry; struct nilfs_dat_entry *entry;
void *kaddr; size_t offset;
kaddr = kmap_local_page(req->pr_entry_bh->b_page); offset = nilfs_palloc_entry_offset(dat, req->pr_entry_nr,
entry = nilfs_palloc_block_get_entry(dat, req->pr_entry_nr, req->pr_entry_bh);
req->pr_entry_bh, kaddr); entry = kmap_local_folio(req->pr_entry_bh->b_folio, offset);
entry->de_start = cpu_to_le64(nilfs_mdt_cno(dat)); entry->de_start = cpu_to_le64(nilfs_mdt_cno(dat));
entry->de_blocknr = cpu_to_le64(blocknr); entry->de_blocknr = cpu_to_le64(blocknr);
kunmap_local(kaddr); kunmap_local(entry);
nilfs_dat_commit_entry(dat, req); nilfs_dat_commit_entry(dat, req);
} }
@ -160,19 +160,19 @@ int nilfs_dat_prepare_end(struct inode *dat, struct nilfs_palloc_req *req)
struct nilfs_dat_entry *entry; struct nilfs_dat_entry *entry;
__u64 start; __u64 start;
sector_t blocknr; sector_t blocknr;
void *kaddr; size_t offset;
int ret; int ret;
ret = nilfs_dat_prepare_entry(dat, req, 0); ret = nilfs_dat_prepare_entry(dat, req, 0);
if (ret < 0) if (ret < 0)
return ret; return ret;
kaddr = kmap_local_page(req->pr_entry_bh->b_page); offset = nilfs_palloc_entry_offset(dat, req->pr_entry_nr,
entry = nilfs_palloc_block_get_entry(dat, req->pr_entry_nr, req->pr_entry_bh);
req->pr_entry_bh, kaddr); entry = kmap_local_folio(req->pr_entry_bh->b_folio, offset);
start = le64_to_cpu(entry->de_start); start = le64_to_cpu(entry->de_start);
blocknr = le64_to_cpu(entry->de_blocknr); blocknr = le64_to_cpu(entry->de_blocknr);
kunmap_local(kaddr); kunmap_local(entry);
if (blocknr == 0) { if (blocknr == 0) {
ret = nilfs_palloc_prepare_free_entry(dat, req); ret = nilfs_palloc_prepare_free_entry(dat, req);
@ -200,11 +200,11 @@ void nilfs_dat_commit_end(struct inode *dat, struct nilfs_palloc_req *req,
struct nilfs_dat_entry *entry; struct nilfs_dat_entry *entry;
__u64 start, end; __u64 start, end;
sector_t blocknr; sector_t blocknr;
void *kaddr; size_t offset;
kaddr = kmap_local_page(req->pr_entry_bh->b_page); offset = nilfs_palloc_entry_offset(dat, req->pr_entry_nr,
entry = nilfs_palloc_block_get_entry(dat, req->pr_entry_nr, req->pr_entry_bh);
req->pr_entry_bh, kaddr); entry = kmap_local_folio(req->pr_entry_bh->b_folio, offset);
end = start = le64_to_cpu(entry->de_start); end = start = le64_to_cpu(entry->de_start);
if (!dead) { if (!dead) {
end = nilfs_mdt_cno(dat); end = nilfs_mdt_cno(dat);
@ -212,7 +212,7 @@ void nilfs_dat_commit_end(struct inode *dat, struct nilfs_palloc_req *req,
} }
entry->de_end = cpu_to_le64(end); entry->de_end = cpu_to_le64(end);
blocknr = le64_to_cpu(entry->de_blocknr); blocknr = le64_to_cpu(entry->de_blocknr);
kunmap_local(kaddr); kunmap_local(entry);
if (blocknr == 0) if (blocknr == 0)
nilfs_dat_commit_free(dat, req); nilfs_dat_commit_free(dat, req);
@ -225,14 +225,14 @@ void nilfs_dat_abort_end(struct inode *dat, struct nilfs_palloc_req *req)
struct nilfs_dat_entry *entry; struct nilfs_dat_entry *entry;
__u64 start; __u64 start;
sector_t blocknr; sector_t blocknr;
void *kaddr; size_t offset;
kaddr = kmap_local_page(req->pr_entry_bh->b_page); offset = nilfs_palloc_entry_offset(dat, req->pr_entry_nr,
entry = nilfs_palloc_block_get_entry(dat, req->pr_entry_nr, req->pr_entry_bh);
req->pr_entry_bh, kaddr); entry = kmap_local_folio(req->pr_entry_bh->b_folio, offset);
start = le64_to_cpu(entry->de_start); start = le64_to_cpu(entry->de_start);
blocknr = le64_to_cpu(entry->de_blocknr); blocknr = le64_to_cpu(entry->de_blocknr);
kunmap_local(kaddr); kunmap_local(entry);
if (start == nilfs_mdt_cno(dat) && blocknr == 0) if (start == nilfs_mdt_cno(dat) && blocknr == 0)
nilfs_palloc_abort_free_entry(dat, req); nilfs_palloc_abort_free_entry(dat, req);
@ -336,7 +336,7 @@ int nilfs_dat_move(struct inode *dat, __u64 vblocknr, sector_t blocknr)
{ {
struct buffer_head *entry_bh; struct buffer_head *entry_bh;
struct nilfs_dat_entry *entry; struct nilfs_dat_entry *entry;
void *kaddr; size_t offset;
int ret; int ret;
ret = nilfs_palloc_get_entry_block(dat, vblocknr, 0, &entry_bh); ret = nilfs_palloc_get_entry_block(dat, vblocknr, 0, &entry_bh);
@ -359,21 +359,21 @@ int nilfs_dat_move(struct inode *dat, __u64 vblocknr, sector_t blocknr)
} }
} }
kaddr = kmap_local_page(entry_bh->b_page); offset = nilfs_palloc_entry_offset(dat, vblocknr, entry_bh);
entry = nilfs_palloc_block_get_entry(dat, vblocknr, entry_bh, kaddr); entry = kmap_local_folio(entry_bh->b_folio, offset);
if (unlikely(entry->de_blocknr == cpu_to_le64(0))) { if (unlikely(entry->de_blocknr == cpu_to_le64(0))) {
nilfs_crit(dat->i_sb, nilfs_crit(dat->i_sb,
"%s: invalid vblocknr = %llu, [%llu, %llu)", "%s: invalid vblocknr = %llu, [%llu, %llu)",
__func__, (unsigned long long)vblocknr, __func__, (unsigned long long)vblocknr,
(unsigned long long)le64_to_cpu(entry->de_start), (unsigned long long)le64_to_cpu(entry->de_start),
(unsigned long long)le64_to_cpu(entry->de_end)); (unsigned long long)le64_to_cpu(entry->de_end));
kunmap_local(kaddr); kunmap_local(entry);
brelse(entry_bh); brelse(entry_bh);
return -EINVAL; return -EINVAL;
} }
WARN_ON(blocknr == 0); WARN_ON(blocknr == 0);
entry->de_blocknr = cpu_to_le64(blocknr); entry->de_blocknr = cpu_to_le64(blocknr);
kunmap_local(kaddr); kunmap_local(entry);
mark_buffer_dirty(entry_bh); mark_buffer_dirty(entry_bh);
nilfs_mdt_mark_dirty(dat); nilfs_mdt_mark_dirty(dat);
@ -407,7 +407,7 @@ int nilfs_dat_translate(struct inode *dat, __u64 vblocknr, sector_t *blocknrp)
struct buffer_head *entry_bh, *bh; struct buffer_head *entry_bh, *bh;
struct nilfs_dat_entry *entry; struct nilfs_dat_entry *entry;
sector_t blocknr; sector_t blocknr;
void *kaddr; size_t offset;
int ret; int ret;
ret = nilfs_palloc_get_entry_block(dat, vblocknr, 0, &entry_bh); ret = nilfs_palloc_get_entry_block(dat, vblocknr, 0, &entry_bh);
@ -423,8 +423,8 @@ int nilfs_dat_translate(struct inode *dat, __u64 vblocknr, sector_t *blocknrp)
} }
} }
kaddr = kmap_local_page(entry_bh->b_page); offset = nilfs_palloc_entry_offset(dat, vblocknr, entry_bh);
entry = nilfs_palloc_block_get_entry(dat, vblocknr, entry_bh, kaddr); entry = kmap_local_folio(entry_bh->b_folio, offset);
blocknr = le64_to_cpu(entry->de_blocknr); blocknr = le64_to_cpu(entry->de_blocknr);
if (blocknr == 0) { if (blocknr == 0) {
ret = -ENOENT; ret = -ENOENT;
@ -433,7 +433,7 @@ int nilfs_dat_translate(struct inode *dat, __u64 vblocknr, sector_t *blocknrp)
*blocknrp = blocknr; *blocknrp = blocknr;
out: out:
kunmap_local(kaddr); kunmap_local(entry);
brelse(entry_bh); brelse(entry_bh);
return ret; return ret;
} }
@ -442,11 +442,12 @@ ssize_t nilfs_dat_get_vinfo(struct inode *dat, void *buf, unsigned int visz,
size_t nvi) size_t nvi)
{ {
struct buffer_head *entry_bh; struct buffer_head *entry_bh;
struct nilfs_dat_entry *entry; struct nilfs_dat_entry *entry, *first_entry;
struct nilfs_vinfo *vinfo = buf; struct nilfs_vinfo *vinfo = buf;
__u64 first, last; __u64 first, last;
void *kaddr; size_t offset;
unsigned long entries_per_block = NILFS_MDT(dat)->mi_entries_per_block; unsigned long entries_per_block = NILFS_MDT(dat)->mi_entries_per_block;
unsigned int entry_size = NILFS_MDT(dat)->mi_entry_size;
int i, j, n, ret; int i, j, n, ret;
for (i = 0; i < nvi; i += n) { for (i = 0; i < nvi; i += n) {
@ -454,23 +455,28 @@ ssize_t nilfs_dat_get_vinfo(struct inode *dat, void *buf, unsigned int visz,
0, &entry_bh); 0, &entry_bh);
if (ret < 0) if (ret < 0)
return ret; return ret;
kaddr = kmap_local_page(entry_bh->b_page);
/* last virtual block number in this block */
first = vinfo->vi_vblocknr; first = vinfo->vi_vblocknr;
first = div64_ul(first, entries_per_block); first = div64_ul(first, entries_per_block);
first *= entries_per_block; first *= entries_per_block;
/* first virtual block number in this block */
last = first + entries_per_block - 1; last = first + entries_per_block - 1;
/* last virtual block number in this block */
offset = nilfs_palloc_entry_offset(dat, first, entry_bh);
first_entry = kmap_local_folio(entry_bh->b_folio, offset);
for (j = i, n = 0; for (j = i, n = 0;
j < nvi && vinfo->vi_vblocknr >= first && j < nvi && vinfo->vi_vblocknr >= first &&
vinfo->vi_vblocknr <= last; vinfo->vi_vblocknr <= last;
j++, n++, vinfo = (void *)vinfo + visz) { j++, n++, vinfo = (void *)vinfo + visz) {
entry = nilfs_palloc_block_get_entry( entry = (void *)first_entry +
dat, vinfo->vi_vblocknr, entry_bh, kaddr); (vinfo->vi_vblocknr - first) * entry_size;
vinfo->vi_start = le64_to_cpu(entry->de_start); vinfo->vi_start = le64_to_cpu(entry->de_start);
vinfo->vi_end = le64_to_cpu(entry->de_end); vinfo->vi_end = le64_to_cpu(entry->de_end);
vinfo->vi_blocknr = le64_to_cpu(entry->de_blocknr); vinfo->vi_blocknr = le64_to_cpu(entry->de_blocknr);
} }
kunmap_local(kaddr); kunmap_local(first_entry);
brelse(entry_bh); brelse(entry_bh);
} }

View File

@ -95,7 +95,7 @@ static void nilfs_commit_chunk(struct folio *folio,
unsigned int nr_dirty; unsigned int nr_dirty;
int err; int err;
nr_dirty = nilfs_page_count_clean_buffers(&folio->page, from, to); nr_dirty = nilfs_page_count_clean_buffers(folio, from, to);
copied = block_write_end(NULL, mapping, pos, len, len, folio, NULL); copied = block_write_end(NULL, mapping, pos, len, len, folio, NULL);
if (pos + copied > dir->i_size) if (pos + copied > dir->i_size)
i_size_write(dir, pos + copied); i_size_write(dir, pos + copied);

View File

@ -98,7 +98,7 @@ int nilfs_ifile_delete_inode(struct inode *ifile, ino_t ino)
.pr_entry_nr = ino, .pr_entry_bh = NULL .pr_entry_nr = ino, .pr_entry_bh = NULL
}; };
struct nilfs_inode *raw_inode; struct nilfs_inode *raw_inode;
void *kaddr; size_t offset;
int ret; int ret;
ret = nilfs_palloc_prepare_free_entry(ifile, &req); ret = nilfs_palloc_prepare_free_entry(ifile, &req);
@ -113,11 +113,11 @@ int nilfs_ifile_delete_inode(struct inode *ifile, ino_t ino)
return ret; return ret;
} }
kaddr = kmap_local_page(req.pr_entry_bh->b_page); offset = nilfs_palloc_entry_offset(ifile, req.pr_entry_nr,
raw_inode = nilfs_palloc_block_get_entry(ifile, req.pr_entry_nr, req.pr_entry_bh);
req.pr_entry_bh, kaddr); raw_inode = kmap_local_folio(req.pr_entry_bh->b_folio, offset);
raw_inode->i_flags = 0; raw_inode->i_flags = 0;
kunmap_local(kaddr); kunmap_local(raw_inode);
mark_buffer_dirty(req.pr_entry_bh); mark_buffer_dirty(req.pr_entry_bh);
brelse(req.pr_entry_bh); brelse(req.pr_entry_bh);

View File

@ -21,9 +21,9 @@
static inline struct nilfs_inode * static inline struct nilfs_inode *
nilfs_ifile_map_inode(struct inode *ifile, ino_t ino, struct buffer_head *ibh) nilfs_ifile_map_inode(struct inode *ifile, ino_t ino, struct buffer_head *ibh)
{ {
void *kaddr = kmap_local_page(ibh->b_page); size_t __offset_in_folio = nilfs_palloc_entry_offset(ifile, ino, ibh);
return nilfs_palloc_block_get_entry(ifile, ino, ibh, kaddr); return kmap_local_folio(ibh->b_folio, __offset_in_folio);
} }
static inline void nilfs_ifile_unmap_inode(struct nilfs_inode *raw_inode) static inline void nilfs_ifile_unmap_inode(struct nilfs_inode *raw_inode)

View File

@ -170,37 +170,6 @@ static int nilfs_writepages(struct address_space *mapping,
return err; return err;
} }
static int nilfs_writepage(struct page *page, struct writeback_control *wbc)
{
struct folio *folio = page_folio(page);
struct inode *inode = folio->mapping->host;
int err;
if (sb_rdonly(inode->i_sb)) {
/*
* It means that filesystem was remounted in read-only
* mode because of error or metadata corruption. But we
* have dirty pages that try to be flushed in background.
* So, here we simply discard this dirty page.
*/
nilfs_clear_folio_dirty(folio);
folio_unlock(folio);
return -EROFS;
}
folio_redirty_for_writepage(wbc, folio);
folio_unlock(folio);
if (wbc->sync_mode == WB_SYNC_ALL) {
err = nilfs_construct_segment(inode->i_sb);
if (unlikely(err))
return err;
} else if (wbc->for_reclaim)
nilfs_flush_segment(inode->i_sb, inode->i_ino);
return 0;
}
static bool nilfs_dirty_folio(struct address_space *mapping, static bool nilfs_dirty_folio(struct address_space *mapping,
struct folio *folio) struct folio *folio)
{ {
@ -273,7 +242,7 @@ static int nilfs_write_end(struct file *file, struct address_space *mapping,
unsigned int nr_dirty; unsigned int nr_dirty;
int err; int err;
nr_dirty = nilfs_page_count_clean_buffers(&folio->page, start, nr_dirty = nilfs_page_count_clean_buffers(folio, start,
start + copied); start + copied);
copied = generic_write_end(file, mapping, pos, len, copied, folio, copied = generic_write_end(file, mapping, pos, len, copied, folio,
fsdata); fsdata);
@ -295,7 +264,6 @@ nilfs_direct_IO(struct kiocb *iocb, struct iov_iter *iter)
} }
const struct address_space_operations nilfs_aops = { const struct address_space_operations nilfs_aops = {
.writepage = nilfs_writepage,
.read_folio = nilfs_read_folio, .read_folio = nilfs_read_folio,
.writepages = nilfs_writepages, .writepages = nilfs_writepages,
.dirty_folio = nilfs_dirty_folio, .dirty_folio = nilfs_dirty_folio,
@ -304,6 +272,7 @@ const struct address_space_operations nilfs_aops = {
.write_end = nilfs_write_end, .write_end = nilfs_write_end,
.invalidate_folio = block_invalidate_folio, .invalidate_folio = block_invalidate_folio,
.direct_IO = nilfs_direct_IO, .direct_IO = nilfs_direct_IO,
.migrate_folio = buffer_migrate_folio_norefs,
.is_partially_uptodate = block_is_partially_uptodate, .is_partially_uptodate = block_is_partially_uptodate,
}; };

View File

@ -33,7 +33,8 @@ nilfs_mdt_insert_new_block(struct inode *inode, unsigned long block,
struct buffer_head *, void *)) struct buffer_head *, void *))
{ {
struct nilfs_inode_info *ii = NILFS_I(inode); struct nilfs_inode_info *ii = NILFS_I(inode);
void *kaddr; struct folio *folio = bh->b_folio;
void *from;
int ret; int ret;
/* Caller exclude read accesses using page lock */ /* Caller exclude read accesses using page lock */
@ -47,12 +48,14 @@ nilfs_mdt_insert_new_block(struct inode *inode, unsigned long block,
set_buffer_mapped(bh); set_buffer_mapped(bh);
kaddr = kmap_local_page(bh->b_page); /* Initialize block (block size > PAGE_SIZE not yet supported) */
memset(kaddr + bh_offset(bh), 0, i_blocksize(inode)); from = kmap_local_folio(folio, offset_in_folio(folio, bh->b_data));
memset(from, 0, bh->b_size);
if (init_block) if (init_block)
init_block(inode, bh, kaddr); init_block(inode, bh, from);
flush_dcache_page(bh->b_page); kunmap_local(from);
kunmap_local(kaddr);
flush_dcache_folio(folio);
set_buffer_uptodate(bh); set_buffer_uptodate(bh);
mark_buffer_dirty(bh); mark_buffer_dirty(bh);
@ -395,10 +398,9 @@ int nilfs_mdt_fetch_dirty(struct inode *inode)
return test_bit(NILFS_I_DIRTY, &ii->i_state); return test_bit(NILFS_I_DIRTY, &ii->i_state);
} }
static int static int nilfs_mdt_write_folio(struct folio *folio,
nilfs_mdt_write_page(struct page *page, struct writeback_control *wbc) struct writeback_control *wbc)
{ {
struct folio *folio = page_folio(page);
struct inode *inode = folio->mapping->host; struct inode *inode = folio->mapping->host;
struct super_block *sb; struct super_block *sb;
int err = 0; int err = 0;
@ -431,11 +433,23 @@ nilfs_mdt_write_page(struct page *page, struct writeback_control *wbc)
return err; return err;
} }
static int nilfs_mdt_writeback(struct address_space *mapping,
struct writeback_control *wbc)
{
struct folio *folio = NULL;
int error;
while ((folio = writeback_iter(mapping, wbc, folio, &error)))
error = nilfs_mdt_write_folio(folio, wbc);
return error;
}
static const struct address_space_operations def_mdt_aops = { static const struct address_space_operations def_mdt_aops = {
.dirty_folio = block_dirty_folio, .dirty_folio = block_dirty_folio,
.invalidate_folio = block_invalidate_folio, .invalidate_folio = block_invalidate_folio,
.writepage = nilfs_mdt_write_page, .writepages = nilfs_mdt_writeback,
.migrate_folio = buffer_migrate_folio_norefs,
}; };
static const struct inode_operations def_mdt_iops; static const struct inode_operations def_mdt_iops;
@ -570,7 +584,8 @@ int nilfs_mdt_freeze_buffer(struct inode *inode, struct buffer_head *bh)
if (!bh_frozen) if (!bh_frozen)
bh_frozen = create_empty_buffers(folio, 1 << blkbits, 0); bh_frozen = create_empty_buffers(folio, 1 << blkbits, 0);
bh_frozen = get_nth_bh(bh_frozen, bh_offset(bh) >> blkbits); bh_frozen = get_nth_bh(bh_frozen,
offset_in_folio(folio, bh->b_data) >> blkbits);
if (!buffer_uptodate(bh_frozen)) if (!buffer_uptodate(bh_frozen))
nilfs_copy_buffer(bh_frozen, bh); nilfs_copy_buffer(bh_frozen, bh);
@ -600,7 +615,8 @@ nilfs_mdt_get_frozen_buffer(struct inode *inode, struct buffer_head *bh)
if (!IS_ERR(folio)) { if (!IS_ERR(folio)) {
bh_frozen = folio_buffers(folio); bh_frozen = folio_buffers(folio);
if (bh_frozen) { if (bh_frozen) {
n = bh_offset(bh) >> inode->i_blkbits; n = offset_in_folio(folio, bh->b_data) >>
inode->i_blkbits;
bh_frozen = get_nth_bh(bh_frozen, n); bh_frozen = get_nth_bh(bh_frozen, n);
} }
folio_unlock(folio); folio_unlock(folio);

View File

@ -422,14 +422,14 @@ void nilfs_clear_folio_dirty(struct folio *folio)
__nilfs_clear_folio_dirty(folio); __nilfs_clear_folio_dirty(folio);
} }
unsigned int nilfs_page_count_clean_buffers(struct page *page, unsigned int nilfs_page_count_clean_buffers(struct folio *folio,
unsigned int from, unsigned int to) unsigned int from, unsigned int to)
{ {
unsigned int block_start, block_end; unsigned int block_start, block_end;
struct buffer_head *bh, *head; struct buffer_head *bh, *head;
unsigned int nc = 0; unsigned int nc = 0;
for (bh = head = page_buffers(page), block_start = 0; for (bh = head = folio_buffers(folio), block_start = 0;
bh != head || !block_start; bh != head || !block_start;
block_start = block_end, bh = bh->b_this_page) { block_start = block_end, bh = bh->b_this_page) {
block_end = block_start + bh->b_size; block_end = block_start + bh->b_size;

View File

@ -43,8 +43,8 @@ int nilfs_copy_dirty_pages(struct address_space *, struct address_space *);
void nilfs_copy_back_pages(struct address_space *, struct address_space *); void nilfs_copy_back_pages(struct address_space *, struct address_space *);
void nilfs_clear_folio_dirty(struct folio *folio); void nilfs_clear_folio_dirty(struct folio *folio);
void nilfs_clear_dirty_pages(struct address_space *mapping); void nilfs_clear_dirty_pages(struct address_space *mapping);
unsigned int nilfs_page_count_clean_buffers(struct page *, unsigned int, unsigned int nilfs_page_count_clean_buffers(struct folio *folio,
unsigned int); unsigned int from, unsigned int to);
unsigned long nilfs_find_uncommitted_extent(struct inode *inode, unsigned long nilfs_find_uncommitted_extent(struct inode *inode,
sector_t start_blk, sector_t start_blk,
sector_t *blkoff); sector_t *blkoff);

View File

@ -481,19 +481,16 @@ static int nilfs_prepare_segment_for_recovery(struct the_nilfs *nilfs,
static int nilfs_recovery_copy_block(struct the_nilfs *nilfs, static int nilfs_recovery_copy_block(struct the_nilfs *nilfs,
struct nilfs_recovery_block *rb, struct nilfs_recovery_block *rb,
loff_t pos, struct page *page) loff_t pos, struct folio *folio)
{ {
struct buffer_head *bh_org; struct buffer_head *bh_org;
size_t from = pos & ~PAGE_MASK; size_t from = offset_in_folio(folio, pos);
void *kaddr;
bh_org = __bread(nilfs->ns_bdev, rb->blocknr, nilfs->ns_blocksize); bh_org = __bread(nilfs->ns_bdev, rb->blocknr, nilfs->ns_blocksize);
if (unlikely(!bh_org)) if (unlikely(!bh_org))
return -EIO; return -EIO;
kaddr = kmap_local_page(page); memcpy_to_folio(folio, from, bh_org->b_data, bh_org->b_size);
memcpy(kaddr + from, bh_org->b_data, bh_org->b_size);
kunmap_local(kaddr);
brelse(bh_org); brelse(bh_org);
return 0; return 0;
} }
@ -531,13 +528,13 @@ static int nilfs_recover_dsync_blocks(struct the_nilfs *nilfs,
goto failed_inode; goto failed_inode;
} }
err = nilfs_recovery_copy_block(nilfs, rb, pos, &folio->page); err = nilfs_recovery_copy_block(nilfs, rb, pos, folio);
if (unlikely(err)) if (unlikely(err))
goto failed_page; goto failed_folio;
err = nilfs_set_file_dirty(inode, 1); err = nilfs_set_file_dirty(inode, 1);
if (unlikely(err)) if (unlikely(err))
goto failed_page; goto failed_folio;
block_write_end(NULL, inode->i_mapping, pos, blocksize, block_write_end(NULL, inode->i_mapping, pos, blocksize,
blocksize, folio, NULL); blocksize, folio, NULL);
@ -548,7 +545,7 @@ static int nilfs_recover_dsync_blocks(struct the_nilfs *nilfs,
(*nr_salvaged_blocks)++; (*nr_salvaged_blocks)++;
goto next; goto next;
failed_page: failed_folio:
folio_unlock(folio); folio_unlock(folio);
folio_put(folio); folio_put(folio);

View File

@ -205,7 +205,6 @@ static void nilfs_segbuf_fill_in_data_crc(struct nilfs_segment_buffer *segbuf,
{ {
struct buffer_head *bh; struct buffer_head *bh;
struct nilfs_segment_summary *raw_sum; struct nilfs_segment_summary *raw_sum;
void *kaddr;
u32 crc; u32 crc;
bh = list_entry(segbuf->sb_segsum_buffers.next, struct buffer_head, bh = list_entry(segbuf->sb_segsum_buffers.next, struct buffer_head,
@ -220,9 +219,13 @@ static void nilfs_segbuf_fill_in_data_crc(struct nilfs_segment_buffer *segbuf,
crc = crc32_le(crc, bh->b_data, bh->b_size); crc = crc32_le(crc, bh->b_data, bh->b_size);
} }
list_for_each_entry(bh, &segbuf->sb_payload_buffers, b_assoc_buffers) { list_for_each_entry(bh, &segbuf->sb_payload_buffers, b_assoc_buffers) {
kaddr = kmap_local_page(bh->b_page); size_t offset = offset_in_folio(bh->b_folio, bh->b_data);
crc = crc32_le(crc, kaddr + bh_offset(bh), bh->b_size); unsigned char *from;
kunmap_local(kaddr);
/* Do not support block sizes larger than PAGE_SIZE */
from = kmap_local_folio(bh->b_folio, offset);
crc = crc32_le(crc, from, bh->b_size);
kunmap_local(from);
} }
raw_sum->ss_datasum = cpu_to_le32(crc); raw_sum->ss_datasum = cpu_to_le32(crc);
} }
@ -374,7 +377,7 @@ static int nilfs_segbuf_submit_bh(struct nilfs_segment_buffer *segbuf,
struct nilfs_write_info *wi, struct nilfs_write_info *wi,
struct buffer_head *bh) struct buffer_head *bh)
{ {
int len, err; int err;
BUG_ON(wi->nr_vecs <= 0); BUG_ON(wi->nr_vecs <= 0);
repeat: repeat:
@ -385,8 +388,8 @@ static int nilfs_segbuf_submit_bh(struct nilfs_segment_buffer *segbuf,
(wi->nilfs->ns_blocksize_bits - 9); (wi->nilfs->ns_blocksize_bits - 9);
} }
len = bio_add_page(wi->bio, bh->b_page, bh->b_size, bh_offset(bh)); if (bio_add_folio(wi->bio, bh->b_folio, bh->b_size,
if (len == bh->b_size) { offset_in_folio(bh->b_folio, bh->b_data))) {
wi->end++; wi->end++;
return 0; return 0;
} }

View File

@ -70,11 +70,20 @@ nilfs_sufile_segment_usages_in_block(const struct inode *sufile, __u64 curr,
max - curr + 1); max - curr + 1);
} }
static struct nilfs_segment_usage * /**
nilfs_sufile_block_get_segment_usage(const struct inode *sufile, __u64 segnum, * nilfs_sufile_segment_usage_offset - calculate the byte offset of a segment
struct buffer_head *bh, void *kaddr) * usage entry in the folio containing it
* @sufile: segment usage file inode
* @segnum: number of segment usage
* @bh: buffer head of block containing segment usage indexed by @segnum
*
* Return: Byte offset in the folio of the segment usage entry.
*/
static size_t nilfs_sufile_segment_usage_offset(const struct inode *sufile,
__u64 segnum,
struct buffer_head *bh)
{ {
return kaddr + bh_offset(bh) + return offset_in_folio(bh->b_folio, bh->b_data) +
nilfs_sufile_get_offset(sufile, segnum) * nilfs_sufile_get_offset(sufile, segnum) *
NILFS_MDT(sufile)->mi_entry_size; NILFS_MDT(sufile)->mi_entry_size;
} }
@ -112,13 +121,11 @@ static void nilfs_sufile_mod_counter(struct buffer_head *header_bh,
u64 ncleanadd, u64 ndirtyadd) u64 ncleanadd, u64 ndirtyadd)
{ {
struct nilfs_sufile_header *header; struct nilfs_sufile_header *header;
void *kaddr;
kaddr = kmap_local_page(header_bh->b_page); header = kmap_local_folio(header_bh->b_folio, 0);
header = kaddr + bh_offset(header_bh);
le64_add_cpu(&header->sh_ncleansegs, ncleanadd); le64_add_cpu(&header->sh_ncleansegs, ncleanadd);
le64_add_cpu(&header->sh_ndirtysegs, ndirtyadd); le64_add_cpu(&header->sh_ndirtysegs, ndirtyadd);
kunmap_local(kaddr); kunmap_local(header);
mark_buffer_dirty(header_bh); mark_buffer_dirty(header_bh);
} }
@ -313,6 +320,7 @@ int nilfs_sufile_alloc(struct inode *sufile, __u64 *segnump)
struct nilfs_sufile_info *sui = NILFS_SUI(sufile); struct nilfs_sufile_info *sui = NILFS_SUI(sufile);
size_t susz = NILFS_MDT(sufile)->mi_entry_size; size_t susz = NILFS_MDT(sufile)->mi_entry_size;
__u64 segnum, maxsegnum, last_alloc; __u64 segnum, maxsegnum, last_alloc;
size_t offset;
void *kaddr; void *kaddr;
unsigned long nsegments, nsus, cnt; unsigned long nsegments, nsus, cnt;
int ret, j; int ret, j;
@ -322,10 +330,9 @@ int nilfs_sufile_alloc(struct inode *sufile, __u64 *segnump)
ret = nilfs_sufile_get_header_block(sufile, &header_bh); ret = nilfs_sufile_get_header_block(sufile, &header_bh);
if (ret < 0) if (ret < 0)
goto out_sem; goto out_sem;
kaddr = kmap_local_page(header_bh->b_page); header = kmap_local_folio(header_bh->b_folio, 0);
header = kaddr + bh_offset(header_bh);
last_alloc = le64_to_cpu(header->sh_last_alloc); last_alloc = le64_to_cpu(header->sh_last_alloc);
kunmap_local(kaddr); kunmap_local(header);
nsegments = nilfs_sufile_get_nsegments(sufile); nsegments = nilfs_sufile_get_nsegments(sufile);
maxsegnum = sui->allocmax; maxsegnum = sui->allocmax;
@ -359,9 +366,10 @@ int nilfs_sufile_alloc(struct inode *sufile, __u64 *segnump)
&su_bh); &su_bh);
if (ret < 0) if (ret < 0)
goto out_header; goto out_header;
kaddr = kmap_local_page(su_bh->b_page);
su = nilfs_sufile_block_get_segment_usage( offset = nilfs_sufile_segment_usage_offset(sufile, segnum,
sufile, segnum, su_bh, kaddr); su_bh);
su = kaddr = kmap_local_folio(su_bh->b_folio, offset);
nsus = nilfs_sufile_segment_usages_in_block( nsus = nilfs_sufile_segment_usages_in_block(
sufile, segnum, maxsegnum); sufile, segnum, maxsegnum);
@ -372,12 +380,11 @@ int nilfs_sufile_alloc(struct inode *sufile, __u64 *segnump)
nilfs_segment_usage_set_dirty(su); nilfs_segment_usage_set_dirty(su);
kunmap_local(kaddr); kunmap_local(kaddr);
kaddr = kmap_local_page(header_bh->b_page); header = kmap_local_folio(header_bh->b_folio, 0);
header = kaddr + bh_offset(header_bh);
le64_add_cpu(&header->sh_ncleansegs, -1); le64_add_cpu(&header->sh_ncleansegs, -1);
le64_add_cpu(&header->sh_ndirtysegs, 1); le64_add_cpu(&header->sh_ndirtysegs, 1);
header->sh_last_alloc = cpu_to_le64(segnum); header->sh_last_alloc = cpu_to_le64(segnum);
kunmap_local(kaddr); kunmap_local(header);
sui->ncleansegs--; sui->ncleansegs--;
mark_buffer_dirty(header_bh); mark_buffer_dirty(header_bh);
@ -411,18 +418,18 @@ void nilfs_sufile_do_cancel_free(struct inode *sufile, __u64 segnum,
struct buffer_head *su_bh) struct buffer_head *su_bh)
{ {
struct nilfs_segment_usage *su; struct nilfs_segment_usage *su;
void *kaddr; size_t offset;
kaddr = kmap_local_page(su_bh->b_page); offset = nilfs_sufile_segment_usage_offset(sufile, segnum, su_bh);
su = nilfs_sufile_block_get_segment_usage(sufile, segnum, su_bh, kaddr); su = kmap_local_folio(su_bh->b_folio, offset);
if (unlikely(!nilfs_segment_usage_clean(su))) { if (unlikely(!nilfs_segment_usage_clean(su))) {
nilfs_warn(sufile->i_sb, "%s: segment %llu must be clean", nilfs_warn(sufile->i_sb, "%s: segment %llu must be clean",
__func__, (unsigned long long)segnum); __func__, (unsigned long long)segnum);
kunmap_local(kaddr); kunmap_local(su);
return; return;
} }
nilfs_segment_usage_set_dirty(su); nilfs_segment_usage_set_dirty(su);
kunmap_local(kaddr); kunmap_local(su);
nilfs_sufile_mod_counter(header_bh, -1, 1); nilfs_sufile_mod_counter(header_bh, -1, 1);
NILFS_SUI(sufile)->ncleansegs--; NILFS_SUI(sufile)->ncleansegs--;
@ -436,14 +443,14 @@ void nilfs_sufile_do_scrap(struct inode *sufile, __u64 segnum,
struct buffer_head *su_bh) struct buffer_head *su_bh)
{ {
struct nilfs_segment_usage *su; struct nilfs_segment_usage *su;
void *kaddr; size_t offset;
int clean, dirty; int clean, dirty;
kaddr = kmap_local_page(su_bh->b_page); offset = nilfs_sufile_segment_usage_offset(sufile, segnum, su_bh);
su = nilfs_sufile_block_get_segment_usage(sufile, segnum, su_bh, kaddr); su = kmap_local_folio(su_bh->b_folio, offset);
if (su->su_flags == cpu_to_le32(BIT(NILFS_SEGMENT_USAGE_DIRTY)) && if (su->su_flags == cpu_to_le32(BIT(NILFS_SEGMENT_USAGE_DIRTY)) &&
su->su_nblocks == cpu_to_le32(0)) { su->su_nblocks == cpu_to_le32(0)) {
kunmap_local(kaddr); kunmap_local(su);
return; return;
} }
clean = nilfs_segment_usage_clean(su); clean = nilfs_segment_usage_clean(su);
@ -453,7 +460,7 @@ void nilfs_sufile_do_scrap(struct inode *sufile, __u64 segnum,
su->su_lastmod = cpu_to_le64(0); su->su_lastmod = cpu_to_le64(0);
su->su_nblocks = cpu_to_le32(0); su->su_nblocks = cpu_to_le32(0);
su->su_flags = cpu_to_le32(BIT(NILFS_SEGMENT_USAGE_DIRTY)); su->su_flags = cpu_to_le32(BIT(NILFS_SEGMENT_USAGE_DIRTY));
kunmap_local(kaddr); kunmap_local(su);
nilfs_sufile_mod_counter(header_bh, clean ? (u64)-1 : 0, dirty ? 0 : 1); nilfs_sufile_mod_counter(header_bh, clean ? (u64)-1 : 0, dirty ? 0 : 1);
NILFS_SUI(sufile)->ncleansegs -= clean; NILFS_SUI(sufile)->ncleansegs -= clean;
@ -467,15 +474,15 @@ void nilfs_sufile_do_free(struct inode *sufile, __u64 segnum,
struct buffer_head *su_bh) struct buffer_head *su_bh)
{ {
struct nilfs_segment_usage *su; struct nilfs_segment_usage *su;
void *kaddr; size_t offset;
int sudirty; int sudirty;
kaddr = kmap_local_page(su_bh->b_page); offset = nilfs_sufile_segment_usage_offset(sufile, segnum, su_bh);
su = nilfs_sufile_block_get_segment_usage(sufile, segnum, su_bh, kaddr); su = kmap_local_folio(su_bh->b_folio, offset);
if (nilfs_segment_usage_clean(su)) { if (nilfs_segment_usage_clean(su)) {
nilfs_warn(sufile->i_sb, "%s: segment %llu is already clean", nilfs_warn(sufile->i_sb, "%s: segment %llu is already clean",
__func__, (unsigned long long)segnum); __func__, (unsigned long long)segnum);
kunmap_local(kaddr); kunmap_local(su);
return; return;
} }
if (unlikely(nilfs_segment_usage_error(su))) if (unlikely(nilfs_segment_usage_error(su)))
@ -488,7 +495,7 @@ void nilfs_sufile_do_free(struct inode *sufile, __u64 segnum,
(unsigned long long)segnum); (unsigned long long)segnum);
nilfs_segment_usage_set_clean(su); nilfs_segment_usage_set_clean(su);
kunmap_local(kaddr); kunmap_local(su);
mark_buffer_dirty(su_bh); mark_buffer_dirty(su_bh);
nilfs_sufile_mod_counter(header_bh, 1, sudirty ? (u64)-1 : 0); nilfs_sufile_mod_counter(header_bh, 1, sudirty ? (u64)-1 : 0);
@ -507,7 +514,7 @@ void nilfs_sufile_do_free(struct inode *sufile, __u64 segnum,
int nilfs_sufile_mark_dirty(struct inode *sufile, __u64 segnum) int nilfs_sufile_mark_dirty(struct inode *sufile, __u64 segnum)
{ {
struct buffer_head *bh; struct buffer_head *bh;
void *kaddr; size_t offset;
struct nilfs_segment_usage *su; struct nilfs_segment_usage *su;
int ret; int ret;
@ -523,12 +530,12 @@ int nilfs_sufile_mark_dirty(struct inode *sufile, __u64 segnum)
goto out_sem; goto out_sem;
} }
kaddr = kmap_local_page(bh->b_page); offset = nilfs_sufile_segment_usage_offset(sufile, segnum, bh);
su = nilfs_sufile_block_get_segment_usage(sufile, segnum, bh, kaddr); su = kmap_local_folio(bh->b_folio, offset);
if (unlikely(nilfs_segment_usage_error(su))) { if (unlikely(nilfs_segment_usage_error(su))) {
struct the_nilfs *nilfs = sufile->i_sb->s_fs_info; struct the_nilfs *nilfs = sufile->i_sb->s_fs_info;
kunmap_local(kaddr); kunmap_local(su);
brelse(bh); brelse(bh);
if (nilfs_segment_is_active(nilfs, segnum)) { if (nilfs_segment_is_active(nilfs, segnum)) {
nilfs_error(sufile->i_sb, nilfs_error(sufile->i_sb,
@ -546,7 +553,7 @@ int nilfs_sufile_mark_dirty(struct inode *sufile, __u64 segnum)
ret = -EIO; ret = -EIO;
} else { } else {
nilfs_segment_usage_set_dirty(su); nilfs_segment_usage_set_dirty(su);
kunmap_local(kaddr); kunmap_local(su);
mark_buffer_dirty(bh); mark_buffer_dirty(bh);
nilfs_mdt_mark_dirty(sufile); nilfs_mdt_mark_dirty(sufile);
brelse(bh); brelse(bh);
@ -568,7 +575,7 @@ int nilfs_sufile_set_segment_usage(struct inode *sufile, __u64 segnum,
{ {
struct buffer_head *bh; struct buffer_head *bh;
struct nilfs_segment_usage *su; struct nilfs_segment_usage *su;
void *kaddr; size_t offset;
int ret; int ret;
down_write(&NILFS_MDT(sufile)->mi_sem); down_write(&NILFS_MDT(sufile)->mi_sem);
@ -576,8 +583,8 @@ int nilfs_sufile_set_segment_usage(struct inode *sufile, __u64 segnum,
if (ret < 0) if (ret < 0)
goto out_sem; goto out_sem;
kaddr = kmap_local_page(bh->b_page); offset = nilfs_sufile_segment_usage_offset(sufile, segnum, bh);
su = nilfs_sufile_block_get_segment_usage(sufile, segnum, bh, kaddr); su = kmap_local_folio(bh->b_folio, offset);
if (modtime) { if (modtime) {
/* /*
* Check segusage error and set su_lastmod only when updating * Check segusage error and set su_lastmod only when updating
@ -587,7 +594,7 @@ int nilfs_sufile_set_segment_usage(struct inode *sufile, __u64 segnum,
su->su_lastmod = cpu_to_le64(modtime); su->su_lastmod = cpu_to_le64(modtime);
} }
su->su_nblocks = cpu_to_le32(nblocks); su->su_nblocks = cpu_to_le32(nblocks);
kunmap_local(kaddr); kunmap_local(su);
mark_buffer_dirty(bh); mark_buffer_dirty(bh);
nilfs_mdt_mark_dirty(sufile); nilfs_mdt_mark_dirty(sufile);
@ -619,7 +626,6 @@ int nilfs_sufile_get_stat(struct inode *sufile, struct nilfs_sustat *sustat)
struct buffer_head *header_bh; struct buffer_head *header_bh;
struct nilfs_sufile_header *header; struct nilfs_sufile_header *header;
struct the_nilfs *nilfs = sufile->i_sb->s_fs_info; struct the_nilfs *nilfs = sufile->i_sb->s_fs_info;
void *kaddr;
int ret; int ret;
down_read(&NILFS_MDT(sufile)->mi_sem); down_read(&NILFS_MDT(sufile)->mi_sem);
@ -628,8 +634,7 @@ int nilfs_sufile_get_stat(struct inode *sufile, struct nilfs_sustat *sustat)
if (ret < 0) if (ret < 0)
goto out_sem; goto out_sem;
kaddr = kmap_local_page(header_bh->b_page); header = kmap_local_folio(header_bh->b_folio, 0);
header = kaddr + bh_offset(header_bh);
sustat->ss_nsegs = nilfs_sufile_get_nsegments(sufile); sustat->ss_nsegs = nilfs_sufile_get_nsegments(sufile);
sustat->ss_ncleansegs = le64_to_cpu(header->sh_ncleansegs); sustat->ss_ncleansegs = le64_to_cpu(header->sh_ncleansegs);
sustat->ss_ndirtysegs = le64_to_cpu(header->sh_ndirtysegs); sustat->ss_ndirtysegs = le64_to_cpu(header->sh_ndirtysegs);
@ -638,7 +643,7 @@ int nilfs_sufile_get_stat(struct inode *sufile, struct nilfs_sustat *sustat)
spin_lock(&nilfs->ns_last_segment_lock); spin_lock(&nilfs->ns_last_segment_lock);
sustat->ss_prot_seq = nilfs->ns_prot_seq; sustat->ss_prot_seq = nilfs->ns_prot_seq;
spin_unlock(&nilfs->ns_last_segment_lock); spin_unlock(&nilfs->ns_last_segment_lock);
kunmap_local(kaddr); kunmap_local(header);
brelse(header_bh); brelse(header_bh);
out_sem: out_sem:
@ -651,18 +656,18 @@ void nilfs_sufile_do_set_error(struct inode *sufile, __u64 segnum,
struct buffer_head *su_bh) struct buffer_head *su_bh)
{ {
struct nilfs_segment_usage *su; struct nilfs_segment_usage *su;
void *kaddr; size_t offset;
int suclean; int suclean;
kaddr = kmap_local_page(su_bh->b_page); offset = nilfs_sufile_segment_usage_offset(sufile, segnum, su_bh);
su = nilfs_sufile_block_get_segment_usage(sufile, segnum, su_bh, kaddr); su = kmap_local_folio(su_bh->b_folio, offset);
if (nilfs_segment_usage_error(su)) { if (nilfs_segment_usage_error(su)) {
kunmap_local(kaddr); kunmap_local(su);
return; return;
} }
suclean = nilfs_segment_usage_clean(su); suclean = nilfs_segment_usage_clean(su);
nilfs_segment_usage_set_error(su); nilfs_segment_usage_set_error(su);
kunmap_local(kaddr); kunmap_local(su);
if (suclean) { if (suclean) {
nilfs_sufile_mod_counter(header_bh, -1, 0); nilfs_sufile_mod_counter(header_bh, -1, 0);
@ -700,7 +705,7 @@ static int nilfs_sufile_truncate_range(struct inode *sufile,
unsigned long segusages_per_block; unsigned long segusages_per_block;
unsigned long nsegs, ncleaned; unsigned long nsegs, ncleaned;
__u64 segnum; __u64 segnum;
void *kaddr; size_t offset;
ssize_t n, nc; ssize_t n, nc;
int ret; int ret;
int j; int j;
@ -731,16 +736,16 @@ static int nilfs_sufile_truncate_range(struct inode *sufile,
/* hole */ /* hole */
continue; continue;
} }
kaddr = kmap_local_page(su_bh->b_page); offset = nilfs_sufile_segment_usage_offset(sufile, segnum,
su = nilfs_sufile_block_get_segment_usage( su_bh);
sufile, segnum, su_bh, kaddr); su = kmap_local_folio(su_bh->b_folio, offset);
su2 = su; su2 = su;
for (j = 0; j < n; j++, su = (void *)su + susz) { for (j = 0; j < n; j++, su = (void *)su + susz) {
if ((le32_to_cpu(su->su_flags) & if ((le32_to_cpu(su->su_flags) &
~BIT(NILFS_SEGMENT_USAGE_ERROR)) || ~BIT(NILFS_SEGMENT_USAGE_ERROR)) ||
nilfs_segment_is_active(nilfs, segnum + j)) { nilfs_segment_is_active(nilfs, segnum + j)) {
ret = -EBUSY; ret = -EBUSY;
kunmap_local(kaddr); kunmap_local(su2);
brelse(su_bh); brelse(su_bh);
goto out_header; goto out_header;
} }
@ -752,7 +757,7 @@ static int nilfs_sufile_truncate_range(struct inode *sufile,
nc++; nc++;
} }
} }
kunmap_local(kaddr); kunmap_local(su2);
if (nc > 0) { if (nc > 0) {
mark_buffer_dirty(su_bh); mark_buffer_dirty(su_bh);
ncleaned += nc; ncleaned += nc;
@ -799,7 +804,6 @@ int nilfs_sufile_resize(struct inode *sufile, __u64 newnsegs)
struct buffer_head *header_bh; struct buffer_head *header_bh;
struct nilfs_sufile_header *header; struct nilfs_sufile_header *header;
struct nilfs_sufile_info *sui = NILFS_SUI(sufile); struct nilfs_sufile_info *sui = NILFS_SUI(sufile);
void *kaddr;
unsigned long nsegs, nrsvsegs; unsigned long nsegs, nrsvsegs;
int ret = 0; int ret = 0;
@ -837,10 +841,9 @@ int nilfs_sufile_resize(struct inode *sufile, __u64 newnsegs)
sui->allocmin = 0; sui->allocmin = 0;
} }
kaddr = kmap_local_page(header_bh->b_page); header = kmap_local_folio(header_bh->b_folio, 0);
header = kaddr + bh_offset(header_bh);
header->sh_ncleansegs = cpu_to_le64(sui->ncleansegs); header->sh_ncleansegs = cpu_to_le64(sui->ncleansegs);
kunmap_local(kaddr); kunmap_local(header);
mark_buffer_dirty(header_bh); mark_buffer_dirty(header_bh);
nilfs_mdt_mark_dirty(sufile); nilfs_mdt_mark_dirty(sufile);
@ -874,6 +877,7 @@ ssize_t nilfs_sufile_get_suinfo(struct inode *sufile, __u64 segnum, void *buf,
struct nilfs_suinfo *si = buf; struct nilfs_suinfo *si = buf;
size_t susz = NILFS_MDT(sufile)->mi_entry_size; size_t susz = NILFS_MDT(sufile)->mi_entry_size;
struct the_nilfs *nilfs = sufile->i_sb->s_fs_info; struct the_nilfs *nilfs = sufile->i_sb->s_fs_info;
size_t offset;
void *kaddr; void *kaddr;
unsigned long nsegs, segusages_per_block; unsigned long nsegs, segusages_per_block;
ssize_t n; ssize_t n;
@ -901,9 +905,9 @@ ssize_t nilfs_sufile_get_suinfo(struct inode *sufile, __u64 segnum, void *buf,
continue; continue;
} }
kaddr = kmap_local_page(su_bh->b_page); offset = nilfs_sufile_segment_usage_offset(sufile, segnum,
su = nilfs_sufile_block_get_segment_usage( su_bh);
sufile, segnum, su_bh, kaddr); su = kaddr = kmap_local_folio(su_bh->b_folio, offset);
for (j = 0; j < n; for (j = 0; j < n;
j++, su = (void *)su + susz, si = (void *)si + sisz) { j++, su = (void *)su + susz, si = (void *)si + sisz) {
si->sui_lastmod = le64_to_cpu(su->su_lastmod); si->sui_lastmod = le64_to_cpu(su->su_lastmod);
@ -951,7 +955,7 @@ ssize_t nilfs_sufile_set_suinfo(struct inode *sufile, void *buf,
struct buffer_head *header_bh, *bh; struct buffer_head *header_bh, *bh;
struct nilfs_suinfo_update *sup, *supend = buf + supsz * nsup; struct nilfs_suinfo_update *sup, *supend = buf + supsz * nsup;
struct nilfs_segment_usage *su; struct nilfs_segment_usage *su;
void *kaddr; size_t offset;
unsigned long blkoff, prev_blkoff; unsigned long blkoff, prev_blkoff;
int cleansi, cleansu, dirtysi, dirtysu; int cleansi, cleansu, dirtysi, dirtysu;
long ncleaned = 0, ndirtied = 0; long ncleaned = 0, ndirtied = 0;
@ -983,9 +987,9 @@ ssize_t nilfs_sufile_set_suinfo(struct inode *sufile, void *buf,
goto out_header; goto out_header;
for (;;) { for (;;) {
kaddr = kmap_local_page(bh->b_page); offset = nilfs_sufile_segment_usage_offset(
su = nilfs_sufile_block_get_segment_usage( sufile, sup->sup_segnum, bh);
sufile, sup->sup_segnum, bh, kaddr); su = kmap_local_folio(bh->b_folio, offset);
if (nilfs_suinfo_update_lastmod(sup)) if (nilfs_suinfo_update_lastmod(sup))
su->su_lastmod = cpu_to_le64(sup->sup_sui.sui_lastmod); su->su_lastmod = cpu_to_le64(sup->sup_sui.sui_lastmod);
@ -1020,7 +1024,7 @@ ssize_t nilfs_sufile_set_suinfo(struct inode *sufile, void *buf,
su->su_flags = cpu_to_le32(sup->sup_sui.sui_flags); su->su_flags = cpu_to_le32(sup->sup_sui.sui_flags);
} }
kunmap_local(kaddr); kunmap_local(su);
sup = (void *)sup + supsz; sup = (void *)sup + supsz;
if (sup >= supend) if (sup >= supend)
@ -1076,6 +1080,7 @@ int nilfs_sufile_trim_fs(struct inode *sufile, struct fstrim_range *range)
struct the_nilfs *nilfs = sufile->i_sb->s_fs_info; struct the_nilfs *nilfs = sufile->i_sb->s_fs_info;
struct buffer_head *su_bh; struct buffer_head *su_bh;
struct nilfs_segment_usage *su; struct nilfs_segment_usage *su;
size_t offset;
void *kaddr; void *kaddr;
size_t n, i, susz = NILFS_MDT(sufile)->mi_entry_size; size_t n, i, susz = NILFS_MDT(sufile)->mi_entry_size;
sector_t seg_start, seg_end, start_block, end_block; sector_t seg_start, seg_end, start_block, end_block;
@ -1125,9 +1130,9 @@ int nilfs_sufile_trim_fs(struct inode *sufile, struct fstrim_range *range)
continue; continue;
} }
kaddr = kmap_local_page(su_bh->b_page); offset = nilfs_sufile_segment_usage_offset(sufile, segnum,
su = nilfs_sufile_block_get_segment_usage(sufile, segnum, su_bh);
su_bh, kaddr); su = kaddr = kmap_local_folio(su_bh->b_folio, offset);
for (i = 0; i < n; ++i, ++segnum, su = (void *)su + susz) { for (i = 0; i < n; ++i, ++segnum, su = (void *)su + susz) {
if (!nilfs_segment_usage_clean(su)) if (!nilfs_segment_usage_clean(su))
continue; continue;
@ -1167,9 +1172,10 @@ int nilfs_sufile_trim_fs(struct inode *sufile, struct fstrim_range *range)
} }
ndiscarded += nblocks; ndiscarded += nblocks;
kaddr = kmap_local_page(su_bh->b_page); offset = nilfs_sufile_segment_usage_offset(
su = nilfs_sufile_block_get_segment_usage( sufile, segnum, su_bh);
sufile, segnum, su_bh, kaddr); su = kaddr = kmap_local_folio(su_bh->b_folio,
offset);
} }
/* start new extent */ /* start new extent */
@ -1221,7 +1227,6 @@ int nilfs_sufile_read(struct super_block *sb, size_t susize,
struct nilfs_sufile_info *sui; struct nilfs_sufile_info *sui;
struct buffer_head *header_bh; struct buffer_head *header_bh;
struct nilfs_sufile_header *header; struct nilfs_sufile_header *header;
void *kaddr;
int err; int err;
if (susize > sb->s_blocksize) { if (susize > sb->s_blocksize) {
@ -1262,10 +1267,9 @@ int nilfs_sufile_read(struct super_block *sb, size_t susize,
} }
sui = NILFS_SUI(sufile); sui = NILFS_SUI(sufile);
kaddr = kmap_local_page(header_bh->b_page); header = kmap_local_folio(header_bh->b_folio, 0);
header = kaddr + bh_offset(header_bh);
sui->ncleansegs = le64_to_cpu(header->sh_ncleansegs); sui->ncleansegs = le64_to_cpu(header->sh_ncleansegs);
kunmap_local(kaddr); kunmap_local(header);
brelse(header_bh); brelse(header_bh);
sui->allocmax = nilfs_sufile_get_nsegments(sufile) - 1; sui->allocmax = nilfs_sufile_get_nsegments(sufile) - 1;

View File

@ -4767,7 +4767,7 @@ bail:
} }
/* /*
* Allcate and add clusters into the extent b-tree. * Allocate and add clusters into the extent b-tree.
* The new clusters(clusters_to_add) will be inserted at logical_offset. * The new clusters(clusters_to_add) will be inserted at logical_offset.
* The extent b-tree's root is specified by et, and * The extent b-tree's root is specified by et, and
* it is not limited to the file storage. Any extent tree can use this * it is not limited to the file storage. Any extent tree can use this

View File

@ -70,6 +70,8 @@ enum ocfs2_iocb_lock_bits {
OCFS2_IOCB_NUM_LOCKS OCFS2_IOCB_NUM_LOCKS
}; };
#define ocfs2_iocb_init_rw_locked(iocb) \
(iocb->private = NULL)
#define ocfs2_iocb_clear_rw_locked(iocb) \ #define ocfs2_iocb_clear_rw_locked(iocb) \
clear_bit(OCFS2_IOCB_RW_LOCK, (unsigned long *)&iocb->private) clear_bit(OCFS2_IOCB_RW_LOCK, (unsigned long *)&iocb->private)
#define ocfs2_iocb_rw_locked_level(iocb) \ #define ocfs2_iocb_rw_locked_level(iocb) \

View File

@ -60,7 +60,7 @@ static void o2quo_fence_self(void)
switch (o2nm_single_cluster->cl_fence_method) { switch (o2nm_single_cluster->cl_fence_method) {
case O2NM_FENCE_PANIC: case O2NM_FENCE_PANIC:
panic("*** ocfs2 is very sorry to be fencing this system by " panic("*** ocfs2 is very sorry to be fencing this system by "
"panicing ***\n"); "panicking ***\n");
break; break;
default: default:
WARN_ON(o2nm_single_cluster->cl_fence_method >= WARN_ON(o2nm_single_cluster->cl_fence_method >=

View File

@ -62,8 +62,6 @@ enum dlm_status {
DLM_MAXSTATS, /* 41: upper limit for return code validation */ DLM_MAXSTATS, /* 41: upper limit for return code validation */
}; };
/* for pretty-printing dlm_status error messages */
const char *dlm_errmsg(enum dlm_status err);
/* for pretty-printing dlm_status error names */ /* for pretty-printing dlm_status error names */
const char *dlm_errname(enum dlm_status err); const char *dlm_errname(enum dlm_status err);

View File

@ -164,59 +164,6 @@ static const char *dlm_errnames[] = {
[DLM_MAXSTATS] = "DLM_MAXSTATS", [DLM_MAXSTATS] = "DLM_MAXSTATS",
}; };
static const char *dlm_errmsgs[] = {
[DLM_NORMAL] = "request in progress",
[DLM_GRANTED] = "request granted",
[DLM_DENIED] = "request denied",
[DLM_DENIED_NOLOCKS] = "request denied, out of system resources",
[DLM_WORKING] = "async request in progress",
[DLM_BLOCKED] = "lock request blocked",
[DLM_BLOCKED_ORPHAN] = "lock request blocked by a orphan lock",
[DLM_DENIED_GRACE_PERIOD] = "topological change in progress",
[DLM_SYSERR] = "system error",
[DLM_NOSUPPORT] = "unsupported",
[DLM_CANCELGRANT] = "can't cancel convert: already granted",
[DLM_IVLOCKID] = "bad lockid",
[DLM_SYNC] = "synchronous request granted",
[DLM_BADTYPE] = "bad resource type",
[DLM_BADRESOURCE] = "bad resource handle",
[DLM_MAXHANDLES] = "no more resource handles",
[DLM_NOCLINFO] = "can't contact cluster manager",
[DLM_NOLOCKMGR] = "can't contact lock manager",
[DLM_NOPURGED] = "can't contact purge daemon",
[DLM_BADARGS] = "bad api args",
[DLM_VOID] = "no status",
[DLM_NOTQUEUED] = "NOQUEUE was specified and request failed",
[DLM_IVBUFLEN] = "invalid resource name length",
[DLM_CVTUNGRANT] = "attempted to convert ungranted lock",
[DLM_BADPARAM] = "invalid lock mode specified",
[DLM_VALNOTVALID] = "value block has been invalidated",
[DLM_REJECTED] = "request rejected, unrecognized client",
[DLM_ABORT] = "blocked lock request cancelled",
[DLM_CANCEL] = "conversion request cancelled",
[DLM_IVRESHANDLE] = "invalid resource handle",
[DLM_DEADLOCK] = "deadlock recovery refused this request",
[DLM_DENIED_NOASTS] = "failed to allocate AST",
[DLM_FORWARD] = "request must wait for primary's response",
[DLM_TIMEOUT] = "timeout value for lock has expired",
[DLM_IVGROUPID] = "invalid group specification",
[DLM_VERS_CONFLICT] = "version conflicts prevent request handling",
[DLM_BAD_DEVICE_PATH] = "Locks device does not exist or path wrong",
[DLM_NO_DEVICE_PERMISSION] = "Client has insufficient perms for device",
[DLM_NO_CONTROL_DEVICE] = "Cannot set options on opened device ",
[DLM_RECOVERING] = "lock resource being recovered",
[DLM_MIGRATING] = "lock resource being migrated",
[DLM_MAXSTATS] = "invalid error number",
};
const char *dlm_errmsg(enum dlm_status err)
{
if (err >= DLM_MAXSTATS || err < 0)
return dlm_errmsgs[DLM_MAXSTATS];
return dlm_errmsgs[err];
}
EXPORT_SYMBOL_GPL(dlm_errmsg);
const char *dlm_errname(enum dlm_status err) const char *dlm_errname(enum dlm_status err)
{ {
if (err >= DLM_MAXSTATS || err < 0) if (err >= DLM_MAXSTATS || err < 0)

View File

@ -2398,6 +2398,8 @@ static ssize_t ocfs2_file_write_iter(struct kiocb *iocb,
} else } else
inode_lock(inode); inode_lock(inode);
ocfs2_iocb_init_rw_locked(iocb);
/* /*
* Concurrent O_DIRECT writes are allowed with * Concurrent O_DIRECT writes are allowed with
* mount_option "coherency=buffered". * mount_option "coherency=buffered".
@ -2544,6 +2546,8 @@ static ssize_t ocfs2_file_read_iter(struct kiocb *iocb,
if (!direct_io && nowait) if (!direct_io && nowait)
return -EOPNOTSUPP; return -EOPNOTSUPP;
ocfs2_iocb_init_rw_locked(iocb);
/* /*
* buffered reads protect themselves in ->read_folio(). O_DIRECT reads * buffered reads protect themselves in ->read_folio(). O_DIRECT reads
* need locks to protect pending reads from racing with truncate. * need locks to protect pending reads from racing with truncate.

View File

@ -97,7 +97,6 @@ ssize_t ocfs2_quota_write(struct super_block *sb, int type,
const char *data, size_t len, loff_t off); const char *data, size_t len, loff_t off);
int ocfs2_global_read_info(struct super_block *sb, int type); int ocfs2_global_read_info(struct super_block *sb, int type);
int ocfs2_global_write_info(struct super_block *sb, int type); int ocfs2_global_write_info(struct super_block *sb, int type);
int ocfs2_global_read_dquot(struct dquot *dquot);
int __ocfs2_sync_dquot(struct dquot *dquot, int freeing); int __ocfs2_sync_dquot(struct dquot *dquot, int freeing);
static inline int ocfs2_sync_dquot(struct dquot *dquot) static inline int ocfs2_sync_dquot(struct dquot *dquot)
{ {

View File

@ -109,7 +109,7 @@ void proc_task_name(struct seq_file *m, struct task_struct *p, bool escape)
else if (p->flags & PF_KTHREAD) else if (p->flags & PF_KTHREAD)
get_kthread_comm(tcomm, sizeof(tcomm), p); get_kthread_comm(tcomm, sizeof(tcomm), p);
else else
__get_task_comm(tcomm, sizeof(tcomm), p); get_task_comm(tcomm, p);
if (escape) if (escape)
seq_escape_str(m, tcomm, ESCAPE_SPACE | ESCAPE_SPECIAL, "\n\\"); seq_escape_str(m, tcomm, ESCAPE_SPACE | ESCAPE_SPECIAL, "\n\\");

View File

@ -493,13 +493,13 @@ static ssize_t read_kcore_iter(struct kiocb *iocb, struct iov_iter *iter)
* the previous entry, search for a matching entry. * the previous entry, search for a matching entry.
*/ */
if (!m || start < m->addr || start >= m->addr + m->size) { if (!m || start < m->addr || start >= m->addr + m->size) {
struct kcore_list *iter; struct kcore_list *pos;
m = NULL; m = NULL;
list_for_each_entry(iter, &kclist_head, list) { list_for_each_entry(pos, &kclist_head, list) {
if (start >= iter->addr && if (start >= pos->addr &&
start < iter->addr + iter->size) { start < pos->addr + pos->size) {
m = iter; m = pos;
break; break;
} }
} }

View File

@ -15,7 +15,7 @@
/* ACPICA external files should not include ACPICA headers directly. */ /* ACPICA external files should not include ACPICA headers directly. */
#if !defined(BUILDING_ACPICA) && !defined(_LINUX_ACPI_H) #if !defined(BUILDING_ACPICA) && !defined(_LINUX_ACPI_H)
#error "Please don't include <acpi/acpi.h> directly, include <linux/acpi.h> instead." #error "Please do not include <acpi/acpi.h> directly, include <linux/acpi.h> instead."
#endif #endif
#endif #endif

View File

@ -1,6 +1,6 @@
/* SPDX-License-Identifier: GPL-2.0 */ /* SPDX-License-Identifier: GPL-2.0 */
#ifndef __LINUX_COMPILER_TYPES_H #ifndef __LINUX_COMPILER_TYPES_H
#error "Please don't include <linux/compiler-clang.h> directly, include <linux/compiler.h> instead." #error "Please do not include <linux/compiler-clang.h> directly, include <linux/compiler.h> instead."
#endif #endif
/* Compiler specific definitions for Clang compiler */ /* Compiler specific definitions for Clang compiler */

View File

@ -1,6 +1,6 @@
/* SPDX-License-Identifier: GPL-2.0 */ /* SPDX-License-Identifier: GPL-2.0 */
#ifndef __LINUX_COMPILER_TYPES_H #ifndef __LINUX_COMPILER_TYPES_H
#error "Please don't include <linux/compiler-gcc.h> directly, include <linux/compiler.h> instead." #error "Please do not include <linux/compiler-gcc.h> directly, include <linux/compiler.h> instead."
#endif #endif
/* /*

View File

@ -38,9 +38,176 @@ struct min_heap_callbacks {
void (*swp)(void *lhs, void *rhs, void *args); void (*swp)(void *lhs, void *rhs, void *args);
}; };
/**
* is_aligned - is this pointer & size okay for word-wide copying?
* @base: pointer to data
* @size: size of each element
* @align: required alignment (typically 4 or 8)
*
* Returns true if elements can be copied using word loads and stores.
* The size must be a multiple of the alignment, and the base address must
* be if we do not have CONFIG_HAVE_EFFICIENT_UNALIGNED_ACCESS.
*
* For some reason, gcc doesn't know to optimize "if (a & mask || b & mask)"
* to "if ((a | b) & mask)", so we do that by hand.
*/
__attribute_const__ __always_inline
static bool is_aligned(const void *base, size_t size, unsigned char align)
{
unsigned char lsbits = (unsigned char)size;
(void)base;
#ifndef CONFIG_HAVE_EFFICIENT_UNALIGNED_ACCESS
lsbits |= (unsigned char)(uintptr_t)base;
#endif
return (lsbits & (align - 1)) == 0;
}
/**
* swap_words_32 - swap two elements in 32-bit chunks
* @a: pointer to the first element to swap
* @b: pointer to the second element to swap
* @n: element size (must be a multiple of 4)
*
* Exchange the two objects in memory. This exploits base+index addressing,
* which basically all CPUs have, to minimize loop overhead computations.
*
* For some reason, on x86 gcc 7.3.0 adds a redundant test of n at the
* bottom of the loop, even though the zero flag is still valid from the
* subtract (since the intervening mov instructions don't alter the flags).
* Gcc 8.1.0 doesn't have that problem.
*/
static __always_inline
void swap_words_32(void *a, void *b, size_t n)
{
do {
u32 t = *(u32 *)(a + (n -= 4));
*(u32 *)(a + n) = *(u32 *)(b + n);
*(u32 *)(b + n) = t;
} while (n);
}
/**
* swap_words_64 - swap two elements in 64-bit chunks
* @a: pointer to the first element to swap
* @b: pointer to the second element to swap
* @n: element size (must be a multiple of 8)
*
* Exchange the two objects in memory. This exploits base+index
* addressing, which basically all CPUs have, to minimize loop overhead
* computations.
*
* We'd like to use 64-bit loads if possible. If they're not, emulating
* one requires base+index+4 addressing which x86 has but most other
* processors do not. If CONFIG_64BIT, we definitely have 64-bit loads,
* but it's possible to have 64-bit loads without 64-bit pointers (e.g.
* x32 ABI). Are there any cases the kernel needs to worry about?
*/
static __always_inline
void swap_words_64(void *a, void *b, size_t n)
{
do {
#ifdef CONFIG_64BIT
u64 t = *(u64 *)(a + (n -= 8));
*(u64 *)(a + n) = *(u64 *)(b + n);
*(u64 *)(b + n) = t;
#else
/* Use two 32-bit transfers to avoid base+index+4 addressing */
u32 t = *(u32 *)(a + (n -= 4));
*(u32 *)(a + n) = *(u32 *)(b + n);
*(u32 *)(b + n) = t;
t = *(u32 *)(a + (n -= 4));
*(u32 *)(a + n) = *(u32 *)(b + n);
*(u32 *)(b + n) = t;
#endif
} while (n);
}
/**
* swap_bytes - swap two elements a byte at a time
* @a: pointer to the first element to swap
* @b: pointer to the second element to swap
* @n: element size
*
* This is the fallback if alignment doesn't allow using larger chunks.
*/
static __always_inline
void swap_bytes(void *a, void *b, size_t n)
{
do {
char t = ((char *)a)[--n];
((char *)a)[n] = ((char *)b)[n];
((char *)b)[n] = t;
} while (n);
}
/*
* The values are arbitrary as long as they can't be confused with
* a pointer, but small integers make for the smallest compare
* instructions.
*/
#define SWAP_WORDS_64 ((void (*)(void *, void *, void *))0)
#define SWAP_WORDS_32 ((void (*)(void *, void *, void *))1)
#define SWAP_BYTES ((void (*)(void *, void *, void *))2)
/*
* Selects the appropriate swap function based on the element size.
*/
static __always_inline
void *select_swap_func(const void *base, size_t size)
{
if (is_aligned(base, size, 8))
return SWAP_WORDS_64;
else if (is_aligned(base, size, 4))
return SWAP_WORDS_32;
else
return SWAP_BYTES;
}
static __always_inline
void do_swap(void *a, void *b, size_t size, void (*swap_func)(void *lhs, void *rhs, void *args),
void *priv)
{
if (swap_func == SWAP_WORDS_64)
swap_words_64(a, b, size);
else if (swap_func == SWAP_WORDS_32)
swap_words_32(a, b, size);
else if (swap_func == SWAP_BYTES)
swap_bytes(a, b, size);
else
swap_func(a, b, priv);
}
/**
* parent - given the offset of the child, find the offset of the parent.
* @i: the offset of the heap element whose parent is sought. Non-zero.
* @lsbit: a precomputed 1-bit mask, equal to "size & -size"
* @size: size of each element
*
* In terms of array indexes, the parent of element j = @i/@size is simply
* (j-1)/2. But when working in byte offsets, we can't use implicit
* truncation of integer divides.
*
* Fortunately, we only need one bit of the quotient, not the full divide.
* @size has a least significant bit. That bit will be clear if @i is
* an even multiple of @size, and set if it's an odd multiple.
*
* Logically, we're doing "if (i & lsbit) i -= size;", but since the
* branch is unpredictable, it's done with a bit of clever branch-free
* code instead.
*/
__attribute_const__ __always_inline
static size_t parent(size_t i, unsigned int lsbit, size_t size)
{
i -= size;
i -= size & -(i & lsbit);
return i / 2;
}
/* Initialize a min-heap. */ /* Initialize a min-heap. */
static __always_inline static __always_inline
void __min_heap_init(min_heap_char *heap, void *data, int size) void __min_heap_init_inline(min_heap_char *heap, void *data, int size)
{ {
heap->nr = 0; heap->nr = 0;
heap->size = size; heap->size = size;
@ -50,104 +217,113 @@ void __min_heap_init(min_heap_char *heap, void *data, int size)
heap->data = heap->preallocated; heap->data = heap->preallocated;
} }
#define min_heap_init(_heap, _data, _size) \ #define min_heap_init_inline(_heap, _data, _size) \
__min_heap_init((min_heap_char *)_heap, _data, _size) __min_heap_init_inline((min_heap_char *)_heap, _data, _size)
/* Get the minimum element from the heap. */ /* Get the minimum element from the heap. */
static __always_inline static __always_inline
void *__min_heap_peek(struct min_heap_char *heap) void *__min_heap_peek_inline(struct min_heap_char *heap)
{ {
return heap->nr ? heap->data : NULL; return heap->nr ? heap->data : NULL;
} }
#define min_heap_peek(_heap) \ #define min_heap_peek_inline(_heap) \
(__minheap_cast(_heap) __min_heap_peek((min_heap_char *)_heap)) (__minheap_cast(_heap) __min_heap_peek_inline((min_heap_char *)_heap))
/* Check if the heap is full. */ /* Check if the heap is full. */
static __always_inline static __always_inline
bool __min_heap_full(min_heap_char *heap) bool __min_heap_full_inline(min_heap_char *heap)
{ {
return heap->nr == heap->size; return heap->nr == heap->size;
} }
#define min_heap_full(_heap) \ #define min_heap_full_inline(_heap) \
__min_heap_full((min_heap_char *)_heap) __min_heap_full_inline((min_heap_char *)_heap)
/* Sift the element at pos down the heap. */ /* Sift the element at pos down the heap. */
static __always_inline static __always_inline
void __min_heap_sift_down(min_heap_char *heap, int pos, size_t elem_size, void __min_heap_sift_down_inline(min_heap_char *heap, int pos, size_t elem_size,
const struct min_heap_callbacks *func, void *args) const struct min_heap_callbacks *func, void *args)
{ {
void *left, *right; const unsigned long lsbit = elem_size & -elem_size;
void *data = heap->data; void *data = heap->data;
void *root = data + pos * elem_size; void (*swp)(void *lhs, void *rhs, void *args) = func->swp;
int i = pos, j; /* pre-scale counters for performance */
size_t a = pos * elem_size;
size_t b, c, d;
size_t n = heap->nr * elem_size;
if (!swp)
swp = select_swap_func(data, elem_size);
/* Find the sift-down path all the way to the leaves. */ /* Find the sift-down path all the way to the leaves. */
for (;;) { for (b = a; c = 2 * b + elem_size, (d = c + elem_size) < n;)
if (i * 2 + 2 >= heap->nr) b = func->less(data + c, data + d, args) ? c : d;
break;
left = data + (i * 2 + 1) * elem_size;
right = data + (i * 2 + 2) * elem_size;
i = func->less(left, right, args) ? i * 2 + 1 : i * 2 + 2;
}
/* Special case for the last leaf with no sibling. */ /* Special case for the last leaf with no sibling. */
if (i * 2 + 2 == heap->nr) if (d == n)
i = i * 2 + 1; b = c;
/* Backtrack to the correct location. */ /* Backtrack to the correct location. */
while (i != pos && func->less(root, data + i * elem_size, args)) while (b != a && func->less(data + a, data + b, args))
i = (i - 1) / 2; b = parent(b, lsbit, elem_size);
/* Shift the element into its correct place. */ /* Shift the element into its correct place. */
j = i; c = b;
while (i != pos) { while (b != a) {
i = (i - 1) / 2; b = parent(b, lsbit, elem_size);
func->swp(data + i * elem_size, data + j * elem_size, args); do_swap(data + b, data + c, elem_size, swp, args);
} }
} }
#define min_heap_sift_down(_heap, _pos, _func, _args) \ #define min_heap_sift_down_inline(_heap, _pos, _func, _args) \
__min_heap_sift_down((min_heap_char *)_heap, _pos, __minheap_obj_size(_heap), _func, _args) __min_heap_sift_down_inline((min_heap_char *)_heap, _pos, __minheap_obj_size(_heap), \
_func, _args)
/* Sift up ith element from the heap, O(log2(nr)). */ /* Sift up ith element from the heap, O(log2(nr)). */
static __always_inline static __always_inline
void __min_heap_sift_up(min_heap_char *heap, size_t elem_size, size_t idx, void __min_heap_sift_up_inline(min_heap_char *heap, size_t elem_size, size_t idx,
const struct min_heap_callbacks *func, void *args) const struct min_heap_callbacks *func, void *args)
{ {
const unsigned long lsbit = elem_size & -elem_size;
void *data = heap->data; void *data = heap->data;
size_t parent; void (*swp)(void *lhs, void *rhs, void *args) = func->swp;
/* pre-scale counters for performance */
size_t a = idx * elem_size, b;
while (idx) { if (!swp)
parent = (idx - 1) / 2; swp = select_swap_func(data, elem_size);
if (func->less(data + parent * elem_size, data + idx * elem_size, args))
while (a) {
b = parent(a, lsbit, elem_size);
if (func->less(data + b, data + a, args))
break; break;
func->swp(data + parent * elem_size, data + idx * elem_size, args); do_swap(data + a, data + b, elem_size, swp, args);
idx = parent; a = b;
} }
} }
#define min_heap_sift_up(_heap, _idx, _func, _args) \ #define min_heap_sift_up_inline(_heap, _idx, _func, _args) \
__min_heap_sift_up((min_heap_char *)_heap, __minheap_obj_size(_heap), _idx, _func, _args) __min_heap_sift_up_inline((min_heap_char *)_heap, __minheap_obj_size(_heap), _idx, \
_func, _args)
/* Floyd's approach to heapification that is O(nr). */ /* Floyd's approach to heapification that is O(nr). */
static __always_inline static __always_inline
void __min_heapify_all(min_heap_char *heap, size_t elem_size, void __min_heapify_all_inline(min_heap_char *heap, size_t elem_size,
const struct min_heap_callbacks *func, void *args) const struct min_heap_callbacks *func, void *args)
{ {
int i; int i;
for (i = heap->nr / 2 - 1; i >= 0; i--) for (i = heap->nr / 2 - 1; i >= 0; i--)
__min_heap_sift_down(heap, i, elem_size, func, args); __min_heap_sift_down_inline(heap, i, elem_size, func, args);
} }
#define min_heapify_all(_heap, _func, _args) \ #define min_heapify_all_inline(_heap, _func, _args) \
__min_heapify_all((min_heap_char *)_heap, __minheap_obj_size(_heap), _func, _args) __min_heapify_all_inline((min_heap_char *)_heap, __minheap_obj_size(_heap), _func, _args)
/* Remove minimum element from the heap, O(log2(nr)). */ /* Remove minimum element from the heap, O(log2(nr)). */
static __always_inline static __always_inline
bool __min_heap_pop(min_heap_char *heap, size_t elem_size, bool __min_heap_pop_inline(min_heap_char *heap, size_t elem_size,
const struct min_heap_callbacks *func, void *args) const struct min_heap_callbacks *func, void *args)
{ {
void *data = heap->data; void *data = heap->data;
@ -158,13 +334,13 @@ bool __min_heap_pop(min_heap_char *heap, size_t elem_size,
/* Place last element at the root (position 0) and then sift down. */ /* Place last element at the root (position 0) and then sift down. */
heap->nr--; heap->nr--;
memcpy(data, data + (heap->nr * elem_size), elem_size); memcpy(data, data + (heap->nr * elem_size), elem_size);
__min_heap_sift_down(heap, 0, elem_size, func, args); __min_heap_sift_down_inline(heap, 0, elem_size, func, args);
return true; return true;
} }
#define min_heap_pop(_heap, _func, _args) \ #define min_heap_pop_inline(_heap, _func, _args) \
__min_heap_pop((min_heap_char *)_heap, __minheap_obj_size(_heap), _func, _args) __min_heap_pop_inline((min_heap_char *)_heap, __minheap_obj_size(_heap), _func, _args)
/* /*
* Remove the minimum element and then push the given element. The * Remove the minimum element and then push the given element. The
@ -172,21 +348,20 @@ bool __min_heap_pop(min_heap_char *heap, size_t elem_size,
* efficient than a pop followed by a push that does 2. * efficient than a pop followed by a push that does 2.
*/ */
static __always_inline static __always_inline
void __min_heap_pop_push(min_heap_char *heap, void __min_heap_pop_push_inline(min_heap_char *heap, const void *element, size_t elem_size,
const void *element, size_t elem_size, const struct min_heap_callbacks *func, void *args)
const struct min_heap_callbacks *func,
void *args)
{ {
memcpy(heap->data, element, elem_size); memcpy(heap->data, element, elem_size);
__min_heap_sift_down(heap, 0, elem_size, func, args); __min_heap_sift_down_inline(heap, 0, elem_size, func, args);
} }
#define min_heap_pop_push(_heap, _element, _func, _args) \ #define min_heap_pop_push_inline(_heap, _element, _func, _args) \
__min_heap_pop_push((min_heap_char *)_heap, _element, __minheap_obj_size(_heap), _func, _args) __min_heap_pop_push_inline((min_heap_char *)_heap, _element, __minheap_obj_size(_heap), \
_func, _args)
/* Push an element on to the heap, O(log2(nr)). */ /* Push an element on to the heap, O(log2(nr)). */
static __always_inline static __always_inline
bool __min_heap_push(min_heap_char *heap, const void *element, size_t elem_size, bool __min_heap_push_inline(min_heap_char *heap, const void *element, size_t elem_size,
const struct min_heap_callbacks *func, void *args) const struct min_heap_callbacks *func, void *args)
{ {
void *data = heap->data; void *data = heap->data;
@ -201,35 +376,81 @@ bool __min_heap_push(min_heap_char *heap, const void *element, size_t elem_size,
heap->nr++; heap->nr++;
/* Sift child at pos up. */ /* Sift child at pos up. */
__min_heap_sift_up(heap, elem_size, pos, func, args); __min_heap_sift_up_inline(heap, elem_size, pos, func, args);
return true; return true;
} }
#define min_heap_push(_heap, _element, _func, _args) \ #define min_heap_push_inline(_heap, _element, _func, _args) \
__min_heap_push((min_heap_char *)_heap, _element, __minheap_obj_size(_heap), _func, _args) __min_heap_push_inline((min_heap_char *)_heap, _element, __minheap_obj_size(_heap), \
_func, _args)
/* Remove ith element from the heap, O(log2(nr)). */ /* Remove ith element from the heap, O(log2(nr)). */
static __always_inline static __always_inline
bool __min_heap_del(min_heap_char *heap, size_t elem_size, size_t idx, bool __min_heap_del_inline(min_heap_char *heap, size_t elem_size, size_t idx,
const struct min_heap_callbacks *func, void *args) const struct min_heap_callbacks *func, void *args)
{ {
void *data = heap->data; void *data = heap->data;
void (*swp)(void *lhs, void *rhs, void *args) = func->swp;
if (WARN_ONCE(heap->nr <= 0, "Popping an empty heap")) if (WARN_ONCE(heap->nr <= 0, "Popping an empty heap"))
return false; return false;
if (!swp)
swp = select_swap_func(data, elem_size);
/* Place last element at the root (position 0) and then sift down. */ /* Place last element at the root (position 0) and then sift down. */
heap->nr--; heap->nr--;
if (idx == heap->nr) if (idx == heap->nr)
return true; return true;
func->swp(data + (idx * elem_size), data + (heap->nr * elem_size), args); do_swap(data + (idx * elem_size), data + (heap->nr * elem_size), elem_size, swp, args);
__min_heap_sift_up(heap, elem_size, idx, func, args); __min_heap_sift_up_inline(heap, elem_size, idx, func, args);
__min_heap_sift_down(heap, idx, elem_size, func, args); __min_heap_sift_down_inline(heap, idx, elem_size, func, args);
return true; return true;
} }
#define min_heap_del_inline(_heap, _idx, _func, _args) \
__min_heap_del_inline((min_heap_char *)_heap, __minheap_obj_size(_heap), _idx, \
_func, _args)
void __min_heap_init(min_heap_char *heap, void *data, int size);
void *__min_heap_peek(struct min_heap_char *heap);
bool __min_heap_full(min_heap_char *heap);
void __min_heap_sift_down(min_heap_char *heap, int pos, size_t elem_size,
const struct min_heap_callbacks *func, void *args);
void __min_heap_sift_up(min_heap_char *heap, size_t elem_size, size_t idx,
const struct min_heap_callbacks *func, void *args);
void __min_heapify_all(min_heap_char *heap, size_t elem_size,
const struct min_heap_callbacks *func, void *args);
bool __min_heap_pop(min_heap_char *heap, size_t elem_size,
const struct min_heap_callbacks *func, void *args);
void __min_heap_pop_push(min_heap_char *heap, const void *element, size_t elem_size,
const struct min_heap_callbacks *func, void *args);
bool __min_heap_push(min_heap_char *heap, const void *element, size_t elem_size,
const struct min_heap_callbacks *func, void *args);
bool __min_heap_del(min_heap_char *heap, size_t elem_size, size_t idx,
const struct min_heap_callbacks *func, void *args);
#define min_heap_init(_heap, _data, _size) \
__min_heap_init((min_heap_char *)_heap, _data, _size)
#define min_heap_peek(_heap) \
(__minheap_cast(_heap) __min_heap_peek((min_heap_char *)_heap))
#define min_heap_full(_heap) \
__min_heap_full((min_heap_char *)_heap)
#define min_heap_sift_down(_heap, _pos, _func, _args) \
__min_heap_sift_down((min_heap_char *)_heap, _pos, __minheap_obj_size(_heap), _func, _args)
#define min_heap_sift_up(_heap, _idx, _func, _args) \
__min_heap_sift_up((min_heap_char *)_heap, __minheap_obj_size(_heap), _idx, _func, _args)
#define min_heapify_all(_heap, _func, _args) \
__min_heapify_all((min_heap_char *)_heap, __minheap_obj_size(_heap), _func, _args)
#define min_heap_pop(_heap, _func, _args) \
__min_heap_pop((min_heap_char *)_heap, __minheap_obj_size(_heap), _func, _args)
#define min_heap_pop_push(_heap, _element, _func, _args) \
__min_heap_pop_push((min_heap_char *)_heap, _element, __minheap_obj_size(_heap), \
_func, _args)
#define min_heap_push(_heap, _element, _func, _args) \
__min_heap_push((min_heap_char *)_heap, _element, __minheap_obj_size(_heap), _func, _args)
#define min_heap_del(_heap, _idx, _func, _args) \ #define min_heap_del(_heap, _idx, _func, _args) \
__min_heap_del((min_heap_char *)_heap, __minheap_obj_size(_heap), _idx, _func, _args) __min_heap_del((min_heap_char *)_heap, __minheap_obj_size(_heap), _idx, _func, _args)

View File

@ -237,7 +237,5 @@ static inline int notifier_to_errno(int ret)
#define KBD_KEYSYM 0x0004 /* Keyboard keysym */ #define KBD_KEYSYM 0x0004 /* Keyboard keysym */
#define KBD_POST_KEYSYM 0x0005 /* Called after keyboard keysym interpretation */ #define KBD_POST_KEYSYM 0x0005 /* Called after keyboard keysym interpretation */
extern struct blocking_notifier_head reboot_notifier_list;
#endif /* __KERNEL__ */ #endif /* __KERNEL__ */
#endif /* _LINUX_NOTIFIER_H */ #endif /* _LINUX_NOTIFIER_H */

View File

@ -220,15 +220,20 @@ do { \
(void)__vpp_verify; \ (void)__vpp_verify; \
} while (0) } while (0)
#define PERCPU_PTR(__p) \
({ \
unsigned long __pcpu_ptr = (__force unsigned long)(__p); \
(typeof(*(__p)) __force __kernel *)(__pcpu_ptr); \
})
#ifdef CONFIG_SMP #ifdef CONFIG_SMP
/* /*
* Add an offset to a pointer but keep the pointer as-is. Use RELOC_HIDE() * Add an offset to a pointer. Use RELOC_HIDE() to prevent the compiler
* to prevent the compiler from making incorrect assumptions about the * from making incorrect assumptions about the pointer value.
* pointer value. The weird cast keeps both GCC and sparse happy.
*/ */
#define SHIFT_PERCPU_PTR(__p, __offset) \ #define SHIFT_PERCPU_PTR(__p, __offset) \
RELOC_HIDE((typeof(*(__p)) __kernel __force *)(__p), (__offset)) RELOC_HIDE(PERCPU_PTR(__p), (__offset))
#define per_cpu_ptr(ptr, cpu) \ #define per_cpu_ptr(ptr, cpu) \
({ \ ({ \
@ -254,13 +259,13 @@ do { \
#else /* CONFIG_SMP */ #else /* CONFIG_SMP */
#define VERIFY_PERCPU_PTR(__p) \ #define per_cpu_ptr(ptr, cpu) \
({ \ ({ \
__verify_pcpu_ptr(__p); \ (void)(cpu); \
(typeof(*(__p)) __kernel __force *)(__p); \ __verify_pcpu_ptr(ptr); \
PERCPU_PTR(ptr); \
}) })
#define per_cpu_ptr(ptr, cpu) ({ (void)(cpu); VERIFY_PERCPU_PTR(ptr); })
#define raw_cpu_ptr(ptr) per_cpu_ptr(ptr, 0) #define raw_cpu_ptr(ptr) per_cpu_ptr(ptr, 0)
#define this_cpu_ptr(ptr) raw_cpu_ptr(ptr) #define this_cpu_ptr(ptr) raw_cpu_ptr(ptr)

View File

@ -10,7 +10,7 @@
#define _LINUX_PM_WAKEUP_H #define _LINUX_PM_WAKEUP_H
#ifndef _DEVICE_H_ #ifndef _DEVICE_H_
# error "please don't include this file directly" # error "Please do not include this file directly."
#endif #endif
#include <linux/types.h> #include <linux/types.h>

View File

@ -2,7 +2,7 @@
#define __LINUX_RWLOCK_H #define __LINUX_RWLOCK_H
#ifndef __LINUX_INSIDE_SPINLOCK_H #ifndef __LINUX_INSIDE_SPINLOCK_H
# error "please don't include this file directly" # error "Please do not include this file directly."
#endif #endif
/* /*

View File

@ -2,7 +2,7 @@
#define __LINUX_RWLOCK_API_SMP_H #define __LINUX_RWLOCK_API_SMP_H
#ifndef __LINUX_SPINLOCK_API_SMP_H #ifndef __LINUX_SPINLOCK_API_SMP_H
# error "please don't include this file directly" # error "Please do not include this file directly."
#endif #endif
/* /*

View File

@ -273,7 +273,7 @@ static inline void sg_unmark_end(struct scatterlist *sg)
} }
/* /*
* One 64-bit architectures there is a 4-byte padding in struct scatterlist * On 64-bit architectures there is a 4-byte padding in struct scatterlist
* (assuming also CONFIG_NEED_SG_DMA_LENGTH is set). Use this padding for DMA * (assuming also CONFIG_NEED_SG_DMA_LENGTH is set). Use this padding for DMA
* flags bits to indicate when a specific dma address is a bus address or the * flags bits to indicate when a specific dma address is a bus address or the
* buffer may have been bounced via SWIOTLB. * buffer may have been bounced via SWIOTLB.

View File

@ -1121,9 +1121,12 @@ struct task_struct {
/* /*
* executable name, excluding path. * executable name, excluding path.
* *
* - normally initialized setup_new_exec() * - normally initialized begin_new_exec()
* - access it with [gs]et_task_comm() * - set it with set_task_comm()
* - lock it with task_lock() * - strscpy_pad() to ensure it is always NUL-terminated and
* zero-padded
* - task_lock() to ensure the operation is atomic and the name is
* fully updated.
*/ */
char comm[TASK_COMM_LEN]; char comm[TASK_COMM_LEN];
@ -1939,10 +1942,23 @@ static inline void set_task_comm(struct task_struct *tsk, const char *from)
__set_task_comm(tsk, from, false); __set_task_comm(tsk, from, false);
} }
extern char *__get_task_comm(char *to, size_t len, struct task_struct *tsk); /*
* - Why not use task_lock()?
* User space can randomly change their names anyway, so locking for readers
* doesn't make sense. For writers, locking is probably necessary, as a race
* condition could lead to long-term mixed results.
* The strscpy_pad() in __set_task_comm() can ensure that the task comm is
* always NUL-terminated and zero-padded. Therefore the race condition between
* reader and writer is not an issue.
*
* - BUILD_BUG_ON() can help prevent the buf from being truncated.
* Since the callers don't perform any return value checks, this safeguard is
* necessary.
*/
#define get_task_comm(buf, tsk) ({ \ #define get_task_comm(buf, tsk) ({ \
BUILD_BUG_ON(sizeof(buf) != TASK_COMM_LEN); \ BUILD_BUG_ON(sizeof(buf) < TASK_COMM_LEN); \
__get_task_comm(buf, sizeof(buf), tsk); \ strscpy_pad(buf, (tsk)->comm); \
buf; \
}) })
#ifdef CONFIG_SMP #ifdef CONFIG_SMP

View File

@ -2,7 +2,7 @@
#define __LINUX_SPINLOCK_API_SMP_H #define __LINUX_SPINLOCK_API_SMP_H
#ifndef __LINUX_INSIDE_SPINLOCK_H #ifndef __LINUX_INSIDE_SPINLOCK_H
# error "please don't include this file directly" # error "Please do not include this file directly."
#endif #endif
/* /*

View File

@ -2,7 +2,7 @@
#define __LINUX_SPINLOCK_TYPES_UP_H #define __LINUX_SPINLOCK_TYPES_UP_H
#ifndef __LINUX_SPINLOCK_TYPES_RAW_H #ifndef __LINUX_SPINLOCK_TYPES_RAW_H
# error "please don't include this file directly" # error "Please do not include this file directly."
#endif #endif
/* /*

View File

@ -2,7 +2,7 @@
#define __LINUX_SPINLOCK_UP_H #define __LINUX_SPINLOCK_UP_H
#ifndef __LINUX_INSIDE_SPINLOCK_H #ifndef __LINUX_INSIDE_SPINLOCK_H
# error "please don't include this file directly" # error "Please do not include this file directly."
#endif #endif
#include <asm/processor.h> /* for cpu_relax() */ #include <asm/processor.h> /* for cpu_relax() */

View File

@ -4,19 +4,6 @@
#include <linux/math.h> #include <linux/math.h>
#define __find_closest(x, a, as, op) \
({ \
typeof(as) __fc_i, __fc_as = (as) - 1; \
typeof(x) __fc_x = (x); \
typeof(*a) const *__fc_a = (a); \
for (__fc_i = 0; __fc_i < __fc_as; __fc_i++) { \
if (__fc_x op DIV_ROUND_CLOSEST(__fc_a[__fc_i] + \
__fc_a[__fc_i + 1], 2)) \
break; \
} \
(__fc_i); \
})
/** /**
* find_closest - locate the closest element in a sorted array * find_closest - locate the closest element in a sorted array
* @x: The reference value. * @x: The reference value.
@ -25,8 +12,27 @@
* @as: Size of 'a'. * @as: Size of 'a'.
* *
* Returns the index of the element closest to 'x'. * Returns the index of the element closest to 'x'.
* Note: If using an array of negative numbers (or mixed positive numbers),
* then be sure that 'x' is of a signed-type to get good results.
*/ */
#define find_closest(x, a, as) __find_closest(x, a, as, <=) #define find_closest(x, a, as) \
({ \
typeof(as) __fc_i, __fc_as = (as) - 1; \
long __fc_mid_x, __fc_x = (x); \
long __fc_left, __fc_right; \
typeof(*a) const *__fc_a = (a); \
for (__fc_i = 0; __fc_i < __fc_as; __fc_i++) { \
__fc_mid_x = (__fc_a[__fc_i] + __fc_a[__fc_i + 1]) / 2; \
if (__fc_x <= __fc_mid_x) { \
__fc_left = __fc_x - __fc_a[__fc_i]; \
__fc_right = __fc_a[__fc_i + 1] - __fc_x; \
if (__fc_right < __fc_left) \
__fc_i++; \
break; \
} \
} \
(__fc_i); \
})
/** /**
* find_closest_descending - locate the closest element in a sorted array * find_closest_descending - locate the closest element in a sorted array
@ -36,9 +42,27 @@
* @as: Size of 'a'. * @as: Size of 'a'.
* *
* Similar to find_closest() but 'a' is expected to be sorted in descending * Similar to find_closest() but 'a' is expected to be sorted in descending
* order. * order. The iteration is done in reverse order, so that the comparison
* of '__fc_right' & '__fc_left' also works for unsigned numbers.
*/ */
#define find_closest_descending(x, a, as) __find_closest(x, a, as, >=) #define find_closest_descending(x, a, as) \
({ \
typeof(as) __fc_i, __fc_as = (as) - 1; \
long __fc_mid_x, __fc_x = (x); \
long __fc_left, __fc_right; \
typeof(*a) const *__fc_a = (a); \
for (__fc_i = __fc_as; __fc_i >= 1; __fc_i--) { \
__fc_mid_x = (__fc_a[__fc_i] + __fc_a[__fc_i - 1]) / 2; \
if (__fc_x <= __fc_mid_x) { \
__fc_left = __fc_x - __fc_a[__fc_i]; \
__fc_right = __fc_a[__fc_i - 1] - __fc_x; \
if (__fc_right < __fc_left) \
__fc_i--; \
break; \
} \
} \
(__fc_i); \
})
/** /**
* is_insidevar - check if the @ptr points inside the @var memory range. * is_insidevar - check if the @ptr points inside the @var memory range.

View File

@ -1166,6 +1166,7 @@ config CGROUP_HUGETLB
config CPUSETS config CPUSETS
bool "Cpuset controller" bool "Cpuset controller"
depends on SMP depends on SMP
select UNION_FIND
help help
This option will let you create and manage CPUSETs which This option will let you create and manage CPUSETs which
allow dynamically partitioning a system into sets of CPUs and allow dynamically partitioning a system into sets of CPUs and

View File

@ -978,7 +978,7 @@ SYSCALL_DEFINE4(msgsnd, int, msqid, struct msgbuf __user *, msgp, size_t, msgsz,
struct compat_msgbuf { struct compat_msgbuf {
compat_long_t mtype; compat_long_t mtype;
char mtext[1]; char mtext[];
}; };
long compat_ksys_msgsnd(int msqid, compat_uptr_t msgp, long compat_ksys_msgsnd(int msqid, compat_uptr_t msgp,

View File

@ -83,13 +83,15 @@ static struct ipc_namespace *create_ipc_ns(struct user_namespace *user_ns,
err = msg_init_ns(ns); err = msg_init_ns(ns);
if (err) if (err)
goto fail_put; goto fail_ipc;
sem_init_ns(ns); sem_init_ns(ns);
shm_init_ns(ns); shm_init_ns(ns);
return ns; return ns;
fail_ipc:
retire_ipc_sysctls(ns);
fail_mq: fail_mq:
retire_mq_sysctls(ns); retire_mq_sysctls(ns);

View File

@ -2729,7 +2729,7 @@ void __audit_ptrace(struct task_struct *t)
context->target_uid = task_uid(t); context->target_uid = task_uid(t);
context->target_sessionid = audit_get_sessionid(t); context->target_sessionid = audit_get_sessionid(t);
security_task_getlsmprop_obj(t, &context->target_ref); security_task_getlsmprop_obj(t, &context->target_ref);
memcpy(context->target_comm, t->comm, TASK_COMM_LEN); strscpy(context->target_comm, t->comm);
} }
/** /**
@ -2756,7 +2756,7 @@ int audit_signal_info_syscall(struct task_struct *t)
ctx->target_uid = t_uid; ctx->target_uid = t_uid;
ctx->target_sessionid = audit_get_sessionid(t); ctx->target_sessionid = audit_get_sessionid(t);
security_task_getlsmprop_obj(t, &ctx->target_ref); security_task_getlsmprop_obj(t, &ctx->target_ref);
memcpy(ctx->target_comm, t->comm, TASK_COMM_LEN); strscpy(ctx->target_comm, t->comm);
return 0; return 0;
} }
@ -2777,7 +2777,7 @@ int audit_signal_info_syscall(struct task_struct *t)
axp->target_uid[axp->pid_count] = t_uid; axp->target_uid[axp->pid_count] = t_uid;
axp->target_sessionid[axp->pid_count] = audit_get_sessionid(t); axp->target_sessionid[axp->pid_count] = audit_get_sessionid(t);
security_task_getlsmprop_obj(t, &axp->target_ref[axp->pid_count]); security_task_getlsmprop_obj(t, &axp->target_ref[axp->pid_count]);
memcpy(axp->target_comm[axp->pid_count], t->comm, TASK_COMM_LEN); strscpy(axp->target_comm[axp->pid_count], t->comm);
axp->pid_count++; axp->pid_count++;
return 0; return 0;

View File

@ -505,6 +505,7 @@ int crash_check_hotplug_support(void)
crash_hotplug_lock(); crash_hotplug_lock();
/* Obtain lock while reading crash information */ /* Obtain lock while reading crash information */
if (!kexec_trylock()) { if (!kexec_trylock()) {
if (!kexec_in_progress)
pr_info("kexec_trylock() failed, kdump image may be inaccurate\n"); pr_info("kexec_trylock() failed, kdump image may be inaccurate\n");
crash_hotplug_unlock(); crash_hotplug_unlock();
return 0; return 0;
@ -547,6 +548,7 @@ static void crash_handle_hotplug_event(unsigned int hp_action, unsigned int cpu,
crash_hotplug_lock(); crash_hotplug_lock();
/* Obtain lock while changing crash information */ /* Obtain lock while changing crash information */
if (!kexec_trylock()) { if (!kexec_trylock()) {
if (!kexec_in_progress)
pr_info("kexec_trylock() failed, kdump image may be inaccurate\n"); pr_info("kexec_trylock() failed, kdump image may be inaccurate\n");
crash_hotplug_unlock(); crash_hotplug_unlock();
return; return;

View File

@ -3778,18 +3778,11 @@ static bool perf_less_group_idx(const void *l, const void *r, void __always_unus
return le->group_index < re->group_index; return le->group_index < re->group_index;
} }
static void swap_ptr(void *l, void *r, void __always_unused *args)
{
void **lp = l, **rp = r;
swap(*lp, *rp);
}
DEFINE_MIN_HEAP(struct perf_event *, perf_event_min_heap); DEFINE_MIN_HEAP(struct perf_event *, perf_event_min_heap);
static const struct min_heap_callbacks perf_min_heap = { static const struct min_heap_callbacks perf_min_heap = {
.less = perf_less_group_idx, .less = perf_less_group_idx,
.swp = swap_ptr, .swp = NULL,
}; };
static void __heap_add(struct perf_event_min_heap *heap, struct perf_event *event) static void __heap_add(struct perf_event_min_heap *heap, struct perf_event *event)
@ -3870,7 +3863,7 @@ static noinline int visit_groups_merge(struct perf_event_context *ctx,
perf_assert_pmu_disabled((*evt)->pmu_ctx->pmu); perf_assert_pmu_disabled((*evt)->pmu_ctx->pmu);
} }
min_heapify_all(&event_heap, &perf_min_heap, NULL); min_heapify_all_inline(&event_heap, &perf_min_heap, NULL);
while (event_heap.nr) { while (event_heap.nr) {
ret = func(*evt, data); ret = func(*evt, data);
@ -3879,9 +3872,9 @@ static noinline int visit_groups_merge(struct perf_event_context *ctx,
*evt = perf_event_groups_next(*evt, pmu); *evt = perf_event_groups_next(*evt, pmu);
if (*evt) if (*evt)
min_heap_sift_down(&event_heap, 0, &perf_min_heap, NULL); min_heap_sift_down_inline(&event_heap, 0, &perf_min_heap, NULL);
else else
min_heap_pop(&event_heap, &perf_min_heap, NULL); min_heap_pop_inline(&event_heap, &perf_min_heap, NULL);
} }
return 0; return 0;

View File

@ -849,7 +849,7 @@ register_wide_hw_breakpoint(struct perf_event_attr *attr,
cpu_events = alloc_percpu(typeof(*cpu_events)); cpu_events = alloc_percpu(typeof(*cpu_events));
if (!cpu_events) if (!cpu_events)
return (void __percpu __force *)ERR_PTR(-ENOMEM); return ERR_PTR_PCPU(-ENOMEM);
cpus_read_lock(); cpus_read_lock();
for_each_online_cpu(cpu) { for_each_online_cpu(cpu) {
@ -868,7 +868,7 @@ register_wide_hw_breakpoint(struct perf_event_attr *attr,
return cpu_events; return cpu_events;
unregister_wide_hw_breakpoint(cpu_events); unregister_wide_hw_breakpoint(cpu_events);
return (void __percpu __force *)ERR_PTR(err); return ERR_PTR_PCPU(err);
} }
EXPORT_SYMBOL_GPL(register_wide_hw_breakpoint); EXPORT_SYMBOL_GPL(register_wide_hw_breakpoint);

View File

@ -30,6 +30,11 @@
*/ */
static int __read_mostly sysctl_hung_task_check_count = PID_MAX_LIMIT; static int __read_mostly sysctl_hung_task_check_count = PID_MAX_LIMIT;
/*
* Total number of tasks detected as hung since boot:
*/
static unsigned long __read_mostly sysctl_hung_task_detect_count;
/* /*
* Limit number of tasks checked in a batch. * Limit number of tasks checked in a batch.
* *
@ -115,6 +120,12 @@ static void check_hung_task(struct task_struct *t, unsigned long timeout)
if (time_is_after_jiffies(t->last_switch_time + timeout * HZ)) if (time_is_after_jiffies(t->last_switch_time + timeout * HZ))
return; return;
/*
* This counter tracks the total number of tasks detected as hung
* since boot.
*/
sysctl_hung_task_detect_count++;
trace_sched_process_hang(t); trace_sched_process_hang(t);
if (sysctl_hung_task_panic) { if (sysctl_hung_task_panic) {
@ -314,6 +325,13 @@ static struct ctl_table hung_task_sysctls[] = {
.proc_handler = proc_dointvec_minmax, .proc_handler = proc_dointvec_minmax,
.extra1 = SYSCTL_NEG_ONE, .extra1 = SYSCTL_NEG_ONE,
}, },
{
.procname = "hung_task_detect_count",
.data = &sysctl_hung_task_detect_count,
.maxlen = sizeof(unsigned long),
.mode = 0444,
.proc_handler = proc_doulongvec_minmax,
},
}; };
static void __init hung_task_sysctl_init(void) static void __init hung_task_sysctl_init(void)

View File

@ -101,7 +101,7 @@ void get_kthread_comm(char *buf, size_t buf_size, struct task_struct *tsk)
struct kthread *kthread = to_kthread(tsk); struct kthread *kthread = to_kthread(tsk);
if (!kthread || !kthread->full_name) { if (!kthread || !kthread->full_name) {
__get_task_comm(buf, buf_size, tsk); strscpy(buf, tsk->comm, buf_size);
return; return;
} }

View File

@ -5,18 +5,10 @@
#include <linux/notifier.h> #include <linux/notifier.h>
#include <linux/rcupdate.h> #include <linux/rcupdate.h>
#include <linux/vmalloc.h> #include <linux/vmalloc.h>
#include <linux/reboot.h>
#define CREATE_TRACE_POINTS #define CREATE_TRACE_POINTS
#include <trace/events/notifier.h> #include <trace/events/notifier.h>
/*
* Notifier list for kernel code which wants to be called
* at shutdown. This is used to stop any idling DMA operations
* and the like.
*/
BLOCKING_NOTIFIER_HEAD(reboot_notifier_list);
/* /*
* Notifier chain core routines. The exported routines below * Notifier chain core routines. The exported routines below
* are layered on top of these, with appropriate locking added. * are layered on top of these, with appropriate locking added.

View File

@ -72,6 +72,13 @@ static bool poweroff_fallback_to_halt;
*/ */
void __weak (*pm_power_off)(void); void __weak (*pm_power_off)(void);
/*
* Notifier list for kernel code which wants to be called
* at shutdown. This is used to stop any idling DMA operations
* and the like.
*/
static BLOCKING_NOTIFIER_HEAD(reboot_notifier_list);
/** /**
* emergency_restart - reboot the system * emergency_restart - reboot the system
* *
@ -1130,7 +1137,7 @@ static ssize_t mode_show(struct kobject *kobj, struct kobj_attribute *attr, char
val = REBOOT_UNDEFINED_STR; val = REBOOT_UNDEFINED_STR;
} }
return sprintf(buf, "%s\n", val); return sysfs_emit(buf, "%s\n", val);
} }
static ssize_t mode_store(struct kobject *kobj, struct kobj_attribute *attr, static ssize_t mode_store(struct kobject *kobj, struct kobj_attribute *attr,
const char *buf, size_t count) const char *buf, size_t count)
@ -1160,7 +1167,7 @@ static struct kobj_attribute reboot_mode_attr = __ATTR_RW(mode);
#ifdef CONFIG_X86 #ifdef CONFIG_X86
static ssize_t force_show(struct kobject *kobj, struct kobj_attribute *attr, char *buf) static ssize_t force_show(struct kobject *kobj, struct kobj_attribute *attr, char *buf)
{ {
return sprintf(buf, "%d\n", reboot_force); return sysfs_emit(buf, "%d\n", reboot_force);
} }
static ssize_t force_store(struct kobject *kobj, struct kobj_attribute *attr, static ssize_t force_store(struct kobject *kobj, struct kobj_attribute *attr,
const char *buf, size_t count) const char *buf, size_t count)
@ -1207,7 +1214,7 @@ static ssize_t type_show(struct kobject *kobj, struct kobj_attribute *attr, char
val = REBOOT_UNDEFINED_STR; val = REBOOT_UNDEFINED_STR;
} }
return sprintf(buf, "%s\n", val); return sysfs_emit(buf, "%s\n", val);
} }
static ssize_t type_store(struct kobject *kobj, struct kobj_attribute *attr, static ssize_t type_store(struct kobject *kobj, struct kobj_attribute *attr,
const char *buf, size_t count) const char *buf, size_t count)
@ -1240,7 +1247,7 @@ static struct kobj_attribute reboot_type_attr = __ATTR_RW(type);
#ifdef CONFIG_SMP #ifdef CONFIG_SMP
static ssize_t cpu_show(struct kobject *kobj, struct kobj_attribute *attr, char *buf) static ssize_t cpu_show(struct kobject *kobj, struct kobj_attribute *attr, char *buf)
{ {
return sprintf(buf, "%d\n", reboot_cpu); return sysfs_emit(buf, "%d\n", reboot_cpu);
} }
static ssize_t cpu_store(struct kobject *kobj, struct kobj_attribute *attr, static ssize_t cpu_store(struct kobject *kobj, struct kobj_attribute *attr,
const char *buf, size_t count) const char *buf, size_t count)

View File

@ -50,17 +50,35 @@ EXPORT_SYMBOL(iomem_resource);
static DEFINE_RWLOCK(resource_lock); static DEFINE_RWLOCK(resource_lock);
static struct resource *next_resource(struct resource *p, bool skip_children) /*
* Return the next node of @p in pre-order tree traversal. If
* @skip_children is true, skip the descendant nodes of @p in
* traversal. If @p is a descendant of @subtree_root, only traverse
* the subtree under @subtree_root.
*/
static struct resource *next_resource(struct resource *p, bool skip_children,
struct resource *subtree_root)
{ {
if (!skip_children && p->child) if (!skip_children && p->child)
return p->child; return p->child;
while (!p->sibling && p->parent) while (!p->sibling && p->parent) {
p = p->parent; p = p->parent;
if (p == subtree_root)
return NULL;
}
return p->sibling; return p->sibling;
} }
/*
* Traverse the resource subtree under @_root in pre-order, excluding
* @_root itself.
*
* NOTE: '__p' is introduced to avoid shadowing '_p' outside of loop.
* And it is referenced to avoid unused variable warning.
*/
#define for_each_resource(_root, _p, _skip_children) \ #define for_each_resource(_root, _p, _skip_children) \
for ((_p) = (_root)->child; (_p); (_p) = next_resource(_p, _skip_children)) for (typeof(_root) __root = (_root), __p = _p = __root->child; \
__p && _p; _p = next_resource(_p, _skip_children, __root))
#ifdef CONFIG_PROC_FS #ifdef CONFIG_PROC_FS
@ -88,7 +106,7 @@ static void *r_next(struct seq_file *m, void *v, loff_t *pos)
(*pos)++; (*pos)++;
return (void *)next_resource(p, false); return (void *)next_resource(p, false, NULL);
} }
static void r_stop(struct seq_file *m, void *v) static void r_stop(struct seq_file *m, void *v)
@ -297,6 +315,11 @@ int release_resource(struct resource *old)
EXPORT_SYMBOL(release_resource); EXPORT_SYMBOL(release_resource);
static bool is_type_match(struct resource *p, unsigned long flags, unsigned long desc)
{
return (p->flags & flags) == flags && (desc == IORES_DESC_NONE || desc == p->desc);
}
/** /**
* find_next_iomem_res - Finds the lowest iomem resource that covers part of * find_next_iomem_res - Finds the lowest iomem resource that covers part of
* [@start..@end]. * [@start..@end].
@ -339,12 +362,8 @@ static int find_next_iomem_res(resource_size_t start, resource_size_t end,
if (p->end < start) if (p->end < start)
continue; continue;
if ((p->flags & flags) != flags)
continue;
if ((desc != IORES_DESC_NONE) && (desc != p->desc))
continue;
/* Found a match, break */ /* Found a match, break */
if (is_type_match(p, flags, desc))
break; break;
} }
@ -537,21 +556,18 @@ static int __region_intersects(struct resource *parent, resource_size_t start,
size_t size, unsigned long flags, size_t size, unsigned long flags,
unsigned long desc) unsigned long desc)
{ {
resource_size_t ostart, oend;
int type = 0; int other = 0; int type = 0; int other = 0;
struct resource *p, *dp; struct resource *p, *dp;
bool is_type, covered; struct resource res, o;
struct resource res; bool covered;
res.start = start; res.start = start;
res.end = start + size - 1; res.end = start + size - 1;
for (p = parent->child; p ; p = p->sibling) { for (p = parent->child; p ; p = p->sibling) {
if (!resource_overlaps(p, &res)) if (!resource_intersection(p, &res, &o))
continue; continue;
is_type = (p->flags & flags) == flags && if (is_type_match(p, flags, desc)) {
(desc == IORES_DESC_NONE || desc == p->desc);
if (is_type) {
type++; type++;
continue; continue;
} }
@ -568,27 +584,23 @@ static int __region_intersects(struct resource *parent, resource_size_t start,
* |-- "System RAM" --||-- "CXL Window 0a" --| * |-- "System RAM" --||-- "CXL Window 0a" --|
*/ */
covered = false; covered = false;
ostart = max(res.start, p->start);
oend = min(res.end, p->end);
for_each_resource(p, dp, false) { for_each_resource(p, dp, false) {
if (!resource_overlaps(dp, &res)) if (!resource_overlaps(dp, &res))
continue; continue;
is_type = (dp->flags & flags) == flags && if (is_type_match(dp, flags, desc)) {
(desc == IORES_DESC_NONE || desc == dp->desc);
if (is_type) {
type++; type++;
/* /*
* Range from 'ostart' to 'dp->start' * Range from 'o.start' to 'dp->start'
* isn't covered by matched resource. * isn't covered by matched resource.
*/ */
if (dp->start > ostart) if (dp->start > o.start)
break; break;
if (dp->end >= oend) { if (dp->end >= o.end) {
covered = true; covered = true;
break; break;
} }
/* Remove covered range */ /* Remove covered range */
ostart = max(ostart, dp->end + 1); o.start = max(o.start, dp->end + 1);
} }
} }
if (!covered) if (!covered)
@ -744,7 +756,7 @@ EXPORT_SYMBOL_GPL(find_resource_space);
* @root: root resource descriptor * @root: root resource descriptor
* @old: resource descriptor desired by caller * @old: resource descriptor desired by caller
* @newsize: new size of the resource descriptor * @newsize: new size of the resource descriptor
* @constraint: the size and alignment constraints to be met. * @constraint: the memory range and alignment constraints to be met.
*/ */
static int reallocate_resource(struct resource *root, struct resource *old, static int reallocate_resource(struct resource *root, struct resource *old,
resource_size_t newsize, resource_size_t newsize,

View File

@ -998,6 +998,7 @@ static int proc_watchdog_common(int which, const struct ctl_table *table, int wr
mutex_lock(&watchdog_mutex); mutex_lock(&watchdog_mutex);
old = *param;
if (!write) { if (!write) {
/* /*
* On read synchronize the userspace interface. This is a * On read synchronize the userspace interface. This is a
@ -1005,8 +1006,8 @@ static int proc_watchdog_common(int which, const struct ctl_table *table, int wr
*/ */
*param = (watchdog_enabled & which) != 0; *param = (watchdog_enabled & which) != 0;
err = proc_dointvec_minmax(table, write, buffer, lenp, ppos); err = proc_dointvec_minmax(table, write, buffer, lenp, ppos);
*param = old;
} else { } else {
old = READ_ONCE(*param);
err = proc_dointvec_minmax(table, write, buffer, lenp, ppos); err = proc_dointvec_minmax(table, write, buffer, lenp, ppos);
if (!err && old != READ_ONCE(*param)) if (!err && old != READ_ONCE(*param))
proc_watchdog_update(); proc_watchdog_update();

View File

@ -789,3 +789,9 @@ config POLYNOMIAL
config FIRMWARE_TABLE config FIRMWARE_TABLE
bool bool
config UNION_FIND
bool
config MIN_HEAP
bool

View File

@ -2269,6 +2269,7 @@ config TEST_LIST_SORT
config TEST_MIN_HEAP config TEST_MIN_HEAP
tristate "Min heap test" tristate "Min heap test"
depends on DEBUG_KERNEL || m depends on DEBUG_KERNEL || m
select MIN_HEAP
help help
Enable this to turn on min heap function tests. This test is Enable this to turn on min heap function tests. This test is
executed only once during system boot (so affects only boot time), executed only once during system boot (so affects only boot time),
@ -2619,6 +2620,23 @@ config CHECKSUM_KUNIT
If unsure, say N. If unsure, say N.
config UTIL_MACROS_KUNIT
tristate "KUnit test util_macros.h functions at runtime" if !KUNIT_ALL_TESTS
depends on KUNIT
default KUNIT_ALL_TESTS
help
Enable this option to test the util_macros.h function at boot.
KUnit tests run during boot and output the results to the debug log
in TAP format (http://testanything.org/). Only useful for kernel devs
running the KUnit test harness, and not intended for inclusion into a
production build.
For more information on KUnit and unit tests in general please refer
to the KUnit documentation in Documentation/dev-tools/kunit/.
If unsure, say N.
config HASH_KUNIT_TEST config HASH_KUNIT_TEST
tristate "KUnit Test for integer hash functions" if !KUNIT_ALL_TESTS tristate "KUnit Test for integer hash functions" if !KUNIT_ALL_TESTS
depends on KUNIT depends on KUNIT
@ -2840,6 +2858,15 @@ config USERCOPY_KUNIT_TEST
on the copy_to/from_user infrastructure, making sure basic on the copy_to/from_user infrastructure, making sure basic
user/kernel boundary testing is working. user/kernel boundary testing is working.
config CRC16_KUNIT_TEST
tristate "KUnit tests for CRC16"
depends on KUNIT
default KUNIT_ALL_TESTS
select CRC16
help
Enable this option to run unit tests for the kernel's CRC16
implementation (<linux/crc16.h>).
config TEST_UDELAY config TEST_UDELAY
tristate "udelay test driver" tristate "udelay test driver"
help help
@ -2983,6 +3010,22 @@ config TEST_OBJPOOL
If unsure, say N. If unsure, say N.
config INT_POW_TEST
tristate "Integer exponentiation (int_pow) test" if !KUNIT_ALL_TESTS
depends on KUNIT
default KUNIT_ALL_TESTS
help
This option enables the KUnit test suite for the int_pow function,
which performs integer exponentiation. The test suite is designed to
verify that the implementation of int_pow correctly computes the power
of a given base raised to a given exponent.
Enabling this option will include tests that check various scenarios
and edge cases to ensure the accuracy and reliability of the exponentiation
function.
If unsure, say N
endif # RUNTIME_TESTING_MENU endif # RUNTIME_TESTING_MENU
config ARCH_USE_MEMTEST config ARCH_USE_MEMTEST
@ -3078,19 +3121,3 @@ config RUST_KERNEL_DOCTESTS
endmenu # "Rust" endmenu # "Rust"
endmenu # Kernel hacking endmenu # Kernel hacking
config INT_POW_TEST
tristate "Integer exponentiation (int_pow) test" if !KUNIT_ALL_TESTS
depends on KUNIT
default KUNIT_ALL_TESTS
help
This option enables the KUnit test suite for the int_pow function,
which performs integer exponentiation. The test suite is designed to
verify that the implementation of int_pow correctly computes the power
of a given base raised to a given exponent.
Enabling this option will include tests that check various scenarios
and edge cases to ensure the accuracy and reliability of the exponentiation
function.
If unsure, say N

View File

@ -35,10 +35,12 @@ lib-y := ctype.o string.o vsprintf.o cmdline.o \
is_single_threaded.o plist.o decompress.o kobject_uevent.o \ is_single_threaded.o plist.o decompress.o kobject_uevent.o \
earlycpio.o seq_buf.o siphash.o dec_and_lock.o \ earlycpio.o seq_buf.o siphash.o dec_and_lock.o \
nmi_backtrace.o win_minmax.o memcat_p.o \ nmi_backtrace.o win_minmax.o memcat_p.o \
buildid.o objpool.o union_find.o iomem_copy.o buildid.o objpool.o iomem_copy.o
lib-$(CONFIG_UNION_FIND) += union_find.o
lib-$(CONFIG_PRINTK) += dump_stack.o lib-$(CONFIG_PRINTK) += dump_stack.o
lib-$(CONFIG_SMP) += cpumask.o lib-$(CONFIG_SMP) += cpumask.o
lib-$(CONFIG_MIN_HEAP) += min_heap.o
lib-y += kobject.o klist.o lib-y += kobject.o klist.o
obj-y += lockref.o obj-y += lockref.o
@ -371,6 +373,7 @@ obj-$(CONFIG_PLDMFW) += pldmfw/
CFLAGS_bitfield_kunit.o := $(DISABLE_STRUCTLEAK_PLUGIN) CFLAGS_bitfield_kunit.o := $(DISABLE_STRUCTLEAK_PLUGIN)
obj-$(CONFIG_BITFIELD_KUNIT) += bitfield_kunit.o obj-$(CONFIG_BITFIELD_KUNIT) += bitfield_kunit.o
obj-$(CONFIG_CHECKSUM_KUNIT) += checksum_kunit.o obj-$(CONFIG_CHECKSUM_KUNIT) += checksum_kunit.o
obj-$(CONFIG_UTIL_MACROS_KUNIT) += util_macros_kunit.o
obj-$(CONFIG_LIST_KUNIT_TEST) += list-test.o obj-$(CONFIG_LIST_KUNIT_TEST) += list-test.o
obj-$(CONFIG_HASHTABLE_KUNIT_TEST) += hashtable_test.o obj-$(CONFIG_HASHTABLE_KUNIT_TEST) += hashtable_test.o
obj-$(CONFIG_LINEAR_RANGES_TEST) += test_linear_ranges.o obj-$(CONFIG_LINEAR_RANGES_TEST) += test_linear_ranges.o
@ -390,6 +393,7 @@ CFLAGS_fortify_kunit.o += $(DISABLE_STRUCTLEAK_PLUGIN)
obj-$(CONFIG_FORTIFY_KUNIT_TEST) += fortify_kunit.o obj-$(CONFIG_FORTIFY_KUNIT_TEST) += fortify_kunit.o
obj-$(CONFIG_SIPHASH_KUNIT_TEST) += siphash_kunit.o obj-$(CONFIG_SIPHASH_KUNIT_TEST) += siphash_kunit.o
obj-$(CONFIG_USERCOPY_KUNIT_TEST) += usercopy_kunit.o obj-$(CONFIG_USERCOPY_KUNIT_TEST) += usercopy_kunit.o
obj-$(CONFIG_CRC16_KUNIT_TEST) += crc16_kunit.o
obj-$(CONFIG_GENERIC_LIB_DEVMEM_IS_ALLOWED) += devmem_is_allowed.o obj-$(CONFIG_GENERIC_LIB_DEVMEM_IS_ALLOWED) += devmem_is_allowed.o

155
lib/crc16_kunit.c Normal file
View File

@ -0,0 +1,155 @@
// SPDX-License-Identifier: GPL-2.0
/*
* KUnits tests for CRC16.
*
* Copyright (C) 2024, LKCAMP
* Author: Vinicius Peixoto <vpeixoto@lkcamp.dev>
* Author: Fabricio Gasperin <fgasperin@lkcamp.dev>
* Author: Enzo Bertoloti <ebertoloti@lkcamp.dev>
*/
#include <kunit/test.h>
#include <linux/crc16.h>
#include <linux/prandom.h>
#define CRC16_KUNIT_DATA_SIZE 4096
#define CRC16_KUNIT_TEST_SIZE 100
#define CRC16_KUNIT_SEED 0x12345678
/**
* struct crc16_test - CRC16 test data
* @crc: initial input value to CRC16
* @start: Start index within the data buffer
* @length: Length of the data
*/
static struct crc16_test {
u16 crc;
u16 start;
u16 length;
} tests[CRC16_KUNIT_TEST_SIZE];
u8 data[CRC16_KUNIT_DATA_SIZE];
/* Naive implementation of CRC16 for validation purposes */
static inline u16 _crc16_naive_byte(u16 crc, u8 data)
{
u8 i = 0;
crc ^= (u16) data;
for (i = 0; i < 8; i++) {
if (crc & 0x01)
crc = (crc >> 1) ^ 0xa001;
else
crc = crc >> 1;
}
return crc;
}
static inline u16 _crc16_naive(u16 crc, u8 *buffer, size_t len)
{
while (len--)
crc = _crc16_naive_byte(crc, *buffer++);
return crc;
}
/* Small helper for generating pseudorandom 16-bit data */
static inline u16 _rand16(void)
{
static u32 rand = CRC16_KUNIT_SEED;
rand = next_pseudo_random32(rand);
return rand & 0xFFFF;
}
static int crc16_init_test_data(struct kunit_suite *suite)
{
size_t i;
/* Fill the data buffer with random bytes */
for (i = 0; i < CRC16_KUNIT_DATA_SIZE; i++)
data[i] = _rand16() & 0xFF;
/* Generate random test data while ensuring the random
* start + length values won't overflow the 4096-byte
* buffer (0x7FF * 2 = 0xFFE < 0x1000)
*/
for (size_t i = 0; i < CRC16_KUNIT_TEST_SIZE; i++) {
tests[i].crc = _rand16();
tests[i].start = _rand16() & 0x7FF;
tests[i].length = _rand16() & 0x7FF;
}
return 0;
}
static void crc16_test_empty(struct kunit *test)
{
u16 crc;
/* The result for empty data should be the same as the
* initial crc
*/
crc = crc16(0x00, data, 0);
KUNIT_EXPECT_EQ(test, crc, 0);
crc = crc16(0xFF, data, 0);
KUNIT_EXPECT_EQ(test, crc, 0xFF);
}
static void crc16_test_correctness(struct kunit *test)
{
size_t i;
u16 crc, crc_naive;
for (i = 0; i < CRC16_KUNIT_TEST_SIZE; i++) {
/* Compare results with the naive crc16 implementation */
crc = crc16(tests[i].crc, data + tests[i].start,
tests[i].length);
crc_naive = _crc16_naive(tests[i].crc, data + tests[i].start,
tests[i].length);
KUNIT_EXPECT_EQ(test, crc, crc_naive);
}
}
static void crc16_test_combine(struct kunit *test)
{
size_t i, j;
u16 crc, crc_naive;
/* Make sure that combining two consecutive crc16 calculations
* yields the same result as calculating the crc16 for the whole thing
*/
for (i = 0; i < CRC16_KUNIT_TEST_SIZE; i++) {
crc_naive = crc16(tests[i].crc, data + tests[i].start, tests[i].length);
for (j = 0; j < tests[i].length; j++) {
crc = crc16(tests[i].crc, data + tests[i].start, j);
crc = crc16(crc, data + tests[i].start + j, tests[i].length - j);
KUNIT_EXPECT_EQ(test, crc, crc_naive);
}
}
}
static struct kunit_case crc16_test_cases[] = {
KUNIT_CASE(crc16_test_empty),
KUNIT_CASE(crc16_test_combine),
KUNIT_CASE(crc16_test_correctness),
{},
};
static struct kunit_suite crc16_test_suite = {
.name = "crc16",
.test_cases = crc16_test_cases,
.suite_init = crc16_init_test_data,
};
kunit_test_suite(crc16_test_suite);
MODULE_AUTHOR("Fabricio Gasperin <fgasperin@lkcamp.dev>");
MODULE_AUTHOR("Vinicius Peixoto <vpeixoto@lkcamp.dev>");
MODULE_AUTHOR("Enzo Bertoloti <ebertoloti@lkcamp.dev>");
MODULE_DESCRIPTION("Unit tests for crc16");
MODULE_LICENSE("GPL");

View File

@ -412,6 +412,8 @@ static void list_test_list_cut_position(struct kunit *test)
KUNIT_EXPECT_PTR_EQ(test, cur, &entries[i]); KUNIT_EXPECT_PTR_EQ(test, cur, &entries[i]);
i++; i++;
} }
KUNIT_EXPECT_EQ(test, i, 3);
} }
static void list_test_list_cut_before(struct kunit *test) static void list_test_list_cut_before(struct kunit *test)
@ -440,6 +442,8 @@ static void list_test_list_cut_before(struct kunit *test)
KUNIT_EXPECT_PTR_EQ(test, cur, &entries[i]); KUNIT_EXPECT_PTR_EQ(test, cur, &entries[i]);
i++; i++;
} }
KUNIT_EXPECT_EQ(test, i, 3);
} }
static void list_test_list_splice(struct kunit *test) static void list_test_list_splice(struct kunit *test)

View File

@ -1,9 +1,6 @@
// SPDX-License-Identifier: GPL-2.0 // SPDX-License-Identifier: GPL-2.0
#include <linux/kernel.h>
#include <linux/bug.h>
#include <linux/compiler.h> #include <linux/compiler.h>
#include <linux/export.h> #include <linux/export.h>
#include <linux/string.h>
#include <linux/list_sort.h> #include <linux/list_sort.h>
#include <linux/list.h> #include <linux/list.h>

70
lib/min_heap.c Normal file
View File

@ -0,0 +1,70 @@
// SPDX-License-Identifier: GPL-2.0
#include <linux/export.h>
#include <linux/min_heap.h>
void __min_heap_init(min_heap_char *heap, void *data, int size)
{
__min_heap_init_inline(heap, data, size);
}
EXPORT_SYMBOL(__min_heap_init);
void *__min_heap_peek(struct min_heap_char *heap)
{
return __min_heap_peek_inline(heap);
}
EXPORT_SYMBOL(__min_heap_peek);
bool __min_heap_full(min_heap_char *heap)
{
return __min_heap_full_inline(heap);
}
EXPORT_SYMBOL(__min_heap_full);
void __min_heap_sift_down(min_heap_char *heap, int pos, size_t elem_size,
const struct min_heap_callbacks *func, void *args)
{
__min_heap_sift_down_inline(heap, pos, elem_size, func, args);
}
EXPORT_SYMBOL(__min_heap_sift_down);
void __min_heap_sift_up(min_heap_char *heap, size_t elem_size, size_t idx,
const struct min_heap_callbacks *func, void *args)
{
__min_heap_sift_up_inline(heap, elem_size, idx, func, args);
}
EXPORT_SYMBOL(__min_heap_sift_up);
void __min_heapify_all(min_heap_char *heap, size_t elem_size,
const struct min_heap_callbacks *func, void *args)
{
__min_heapify_all_inline(heap, elem_size, func, args);
}
EXPORT_SYMBOL(__min_heapify_all);
bool __min_heap_pop(min_heap_char *heap, size_t elem_size,
const struct min_heap_callbacks *func, void *args)
{
return __min_heap_pop_inline(heap, elem_size, func, args);
}
EXPORT_SYMBOL(__min_heap_pop);
void __min_heap_pop_push(min_heap_char *heap, const void *element, size_t elem_size,
const struct min_heap_callbacks *func, void *args)
{
__min_heap_pop_push_inline(heap, element, elem_size, func, args);
}
EXPORT_SYMBOL(__min_heap_pop_push);
bool __min_heap_push(min_heap_char *heap, const void *element, size_t elem_size,
const struct min_heap_callbacks *func, void *args)
{
return __min_heap_push_inline(heap, element, elem_size, func, args);
}
EXPORT_SYMBOL(__min_heap_push);
bool __min_heap_del(min_heap_char *heap, size_t elem_size, size_t idx,
const struct min_heap_callbacks *func, void *args)
{
return __min_heap_del_inline(heap, elem_size, idx, func, args);
}
EXPORT_SYMBOL(__min_heap_del);

View File

@ -474,14 +474,14 @@ int sg_alloc_append_table_from_pages(struct sg_append_table *sgt_append,
return -EOPNOTSUPP; return -EOPNOTSUPP;
if (sgt_append->prv) { if (sgt_append->prv) {
unsigned long next_pfn = (page_to_phys(sg_page(sgt_append->prv)) + unsigned long next_pfn;
sgt_append->prv->offset + sgt_append->prv->length) / PAGE_SIZE;
if (WARN_ON(offset)) if (WARN_ON(offset))
return -EINVAL; return -EINVAL;
/* Merge contiguous pages into the last SG */ /* Merge contiguous pages into the last SG */
prv_len = sgt_append->prv->length; prv_len = sgt_append->prv->length;
next_pfn = (sg_phys(sgt_append->prv) + prv_len) / PAGE_SIZE;
if (page_to_pfn(pages[0]) == next_pfn) { if (page_to_pfn(pages[0]) == next_pfn) {
last_pg = pfn_to_page(next_pfn - 1); last_pg = pfn_to_page(next_pfn - 1);
while (n_pages && pages_are_mergeable(pages[0], last_pg)) { while (n_pages && pages_are_mergeable(pages[0], last_pg)) {

View File

@ -23,14 +23,6 @@ static __init bool greater_than(const void *lhs, const void *rhs, void __always_
return *(int *)lhs > *(int *)rhs; return *(int *)lhs > *(int *)rhs;
} }
static __init void swap_ints(void *lhs, void *rhs, void __always_unused *args)
{
int temp = *(int *)lhs;
*(int *)lhs = *(int *)rhs;
*(int *)rhs = temp;
}
static __init int pop_verify_heap(bool min_heap, static __init int pop_verify_heap(bool min_heap,
struct min_heap_test *heap, struct min_heap_test *heap,
const struct min_heap_callbacks *funcs) const struct min_heap_callbacks *funcs)
@ -72,7 +64,7 @@ static __init int test_heapify_all(bool min_heap)
}; };
struct min_heap_callbacks funcs = { struct min_heap_callbacks funcs = {
.less = min_heap ? less_than : greater_than, .less = min_heap ? less_than : greater_than,
.swp = swap_ints, .swp = NULL,
}; };
int i, err; int i, err;
@ -104,7 +96,7 @@ static __init int test_heap_push(bool min_heap)
}; };
struct min_heap_callbacks funcs = { struct min_heap_callbacks funcs = {
.less = min_heap ? less_than : greater_than, .less = min_heap ? less_than : greater_than,
.swp = swap_ints, .swp = NULL,
}; };
int i, temp, err; int i, temp, err;
@ -136,7 +128,7 @@ static __init int test_heap_pop_push(bool min_heap)
}; };
struct min_heap_callbacks funcs = { struct min_heap_callbacks funcs = {
.less = min_heap ? less_than : greater_than, .less = min_heap ? less_than : greater_than,
.swp = swap_ints, .swp = NULL,
}; };
int i, temp, err; int i, temp, err;
@ -175,7 +167,7 @@ static __init int test_heap_del(bool min_heap)
heap.nr = ARRAY_SIZE(values); heap.nr = ARRAY_SIZE(values);
struct min_heap_callbacks funcs = { struct min_heap_callbacks funcs = {
.less = min_heap ? less_than : greater_than, .less = min_heap ? less_than : greater_than,
.swp = swap_ints, .swp = NULL,
}; };
int i, err; int i, err;

240
lib/util_macros_kunit.c Normal file
View File

@ -0,0 +1,240 @@
// SPDX-License-Identifier: GPL-2.0+
/*
* Test cases for bitfield helpers.
*/
#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
#include <kunit/test.h>
#include <linux/util_macros.h>
#define FIND_CLOSEST_RANGE_CHECK(from, to, array, exp_idx) \
{ \
int i; \
for (i = from; i <= to; i++) { \
int found = find_closest(i, array, ARRAY_SIZE(array)); \
KUNIT_ASSERT_EQ(ctx, exp_idx, found); \
} \
}
static void test_find_closest(struct kunit *ctx)
{
/* This will test a few arrays that are found in drivers */
static const int ina226_avg_tab[] = { 1, 4, 16, 64, 128, 256, 512, 1024 };
static const unsigned int ad7616_oversampling_avail[] = {
1, 2, 4, 8, 16, 32, 64, 128,
};
static u32 wd_timeout_table[] = { 2, 4, 6, 8, 16, 32, 48, 64 };
static int array_prog1a[] = { 1, 2, 3, 4, 5 };
static u32 array_prog1b[] = { 2, 3, 4, 5, 6 };
static int array_prog1mix[] = { -2, -1, 0, 1, 2 };
static int array_prog2a[] = { 1, 3, 5, 7 };
static u32 array_prog2b[] = { 2, 4, 6, 8 };
static int array_prog3a[] = { 1, 4, 7, 10 };
static u32 array_prog3b[] = { 2, 5, 8, 11 };
static int array_prog4a[] = { 1, 5, 9, 13 };
static u32 array_prog4b[] = { 2, 6, 10, 14 };
FIND_CLOSEST_RANGE_CHECK(-3, 2, ina226_avg_tab, 0);
FIND_CLOSEST_RANGE_CHECK(3, 10, ina226_avg_tab, 1);
FIND_CLOSEST_RANGE_CHECK(11, 40, ina226_avg_tab, 2);
FIND_CLOSEST_RANGE_CHECK(41, 96, ina226_avg_tab, 3);
FIND_CLOSEST_RANGE_CHECK(97, 192, ina226_avg_tab, 4);
FIND_CLOSEST_RANGE_CHECK(193, 384, ina226_avg_tab, 5);
FIND_CLOSEST_RANGE_CHECK(385, 768, ina226_avg_tab, 6);
FIND_CLOSEST_RANGE_CHECK(769, 2048, ina226_avg_tab, 7);
/* The array that found the bug that caused this kunit to exist */
FIND_CLOSEST_RANGE_CHECK(-3, 1, ad7616_oversampling_avail, 0);
FIND_CLOSEST_RANGE_CHECK(2, 3, ad7616_oversampling_avail, 1);
FIND_CLOSEST_RANGE_CHECK(4, 6, ad7616_oversampling_avail, 2);
FIND_CLOSEST_RANGE_CHECK(7, 12, ad7616_oversampling_avail, 3);
FIND_CLOSEST_RANGE_CHECK(13, 24, ad7616_oversampling_avail, 4);
FIND_CLOSEST_RANGE_CHECK(25, 48, ad7616_oversampling_avail, 5);
FIND_CLOSEST_RANGE_CHECK(49, 96, ad7616_oversampling_avail, 6);
FIND_CLOSEST_RANGE_CHECK(97, 256, ad7616_oversampling_avail, 7);
FIND_CLOSEST_RANGE_CHECK(-3, 3, wd_timeout_table, 0);
FIND_CLOSEST_RANGE_CHECK(4, 5, wd_timeout_table, 1);
FIND_CLOSEST_RANGE_CHECK(6, 7, wd_timeout_table, 2);
FIND_CLOSEST_RANGE_CHECK(8, 12, wd_timeout_table, 3);
FIND_CLOSEST_RANGE_CHECK(13, 24, wd_timeout_table, 4);
FIND_CLOSEST_RANGE_CHECK(25, 40, wd_timeout_table, 5);
FIND_CLOSEST_RANGE_CHECK(41, 56, wd_timeout_table, 6);
FIND_CLOSEST_RANGE_CHECK(57, 128, wd_timeout_table, 7);
/* One could argue that find_closest() should not be used for monotonic
* arrays (like 1,2,3,4,5), but even so, it should work as long as the
* array is sorted ascending. */
FIND_CLOSEST_RANGE_CHECK(-3, 1, array_prog1a, 0);
FIND_CLOSEST_RANGE_CHECK(2, 2, array_prog1a, 1);
FIND_CLOSEST_RANGE_CHECK(3, 3, array_prog1a, 2);
FIND_CLOSEST_RANGE_CHECK(4, 4, array_prog1a, 3);
FIND_CLOSEST_RANGE_CHECK(5, 8, array_prog1a, 4);
FIND_CLOSEST_RANGE_CHECK(-3, 2, array_prog1b, 0);
FIND_CLOSEST_RANGE_CHECK(3, 3, array_prog1b, 1);
FIND_CLOSEST_RANGE_CHECK(4, 4, array_prog1b, 2);
FIND_CLOSEST_RANGE_CHECK(5, 5, array_prog1b, 3);
FIND_CLOSEST_RANGE_CHECK(6, 8, array_prog1b, 4);
FIND_CLOSEST_RANGE_CHECK(-4, -2, array_prog1mix, 0);
FIND_CLOSEST_RANGE_CHECK(-1, -1, array_prog1mix, 1);
FIND_CLOSEST_RANGE_CHECK(0, 0, array_prog1mix, 2);
FIND_CLOSEST_RANGE_CHECK(1, 1, array_prog1mix, 3);
FIND_CLOSEST_RANGE_CHECK(2, 5, array_prog1mix, 4);
FIND_CLOSEST_RANGE_CHECK(-3, 2, array_prog2a, 0);
FIND_CLOSEST_RANGE_CHECK(3, 4, array_prog2a, 1);
FIND_CLOSEST_RANGE_CHECK(5, 6, array_prog2a, 2);
FIND_CLOSEST_RANGE_CHECK(7, 10, array_prog2a, 3);
FIND_CLOSEST_RANGE_CHECK(-3, 3, array_prog2b, 0);
FIND_CLOSEST_RANGE_CHECK(4, 5, array_prog2b, 1);
FIND_CLOSEST_RANGE_CHECK(6, 7, array_prog2b, 2);
FIND_CLOSEST_RANGE_CHECK(8, 10, array_prog2b, 3);
FIND_CLOSEST_RANGE_CHECK(-3, 2, array_prog3a, 0);
FIND_CLOSEST_RANGE_CHECK(3, 5, array_prog3a, 1);
FIND_CLOSEST_RANGE_CHECK(6, 8, array_prog3a, 2);
FIND_CLOSEST_RANGE_CHECK(9, 20, array_prog3a, 3);
FIND_CLOSEST_RANGE_CHECK(-3, 3, array_prog3b, 0);
FIND_CLOSEST_RANGE_CHECK(4, 6, array_prog3b, 1);
FIND_CLOSEST_RANGE_CHECK(7, 9, array_prog3b, 2);
FIND_CLOSEST_RANGE_CHECK(10, 20, array_prog3b, 3);
FIND_CLOSEST_RANGE_CHECK(-3, 3, array_prog4a, 0);
FIND_CLOSEST_RANGE_CHECK(4, 7, array_prog4a, 1);
FIND_CLOSEST_RANGE_CHECK(8, 11, array_prog4a, 2);
FIND_CLOSEST_RANGE_CHECK(12, 20, array_prog4a, 3);
FIND_CLOSEST_RANGE_CHECK(-3, 4, array_prog4b, 0);
FIND_CLOSEST_RANGE_CHECK(5, 8, array_prog4b, 1);
FIND_CLOSEST_RANGE_CHECK(9, 12, array_prog4b, 2);
FIND_CLOSEST_RANGE_CHECK(13, 20, array_prog4b, 3);
}
#define FIND_CLOSEST_DESC_RANGE_CHECK(from, to, array, exp_idx) \
{ \
int i; \
for (i = from; i <= to; i++) { \
int found = find_closest_descending(i, array, \
ARRAY_SIZE(array)); \
KUNIT_ASSERT_EQ(ctx, exp_idx, found); \
} \
}
static void test_find_closest_descending(struct kunit *ctx)
{
/* Same arrays as 'test_find_closest' but reversed */
static const int ina226_avg_tab[] = { 1024, 512, 256, 128, 64, 16, 4, 1 };
static const unsigned int ad7616_oversampling_avail[] = {
128, 64, 32, 16, 8, 4, 2, 1
};
static u32 wd_timeout_table[] = { 64, 48, 32, 16, 8, 6, 4, 2 };
static int array_prog1a[] = { 5, 4, 3, 2, 1 };
static u32 array_prog1b[] = { 6, 5, 4, 3, 2 };
static int array_prog1mix[] = { 2, 1, 0, -1, -2 };
static int array_prog2a[] = { 7, 5, 3, 1 };
static u32 array_prog2b[] = { 8, 6, 4, 2 };
static int array_prog3a[] = { 10, 7, 4, 1 };
static u32 array_prog3b[] = { 11, 8, 5, 2 };
static int array_prog4a[] = { 13, 9, 5, 1 };
static u32 array_prog4b[] = { 14, 10, 6, 2 };
FIND_CLOSEST_DESC_RANGE_CHECK(-3, 2, ina226_avg_tab, 7);
FIND_CLOSEST_DESC_RANGE_CHECK(3, 10, ina226_avg_tab, 6);
FIND_CLOSEST_DESC_RANGE_CHECK(11, 40, ina226_avg_tab, 5);
FIND_CLOSEST_DESC_RANGE_CHECK(41, 96, ina226_avg_tab, 4);
FIND_CLOSEST_DESC_RANGE_CHECK(97, 192, ina226_avg_tab, 3);
FIND_CLOSEST_DESC_RANGE_CHECK(193, 384, ina226_avg_tab, 2);
FIND_CLOSEST_DESC_RANGE_CHECK(385, 768, ina226_avg_tab, 1);
FIND_CLOSEST_DESC_RANGE_CHECK(769, 2048, ina226_avg_tab, 0);
FIND_CLOSEST_DESC_RANGE_CHECK(-3, 1, ad7616_oversampling_avail, 7);
FIND_CLOSEST_DESC_RANGE_CHECK(2, 3, ad7616_oversampling_avail, 6);
FIND_CLOSEST_DESC_RANGE_CHECK(4, 6, ad7616_oversampling_avail, 5);
FIND_CLOSEST_DESC_RANGE_CHECK(7, 12, ad7616_oversampling_avail, 4);
FIND_CLOSEST_DESC_RANGE_CHECK(13, 24, ad7616_oversampling_avail, 3);
FIND_CLOSEST_DESC_RANGE_CHECK(25, 48, ad7616_oversampling_avail, 2);
FIND_CLOSEST_DESC_RANGE_CHECK(49, 96, ad7616_oversampling_avail, 1);
FIND_CLOSEST_DESC_RANGE_CHECK(97, 256, ad7616_oversampling_avail, 0);
FIND_CLOSEST_DESC_RANGE_CHECK(-3, 3, wd_timeout_table, 7);
FIND_CLOSEST_DESC_RANGE_CHECK(4, 5, wd_timeout_table, 6);
FIND_CLOSEST_DESC_RANGE_CHECK(6, 7, wd_timeout_table, 5);
FIND_CLOSEST_DESC_RANGE_CHECK(8, 12, wd_timeout_table, 4);
FIND_CLOSEST_DESC_RANGE_CHECK(13, 24, wd_timeout_table, 3);
FIND_CLOSEST_DESC_RANGE_CHECK(25, 40, wd_timeout_table, 2);
FIND_CLOSEST_DESC_RANGE_CHECK(41, 56, wd_timeout_table, 1);
FIND_CLOSEST_DESC_RANGE_CHECK(57, 128, wd_timeout_table, 0);
/* One could argue that find_closest_descending() should not be used
* for monotonic arrays (like 5,4,3,2,1), but even so, it should still
* it should work as long as the array is sorted descending. */
FIND_CLOSEST_DESC_RANGE_CHECK(-3, 1, array_prog1a, 4);
FIND_CLOSEST_DESC_RANGE_CHECK(2, 2, array_prog1a, 3);
FIND_CLOSEST_DESC_RANGE_CHECK(3, 3, array_prog1a, 2);
FIND_CLOSEST_DESC_RANGE_CHECK(4, 4, array_prog1a, 1);
FIND_CLOSEST_DESC_RANGE_CHECK(5, 8, array_prog1a, 0);
FIND_CLOSEST_DESC_RANGE_CHECK(-3, 2, array_prog1b, 4);
FIND_CLOSEST_DESC_RANGE_CHECK(3, 3, array_prog1b, 3);
FIND_CLOSEST_DESC_RANGE_CHECK(4, 4, array_prog1b, 2);
FIND_CLOSEST_DESC_RANGE_CHECK(5, 5, array_prog1b, 1);
FIND_CLOSEST_DESC_RANGE_CHECK(6, 8, array_prog1b, 0);
FIND_CLOSEST_DESC_RANGE_CHECK(-4, -2, array_prog1mix, 4);
FIND_CLOSEST_DESC_RANGE_CHECK(-1, -1, array_prog1mix, 3);
FIND_CLOSEST_DESC_RANGE_CHECK(0, 0, array_prog1mix, 2);
FIND_CLOSEST_DESC_RANGE_CHECK(1, 1, array_prog1mix, 1);
FIND_CLOSEST_DESC_RANGE_CHECK(2, 5, array_prog1mix, 0);
FIND_CLOSEST_DESC_RANGE_CHECK(-3, 2, array_prog2a, 3);
FIND_CLOSEST_DESC_RANGE_CHECK(3, 4, array_prog2a, 2);
FIND_CLOSEST_DESC_RANGE_CHECK(5, 6, array_prog2a, 1);
FIND_CLOSEST_DESC_RANGE_CHECK(7, 10, array_prog2a, 0);
FIND_CLOSEST_DESC_RANGE_CHECK(-3, 3, array_prog2b, 3);
FIND_CLOSEST_DESC_RANGE_CHECK(4, 5, array_prog2b, 2);
FIND_CLOSEST_DESC_RANGE_CHECK(6, 7, array_prog2b, 1);
FIND_CLOSEST_DESC_RANGE_CHECK(8, 10, array_prog2b, 0);
FIND_CLOSEST_DESC_RANGE_CHECK(-3, 2, array_prog3a, 3);
FIND_CLOSEST_DESC_RANGE_CHECK(3, 5, array_prog3a, 2);
FIND_CLOSEST_DESC_RANGE_CHECK(6, 8, array_prog3a, 1);
FIND_CLOSEST_DESC_RANGE_CHECK(9, 20, array_prog3a, 0);
FIND_CLOSEST_DESC_RANGE_CHECK(-3, 3, array_prog3b, 3);
FIND_CLOSEST_DESC_RANGE_CHECK(4, 6, array_prog3b, 2);
FIND_CLOSEST_DESC_RANGE_CHECK(7, 9, array_prog3b, 1);
FIND_CLOSEST_DESC_RANGE_CHECK(10, 20, array_prog3b, 0);
FIND_CLOSEST_DESC_RANGE_CHECK(-3, 3, array_prog4a, 3);
FIND_CLOSEST_DESC_RANGE_CHECK(4, 7, array_prog4a, 2);
FIND_CLOSEST_DESC_RANGE_CHECK(8, 11, array_prog4a, 1);
FIND_CLOSEST_DESC_RANGE_CHECK(12, 20, array_prog4a, 0);
FIND_CLOSEST_DESC_RANGE_CHECK(-3, 4, array_prog4b, 3);
FIND_CLOSEST_DESC_RANGE_CHECK(5, 8, array_prog4b, 2);
FIND_CLOSEST_DESC_RANGE_CHECK(9, 12, array_prog4b, 1);
FIND_CLOSEST_DESC_RANGE_CHECK(13, 20, array_prog4b, 0);
}
static struct kunit_case __refdata util_macros_test_cases[] = {
KUNIT_CASE(test_find_closest),
KUNIT_CASE(test_find_closest_descending),
{}
};
static struct kunit_suite util_macros_test_suite = {
.name = "util_macros.h",
.test_cases = util_macros_test_cases,
};
kunit_test_suites(&util_macros_test_suite);
MODULE_AUTHOR("Alexandru Ardelean <aardelean@baylibre.com>");
MODULE_DESCRIPTION("Test cases for util_macros.h helpers");
MODULE_LICENSE("GPL");

View File

@ -44,6 +44,30 @@ void kfree_const(const void *x)
} }
EXPORT_SYMBOL(kfree_const); EXPORT_SYMBOL(kfree_const);
/**
* __kmemdup_nul - Create a NUL-terminated string from @s, which might be unterminated.
* @s: The data to copy
* @len: The size of the data, not including the NUL terminator
* @gfp: the GFP mask used in the kmalloc() call when allocating memory
*
* Return: newly allocated copy of @s with NUL-termination or %NULL in
* case of error
*/
static __always_inline char *__kmemdup_nul(const char *s, size_t len, gfp_t gfp)
{
char *buf;
/* '+1' for the NUL terminator */
buf = kmalloc_track_caller(len + 1, gfp);
if (!buf)
return NULL;
memcpy(buf, s, len);
/* Ensure the buf is always NUL-terminated, regardless of @s. */
buf[len] = '\0';
return buf;
}
/** /**
* kstrdup - allocate space for and copy an existing string * kstrdup - allocate space for and copy an existing string
* @s: the string to duplicate * @s: the string to duplicate
@ -54,17 +78,7 @@ EXPORT_SYMBOL(kfree_const);
noinline noinline
char *kstrdup(const char *s, gfp_t gfp) char *kstrdup(const char *s, gfp_t gfp)
{ {
size_t len; return s ? __kmemdup_nul(s, strlen(s), gfp) : NULL;
char *buf;
if (!s)
return NULL;
len = strlen(s) + 1;
buf = kmalloc_track_caller(len, gfp);
if (buf)
memcpy(buf, s, len);
return buf;
} }
EXPORT_SYMBOL(kstrdup); EXPORT_SYMBOL(kstrdup);
@ -100,19 +114,7 @@ EXPORT_SYMBOL(kstrdup_const);
*/ */
char *kstrndup(const char *s, size_t max, gfp_t gfp) char *kstrndup(const char *s, size_t max, gfp_t gfp)
{ {
size_t len; return s ? __kmemdup_nul(s, strnlen(s, max), gfp) : NULL;
char *buf;
if (!s)
return NULL;
len = strnlen(s, max);
buf = kmalloc_track_caller(len+1, gfp);
if (buf) {
memcpy(buf, s, len);
buf[len] = '\0';
}
return buf;
} }
EXPORT_SYMBOL(kstrndup); EXPORT_SYMBOL(kstrndup);
@ -186,17 +188,7 @@ EXPORT_SYMBOL(kvmemdup);
*/ */
char *kmemdup_nul(const char *s, size_t len, gfp_t gfp) char *kmemdup_nul(const char *s, size_t len, gfp_t gfp)
{ {
char *buf; return s ? __kmemdup_nul(s, len, gfp) : NULL;
if (!s)
return NULL;
buf = kmalloc_track_caller(len + 1, gfp);
if (buf) {
memcpy(buf, s, len);
buf[len] = '\0';
}
return buf;
} }
EXPORT_SYMBOL(kmemdup_nul); EXPORT_SYMBOL(kmemdup_nul);

View File

@ -52,8 +52,8 @@ static int __init hw_break_module_init(void)
attr.bp_type = HW_BREAKPOINT_W; attr.bp_type = HW_BREAKPOINT_W;
sample_hbp = register_wide_hw_breakpoint(&attr, sample_hbp_handler, NULL); sample_hbp = register_wide_hw_breakpoint(&attr, sample_hbp_handler, NULL);
if (IS_ERR((void __force *)sample_hbp)) { if (IS_ERR_PCPU(sample_hbp)) {
ret = PTR_ERR((void __force *)sample_hbp); ret = PTR_ERR_PCPU(sample_hbp);
goto fail; goto fail;
} }

View File

@ -3209,36 +3209,31 @@ sub process {
# Check Fixes: styles is correct # Check Fixes: styles is correct
if (!$in_header_lines && if (!$in_header_lines &&
$line =~ /^\s*fixes:?\s*(?:commit\s*)?[0-9a-f]{5,}\b/i) { $line =~ /^\s*(fixes:?)\s*(?:commit\s*)?([0-9a-f]{5,40})(?:\s*($balanced_parens))?/i) {
my $orig_commit = ""; my $tag = $1;
my $id = "0123456789ab"; my $orig_commit = $2;
my $title = "commit title"; my $title;
my $tag_case = 1;
my $tag_space = 1;
my $id_length = 1;
my $id_case = 1;
my $title_has_quotes = 0; my $title_has_quotes = 0;
$fixes_tag = 1; $fixes_tag = 1;
if (defined $3) {
if ($line =~ /(\s*fixes:?)\s+([0-9a-f]{5,})\s+($balanced_parens)/i) {
my $tag = $1;
$orig_commit = $2;
$title = $3;
$tag_case = 0 if $tag eq "Fixes:";
$tag_space = 0 if ($line =~ /^fixes:? [0-9a-f]{5,} ($balanced_parens)/i);
$id_length = 0 if ($orig_commit =~ /^[0-9a-f]{12}$/i);
$id_case = 0 if ($orig_commit !~ /[A-F]/);
# Always strip leading/trailing parens then double quotes if existing # Always strip leading/trailing parens then double quotes if existing
$title = substr($title, 1, -1); $title = substr($3, 1, -1);
if ($title =~ /^".*"$/) { if ($title =~ /^".*"$/) {
$title = substr($title, 1, -1); $title = substr($title, 1, -1);
$title_has_quotes = 1; $title_has_quotes = 1;
} }
} else {
$title = "commit title"
} }
my $tag_case = not ($tag eq "Fixes:");
my $tag_space = not ($line =~ /^fixes:? [0-9a-f]{5,40} ($balanced_parens)/i);
my $id_length = not ($orig_commit =~ /^[0-9a-f]{12}$/i);
my $id_case = not ($orig_commit !~ /[A-F]/);
my $id = "0123456789ab";
my ($cid, $ctitle) = git_commit_info($orig_commit, $id, my ($cid, $ctitle) = git_commit_info($orig_commit, $id,
$title); $title);

View File

@ -311,7 +311,12 @@ handle_line() {
parse_symbol # modifies $symbol parse_symbol # modifies $symbol
# Add up the line number to the symbol # Add up the line number to the symbol
if [[ -z ${module} ]]
then
echo "${words[@]}" "$symbol"
else
echo "${words[@]}" "$symbol $module" echo "${words[@]}" "$symbol $module"
fi
} }
while read line; do while read line; do

View File

@ -19,6 +19,9 @@ from linux import cpus, utils, lists, constants
module_type = utils.CachedType("struct module") module_type = utils.CachedType("struct module")
def has_modules():
return utils.gdb_eval_or_none("modules") is not None
def module_list(): def module_list():
global module_type global module_type
modules = utils.gdb_eval_or_none("modules") modules = utils.gdb_eval_or_none("modules")

View File

@ -178,6 +178,9 @@ lx-symbols command."""
self.load_all_symbols() self.load_all_symbols()
if not modules.has_modules():
return
if hasattr(gdb, 'Breakpoint'): if hasattr(gdb, 'Breakpoint'):
if self.breakpoint is not None: if self.breakpoint is not None:
self.breakpoint.delete() self.breakpoint.delete()

View File

@ -141,6 +141,7 @@ anomoly||anomaly
anonynous||anonymous anonynous||anonymous
anway||anyway anway||anyway
aplication||application aplication||application
apeared||appeared
appearence||appearance appearence||appearance
applicaion||application applicaion||application
appliction||application appliction||application
@ -155,6 +156,7 @@ apropriate||appropriate
aquainted||acquainted aquainted||acquainted
aquired||acquired aquired||acquired
aquisition||acquisition aquisition||acquisition
aquires||acquires
arbitary||arbitrary arbitary||arbitrary
architechture||architecture architechture||architecture
archtecture||architecture archtecture||architecture
@ -185,10 +187,12 @@ assotiated||associated
asssert||assert asssert||assert
assum||assume assum||assume
assumtpion||assumption assumtpion||assumption
asume||assume
asuming||assuming asuming||assuming
asycronous||asynchronous asycronous||asynchronous
asychronous||asynchronous asychronous||asynchronous
asynchnous||asynchronous asynchnous||asynchronous
asynchrnous||asynchronous
asynchronus||asynchronous asynchronus||asynchronous
asynchromous||asynchronous asynchromous||asynchronous
asymetric||asymmetric asymetric||asymmetric
@ -269,6 +273,7 @@ caculate||calculate
caculation||calculation caculation||calculation
cadidate||candidate cadidate||candidate
cahces||caches cahces||caches
calcluate||calculate
calender||calendar calender||calendar
calescing||coalescing calescing||coalescing
calibraiton||calibration calibraiton||calibration
@ -331,6 +336,7 @@ chouse||chose
circumvernt||circumvent circumvernt||circumvent
claread||cleared claread||cleared
clared||cleared clared||cleared
clearify||clarify
closeing||closing closeing||closing
clustred||clustered clustred||clustered
cnfiguration||configuration cnfiguration||configuration
@ -379,12 +385,14 @@ comsumed||consumed
comunicate||communicate comunicate||communicate
comunication||communication comunication||communication
conbination||combination conbination||combination
concurent||concurrent
conditionaly||conditionally conditionaly||conditionally
conditon||condition conditon||condition
condtion||condition condtion||condition
condtional||conditional condtional||conditional
conected||connected conected||connected
conector||connector conector||connector
configed||configured
configration||configuration configration||configuration
configred||configured configred||configured
configuartion||configuration configuartion||configuration
@ -394,6 +402,7 @@ configuratoin||configuration
configuraton||configuration configuraton||configuration
configuretion||configuration configuretion||configuration
configutation||configuration configutation||configuration
congiuration||configuration
conider||consider conider||consider
conjuction||conjunction conjuction||conjunction
connecetd||connected connecetd||connected
@ -403,6 +412,7 @@ connnection||connection
connnections||connections connnections||connections
consistancy||consistency consistancy||consistency
consistant||consistent consistant||consistent
consits||consists
containes||contains containes||contains
containts||contains containts||contains
contaisn||contains contaisn||contains
@ -452,6 +462,7 @@ decendants||descendants
decompres||decompress decompres||decompress
decsribed||described decsribed||described
decription||description decription||description
detault||default
dectected||detected dectected||detected
defailt||default defailt||default
deferal||deferral deferal||deferral
@ -487,6 +498,7 @@ depreacte||deprecate
desactivate||deactivate desactivate||deactivate
desciptor||descriptor desciptor||descriptor
desciptors||descriptors desciptors||descriptors
descritpor||descriptor
descripto||descriptor descripto||descriptor
descripton||description descripton||description
descrition||description descrition||description
@ -601,6 +613,7 @@ enchanced||enhanced
encorporating||incorporating encorporating||incorporating
encrupted||encrypted encrupted||encrypted
encrypiton||encryption encrypiton||encryption
encryped||encrypted
encryptio||encryption encryptio||encryption
endianess||endianness endianess||endianness
enpoint||endpoint enpoint||endpoint
@ -630,6 +643,7 @@ etsbalishment||establishment
evalute||evaluate evalute||evaluate
evalutes||evaluates evalutes||evaluates
evalution||evaluation evalution||evaluation
evaulated||evaluated
excecutable||executable excecutable||executable
excceed||exceed excceed||exceed
exceded||exceeded exceded||exceeded
@ -650,6 +664,7 @@ exlcude||exclude
exlcuding||excluding exlcuding||excluding
exlcusive||exclusive exlcusive||exclusive
exlusive||exclusive exlusive||exclusive
exlicitly||explicitly
exmaple||example exmaple||example
expecially||especially expecially||especially
experies||expires experies||expires
@ -659,6 +674,7 @@ explict||explicit
explictely||explicitly explictely||explicitly
explictly||explicitly explictly||explicitly
expresion||expression expresion||expression
exprienced||experienced
exprimental||experimental exprimental||experimental
extened||extended extened||extended
exteneded||extended exteneded||extended
@ -834,6 +850,7 @@ informations||information
informtion||information informtion||information
infromation||information infromation||information
ingore||ignore ingore||ignore
inheritence||inheritance
inital||initial inital||initial
initalized||initialized initalized||initialized
initalised||initialized initalised||initialized
@ -878,6 +895,7 @@ interoprability||interoperability
interuupt||interrupt interuupt||interrupt
interupt||interrupt interupt||interrupt
interupts||interrupts interupts||interrupts
interurpt||interrupt
interrface||interface interrface||interface
interrrupt||interrupt interrrupt||interrupt
interrup||interrupt interrup||interrupt
@ -925,6 +943,7 @@ jumpimng||jumping
juse||just juse||just
jus||just jus||just
kown||known kown||known
lable||label
langage||language langage||language
langauage||language langauage||language
langauge||language langauge||language
@ -995,6 +1014,7 @@ metdata||metadata
micropone||microphone micropone||microphone
microprocesspr||microprocessor microprocesspr||microprocessor
migrateable||migratable migrateable||migratable
miliseconds||milliseconds
millenium||millennium millenium||millennium
milliseonds||milliseconds milliseonds||milliseconds
minimim||minimum minimim||minimum
@ -1132,6 +1152,7 @@ palne||plane
paramameters||parameters paramameters||parameters
paramaters||parameters paramaters||parameters
paramater||parameter paramater||parameter
paramenters||parameters
parametes||parameters parametes||parameters
parametised||parametrised parametised||parametrised
paramter||parameter paramter||parameter
@ -1177,9 +1198,11 @@ poiter||pointer
posible||possible posible||possible
positon||position positon||position
possibilites||possibilities possibilites||possibilities
postion||position
potocol||protocol potocol||protocol
powerfull||powerful powerfull||powerful
pramater||parameter pramater||parameter
preambule||preamble
preamle||preamble preamle||preamble
preample||preamble preample||preamble
preapre||prepare preapre||prepare
@ -1269,6 +1292,7 @@ raoming||roaming
reasearcher||researcher reasearcher||researcher
reasearchers||researchers reasearchers||researchers
reasearch||research reasearch||research
recalcualte||recalculate
receieve||receive receieve||receive
recepient||recipient recepient||recipient
recevied||received recevied||received
@ -1291,6 +1315,7 @@ refcounf||refcount
refence||reference refence||reference
refered||referred refered||referred
referenace||reference referenace||reference
refererence||reference
refering||referring refering||referring
refernces||references refernces||references
refernnce||reference refernnce||reference
@ -1315,12 +1340,14 @@ reloade||reload
remoote||remote remoote||remote
remore||remote remore||remote
removeable||removable removeable||removable
repective||respective
repectively||respectively repectively||respectively
replacable||replaceable replacable||replaceable
replacments||replacements replacments||replacements
replys||replies replys||replies
reponse||response reponse||response
representaion||representation representaion||representation
repsonse||response
reqeust||request reqeust||request
reqister||register reqister||register
requed||requeued requed||requeued
@ -1362,6 +1389,7 @@ reuest||request
reuqest||request reuqest||request
reutnred||returned reutnred||returned
revsion||revision revsion||revision
rewritting||rewriting
rmeoved||removed rmeoved||removed
rmeove||remove rmeove||remove
rmeoves||removes rmeoves||removes
@ -1444,6 +1472,7 @@ soluation||solution
souce||source souce||source
speach||speech speach||speech
specfic||specific specfic||specific
specfication||specification
specfield||specified specfield||specified
speciefied||specified speciefied||specified
specifc||specific specifc||specific
@ -1544,6 +1573,7 @@ syncronus||synchronous
syste||system syste||system
sytem||system sytem||system
sythesis||synthesis sythesis||synthesis
tagert||target
taht||that taht||that
tained||tainted tained||tainted
tarffic||traffic tarffic||traffic
@ -1574,6 +1604,7 @@ tiggers||triggers
tiggered||triggered tiggered||triggered
tipically||typically tipically||typically
timeing||timing timeing||timing
timming||timing
timout||timeout timout||timeout
tmis||this tmis||this
toogle||toggle toogle||toggle
@ -1597,8 +1628,10 @@ transision||transition
transistioned||transitioned transistioned||transitioned
transmittd||transmitted transmittd||transmitted
transormed||transformed transormed||transformed
trasaction||transaction
trasfer||transfer trasfer||transfer
trasmission||transmission trasmission||transmission
trasmitter||transmitter
treshold||threshold treshold||threshold
triggerd||triggered triggerd||triggered
trigerred||triggered trigerred||triggered

View File

@ -207,7 +207,7 @@ static void dump_common_audit_data(struct audit_buffer *ab,
BUILD_BUG_ON(sizeof(a->u) > sizeof(void *)*2); BUILD_BUG_ON(sizeof(a->u) > sizeof(void *)*2);
audit_log_format(ab, " pid=%d comm=", task_tgid_nr(current)); audit_log_format(ab, " pid=%d comm=", task_tgid_nr(current));
audit_log_untrustedstring(ab, memcpy(comm, current->comm, sizeof(comm))); audit_log_untrustedstring(ab, get_task_comm(comm, current));
switch (a->type) { switch (a->type) {
case LSM_AUDIT_DATA_NONE: case LSM_AUDIT_DATA_NONE:
@ -302,7 +302,7 @@ static void dump_common_audit_data(struct audit_buffer *ab,
char comm[sizeof(tsk->comm)]; char comm[sizeof(tsk->comm)];
audit_log_format(ab, " opid=%d ocomm=", pid); audit_log_format(ab, " opid=%d ocomm=", pid);
audit_log_untrustedstring(ab, audit_log_untrustedstring(ab,
memcpy(comm, tsk->comm, sizeof(comm))); get_task_comm(comm, tsk));
} }
} }
break; break;

View File

@ -708,7 +708,7 @@ static ssize_t sel_write_checkreqprot(struct file *file, const char __user *buf,
if (new_value) { if (new_value) {
char comm[sizeof(current->comm)]; char comm[sizeof(current->comm)];
memcpy(comm, current->comm, sizeof(comm)); strscpy(comm, current->comm);
pr_err("SELinux: %s (%d) set checkreqprot to 1. This is no longer supported.\n", pr_err("SELinux: %s (%d) set checkreqprot to 1. This is no longer supported.\n",
comm, current->pid); comm, current->pid);
} }

View File

@ -54,6 +54,7 @@ static void add_ref(struct hashmap *map, struct pid_iter_entry *e)
ref = &refs->refs[refs->ref_cnt]; ref = &refs->refs[refs->ref_cnt];
ref->pid = e->pid; ref->pid = e->pid;
memcpy(ref->comm, e->comm, sizeof(ref->comm)); memcpy(ref->comm, e->comm, sizeof(ref->comm));
ref->comm[sizeof(ref->comm) - 1] = '\0';
refs->ref_cnt++; refs->ref_cnt++;
return; return;
@ -77,6 +78,7 @@ static void add_ref(struct hashmap *map, struct pid_iter_entry *e)
ref = &refs->refs[0]; ref = &refs->refs[0];
ref->pid = e->pid; ref->pid = e->pid;
memcpy(ref->comm, e->comm, sizeof(ref->comm)); memcpy(ref->comm, e->comm, sizeof(ref->comm));
ref->comm[sizeof(ref->comm) - 1] = '\0';
refs->ref_cnt = 1; refs->ref_cnt = 1;
refs->has_bpf_cookie = e->has_bpf_cookie; refs->has_bpf_cookie = e->has_bpf_cookie;
refs->bpf_cookie = e->bpf_cookie; refs->bpf_cookie = e->bpf_cookie;

View File

@ -1,6 +1,6 @@
/* SPDX-License-Identifier: GPL-2.0 */ /* SPDX-License-Identifier: GPL-2.0 */
#ifndef _TOOLS_LINUX_COMPILER_H_ #ifndef _TOOLS_LINUX_COMPILER_H_
#error "Please don't include <linux/compiler-gcc.h> directly, include <linux/compiler.h> instead." #error "Please do not include <linux/compiler-gcc.h> directly, include <linux/compiler.h> instead."
#endif #endif
/* /*

View File

@ -1,8 +1,6 @@
// SPDX-License-Identifier: GPL-2.0 // SPDX-License-Identifier: GPL-2.0
#include <linux/kernel.h>
#include <linux/compiler.h> #include <linux/compiler.h>
#include <linux/export.h> #include <linux/export.h>
#include <linux/string.h>
#include <linux/list_sort.h> #include <linux/list_sort.h>
#include <linux/list.h> #include <linux/list.h>

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