2019-05-29 14:17:56 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-only
|
2017-06-29 19:01:41 +00:00
|
|
|
/* binder_alloc.c
|
|
|
|
*
|
|
|
|
* Android IPC Subsystem
|
|
|
|
*
|
|
|
|
* Copyright (C) 2007-2017 Google, Inc.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
|
|
|
|
|
|
|
|
#include <linux/list.h>
|
|
|
|
#include <linux/sched/mm.h>
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/rtmutex.h>
|
|
|
|
#include <linux/rbtree.h>
|
|
|
|
#include <linux/seq_file.h>
|
|
|
|
#include <linux/vmalloc.h>
|
|
|
|
#include <linux/slab.h>
|
|
|
|
#include <linux/sched.h>
|
2017-08-23 15:46:42 +00:00
|
|
|
#include <linux/list_lru.h>
|
2018-08-07 19:57:13 +00:00
|
|
|
#include <linux/ratelimit.h>
|
2018-07-23 21:47:23 +00:00
|
|
|
#include <asm/cacheflush.h>
|
2019-02-08 18:35:14 +00:00
|
|
|
#include <linux/uaccess.h>
|
|
|
|
#include <linux/highmem.h>
|
binder: Don't modify VMA bounds in ->mmap handler
binder_mmap() tries to prevent the creation of overly big binder mappings
by silently truncating the size of the VMA to 4MiB. However, this violates
the API contract of mmap(). If userspace attempts to create a large binder
VMA, and later attempts to unmap that VMA, it will call munmap() on a range
beyond the end of the VMA, which may have been allocated to another VMA in
the meantime. This can lead to userspace memory corruption.
The following sequence of calls leads to a segfault without this commit:
int main(void) {
int binder_fd = open("/dev/binder", O_RDWR);
if (binder_fd == -1) err(1, "open binder");
void *binder_mapping = mmap(NULL, 0x800000UL, PROT_READ, MAP_SHARED,
binder_fd, 0);
if (binder_mapping == MAP_FAILED) err(1, "mmap binder");
void *data_mapping = mmap(NULL, 0x400000UL, PROT_READ|PROT_WRITE,
MAP_PRIVATE|MAP_ANONYMOUS, -1, 0);
if (data_mapping == MAP_FAILED) err(1, "mmap data");
munmap(binder_mapping, 0x800000UL);
*(char*)data_mapping = 1;
return 0;
}
Cc: stable@vger.kernel.org
Signed-off-by: Jann Horn <jannh@google.com>
Acked-by: Todd Kjos <tkjos@google.com>
Acked-by: Christian Brauner <christian.brauner@ubuntu.com>
Link: https://lore.kernel.org/r/20191016150119.154756-1-jannh@google.com
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2019-10-16 15:01:18 +00:00
|
|
|
#include <linux/sizes.h>
|
2017-06-29 19:01:41 +00:00
|
|
|
#include "binder_alloc.h"
|
|
|
|
#include "binder_trace.h"
|
|
|
|
|
2017-08-23 15:46:42 +00:00
|
|
|
struct list_lru binder_alloc_lru;
|
|
|
|
|
2017-06-29 19:01:41 +00:00
|
|
|
static DEFINE_MUTEX(binder_alloc_mmap_lock);
|
|
|
|
|
|
|
|
enum {
|
2018-08-07 19:57:13 +00:00
|
|
|
BINDER_DEBUG_USER_ERROR = 1U << 0,
|
2017-06-29 19:01:41 +00:00
|
|
|
BINDER_DEBUG_OPEN_CLOSE = 1U << 1,
|
|
|
|
BINDER_DEBUG_BUFFER_ALLOC = 1U << 2,
|
|
|
|
BINDER_DEBUG_BUFFER_ALLOC_ASYNC = 1U << 3,
|
|
|
|
};
|
2018-08-07 19:57:13 +00:00
|
|
|
static uint32_t binder_alloc_debug_mask = BINDER_DEBUG_USER_ERROR;
|
2017-06-29 19:01:41 +00:00
|
|
|
|
|
|
|
module_param_named(debug_mask, binder_alloc_debug_mask,
|
|
|
|
uint, 0644);
|
|
|
|
|
|
|
|
#define binder_alloc_debug(mask, x...) \
|
|
|
|
do { \
|
|
|
|
if (binder_alloc_debug_mask & mask) \
|
2018-08-07 19:57:13 +00:00
|
|
|
pr_info_ratelimited(x); \
|
2017-06-29 19:01:41 +00:00
|
|
|
} while (0)
|
|
|
|
|
2017-08-23 15:46:39 +00:00
|
|
|
static struct binder_buffer *binder_buffer_next(struct binder_buffer *buffer)
|
|
|
|
{
|
|
|
|
return list_entry(buffer->entry.next, struct binder_buffer, entry);
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct binder_buffer *binder_buffer_prev(struct binder_buffer *buffer)
|
|
|
|
{
|
|
|
|
return list_entry(buffer->entry.prev, struct binder_buffer, entry);
|
|
|
|
}
|
|
|
|
|
2017-06-29 19:01:41 +00:00
|
|
|
static size_t binder_alloc_buffer_size(struct binder_alloc *alloc,
|
|
|
|
struct binder_buffer *buffer)
|
|
|
|
{
|
|
|
|
if (list_is_last(&buffer->entry, &alloc->buffers))
|
2019-02-08 18:35:20 +00:00
|
|
|
return alloc->buffer + alloc->buffer_size - buffer->user_data;
|
|
|
|
return binder_buffer_next(buffer)->user_data - buffer->user_data;
|
2017-06-29 19:01:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void binder_insert_free_buffer(struct binder_alloc *alloc,
|
|
|
|
struct binder_buffer *new_buffer)
|
|
|
|
{
|
|
|
|
struct rb_node **p = &alloc->free_buffers.rb_node;
|
|
|
|
struct rb_node *parent = NULL;
|
|
|
|
struct binder_buffer *buffer;
|
|
|
|
size_t buffer_size;
|
|
|
|
size_t new_buffer_size;
|
|
|
|
|
|
|
|
BUG_ON(!new_buffer->free);
|
|
|
|
|
|
|
|
new_buffer_size = binder_alloc_buffer_size(alloc, new_buffer);
|
|
|
|
|
|
|
|
binder_alloc_debug(BINDER_DEBUG_BUFFER_ALLOC,
|
|
|
|
"%d: add free buffer, size %zd, at %pK\n",
|
|
|
|
alloc->pid, new_buffer_size, new_buffer);
|
|
|
|
|
|
|
|
while (*p) {
|
|
|
|
parent = *p;
|
|
|
|
buffer = rb_entry(parent, struct binder_buffer, rb_node);
|
|
|
|
BUG_ON(!buffer->free);
|
|
|
|
|
|
|
|
buffer_size = binder_alloc_buffer_size(alloc, buffer);
|
|
|
|
|
|
|
|
if (new_buffer_size < buffer_size)
|
|
|
|
p = &parent->rb_left;
|
|
|
|
else
|
|
|
|
p = &parent->rb_right;
|
|
|
|
}
|
|
|
|
rb_link_node(&new_buffer->rb_node, parent, p);
|
|
|
|
rb_insert_color(&new_buffer->rb_node, &alloc->free_buffers);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void binder_insert_allocated_buffer_locked(
|
|
|
|
struct binder_alloc *alloc, struct binder_buffer *new_buffer)
|
|
|
|
{
|
|
|
|
struct rb_node **p = &alloc->allocated_buffers.rb_node;
|
|
|
|
struct rb_node *parent = NULL;
|
|
|
|
struct binder_buffer *buffer;
|
|
|
|
|
|
|
|
BUG_ON(new_buffer->free);
|
|
|
|
|
|
|
|
while (*p) {
|
|
|
|
parent = *p;
|
|
|
|
buffer = rb_entry(parent, struct binder_buffer, rb_node);
|
|
|
|
BUG_ON(buffer->free);
|
|
|
|
|
2019-02-08 18:35:20 +00:00
|
|
|
if (new_buffer->user_data < buffer->user_data)
|
2017-06-29 19:01:41 +00:00
|
|
|
p = &parent->rb_left;
|
2019-02-08 18:35:20 +00:00
|
|
|
else if (new_buffer->user_data > buffer->user_data)
|
2017-06-29 19:01:41 +00:00
|
|
|
p = &parent->rb_right;
|
|
|
|
else
|
|
|
|
BUG();
|
|
|
|
}
|
|
|
|
rb_link_node(&new_buffer->rb_node, parent, p);
|
|
|
|
rb_insert_color(&new_buffer->rb_node, &alloc->allocated_buffers);
|
|
|
|
}
|
|
|
|
|
2017-06-29 19:01:51 +00:00
|
|
|
static struct binder_buffer *binder_alloc_prepare_to_free_locked(
|
2017-06-29 19:01:41 +00:00
|
|
|
struct binder_alloc *alloc,
|
|
|
|
uintptr_t user_ptr)
|
|
|
|
{
|
|
|
|
struct rb_node *n = alloc->allocated_buffers.rb_node;
|
|
|
|
struct binder_buffer *buffer;
|
2019-02-08 18:35:20 +00:00
|
|
|
void __user *uptr;
|
2017-06-29 19:01:41 +00:00
|
|
|
|
2019-02-08 18:35:20 +00:00
|
|
|
uptr = (void __user *)user_ptr;
|
2017-06-29 19:01:41 +00:00
|
|
|
|
|
|
|
while (n) {
|
|
|
|
buffer = rb_entry(n, struct binder_buffer, rb_node);
|
|
|
|
BUG_ON(buffer->free);
|
|
|
|
|
2019-02-08 18:35:20 +00:00
|
|
|
if (uptr < buffer->user_data)
|
2017-06-29 19:01:41 +00:00
|
|
|
n = n->rb_left;
|
2019-02-08 18:35:20 +00:00
|
|
|
else if (uptr > buffer->user_data)
|
2017-06-29 19:01:41 +00:00
|
|
|
n = n->rb_right;
|
2017-06-29 19:01:51 +00:00
|
|
|
else {
|
|
|
|
/*
|
|
|
|
* Guard against user threads attempting to
|
2018-11-06 23:55:32 +00:00
|
|
|
* free the buffer when in use by kernel or
|
|
|
|
* after it's already been freed.
|
2017-06-29 19:01:51 +00:00
|
|
|
*/
|
2018-11-06 23:55:32 +00:00
|
|
|
if (!buffer->allow_user_free)
|
|
|
|
return ERR_PTR(-EPERM);
|
|
|
|
buffer->allow_user_free = 0;
|
2017-06-29 19:01:41 +00:00
|
|
|
return buffer;
|
2017-06-29 19:01:51 +00:00
|
|
|
}
|
2017-06-29 19:01:41 +00:00
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-09-30 20:12:50 +00:00
|
|
|
* binder_alloc_prepare_to_free() - get buffer given user ptr
|
2017-06-29 19:01:41 +00:00
|
|
|
* @alloc: binder_alloc for this proc
|
|
|
|
* @user_ptr: User pointer to buffer data
|
|
|
|
*
|
|
|
|
* Validate userspace pointer to buffer data and return buffer corresponding to
|
|
|
|
* that user pointer. Search the rb tree for buffer that matches user data
|
|
|
|
* pointer.
|
|
|
|
*
|
|
|
|
* Return: Pointer to buffer or NULL
|
|
|
|
*/
|
2017-06-29 19:01:51 +00:00
|
|
|
struct binder_buffer *binder_alloc_prepare_to_free(struct binder_alloc *alloc,
|
|
|
|
uintptr_t user_ptr)
|
2017-06-29 19:01:41 +00:00
|
|
|
{
|
|
|
|
struct binder_buffer *buffer;
|
|
|
|
|
|
|
|
mutex_lock(&alloc->mutex);
|
2017-06-29 19:01:51 +00:00
|
|
|
buffer = binder_alloc_prepare_to_free_locked(alloc, user_ptr);
|
2017-06-29 19:01:41 +00:00
|
|
|
mutex_unlock(&alloc->mutex);
|
|
|
|
return buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int binder_update_page_range(struct binder_alloc *alloc, int allocate,
|
2019-02-08 18:35:20 +00:00
|
|
|
void __user *start, void __user *end)
|
2017-06-29 19:01:41 +00:00
|
|
|
{
|
2019-02-08 18:35:20 +00:00
|
|
|
void __user *page_addr;
|
2017-06-29 19:01:41 +00:00
|
|
|
unsigned long user_page_addr;
|
2017-08-23 15:46:42 +00:00
|
|
|
struct binder_lru_page *page;
|
2017-09-16 05:11:56 +00:00
|
|
|
struct vm_area_struct *vma = NULL;
|
2017-08-23 15:46:42 +00:00
|
|
|
struct mm_struct *mm = NULL;
|
|
|
|
bool need_mm = false;
|
2017-06-29 19:01:41 +00:00
|
|
|
|
|
|
|
binder_alloc_debug(BINDER_DEBUG_BUFFER_ALLOC,
|
|
|
|
"%d: %s pages %pK-%pK\n", alloc->pid,
|
|
|
|
allocate ? "allocate" : "free", start, end);
|
|
|
|
|
|
|
|
if (end <= start)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
trace_binder_update_page_range(alloc, allocate, start, end);
|
|
|
|
|
2017-08-23 15:46:42 +00:00
|
|
|
if (allocate == 0)
|
|
|
|
goto free_range;
|
|
|
|
|
|
|
|
for (page_addr = start; page_addr < end; page_addr += PAGE_SIZE) {
|
|
|
|
page = &alloc->pages[(page_addr - alloc->buffer) / PAGE_SIZE];
|
|
|
|
if (!page->page_ptr) {
|
|
|
|
need_mm = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-23 15:21:44 +00:00
|
|
|
if (need_mm && mmget_not_zero(alloc->vma_vm_mm))
|
2017-10-21 00:58:58 +00:00
|
|
|
mm = alloc->vma_vm_mm;
|
2017-06-29 19:01:41 +00:00
|
|
|
|
|
|
|
if (mm) {
|
2020-06-09 04:33:25 +00:00
|
|
|
mmap_read_lock(mm);
|
2017-06-29 19:01:41 +00:00
|
|
|
vma = alloc->vma;
|
|
|
|
}
|
|
|
|
|
2017-08-23 15:46:42 +00:00
|
|
|
if (!vma && need_mm) {
|
2018-08-07 19:57:13 +00:00
|
|
|
binder_alloc_debug(BINDER_DEBUG_USER_ERROR,
|
|
|
|
"%d: binder_alloc_buf failed to map pages in userspace, no vma\n",
|
|
|
|
alloc->pid);
|
2017-06-29 19:01:41 +00:00
|
|
|
goto err_no_vma;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (page_addr = start; page_addr < end; page_addr += PAGE_SIZE) {
|
|
|
|
int ret;
|
2017-08-23 15:46:42 +00:00
|
|
|
bool on_lru;
|
2017-08-23 15:46:43 +00:00
|
|
|
size_t index;
|
2017-06-29 19:01:41 +00:00
|
|
|
|
2017-08-23 15:46:43 +00:00
|
|
|
index = (page_addr - alloc->buffer) / PAGE_SIZE;
|
|
|
|
page = &alloc->pages[index];
|
2017-06-29 19:01:41 +00:00
|
|
|
|
2017-08-23 15:46:42 +00:00
|
|
|
if (page->page_ptr) {
|
2017-08-23 15:46:43 +00:00
|
|
|
trace_binder_alloc_lru_start(alloc, index);
|
|
|
|
|
list_lru: allow explicit memcg and NUMA node selection
JIRA: https://issues.redhat.com/browse/RHEL-40684
Conflicts:
* include/linux/list_lru.h: minor context differences due to missing
upstream v6.8 commit 7679e14098c9 ("mm: list_lru: Update kernel
documentation to follow the requirements")
This patch is a backport of the following upstream commit:
commit 0a97c01cd20bb96359d8c9dedad92a061ed34e0b
Author: Nhat Pham <nphamcs@gmail.com>
Date: Thu Nov 30 11:40:18 2023 -0800
list_lru: allow explicit memcg and NUMA node selection
Patch series "workload-specific and memory pressure-driven zswap
writeback", v8.
There are currently several issues with zswap writeback:
1. There is only a single global LRU for zswap, making it impossible to
perform worload-specific shrinking - an memcg under memory pressure
cannot determine which pages in the pool it owns, and often ends up
writing pages from other memcgs. This issue has been previously
observed in practice and mitigated by simply disabling
memcg-initiated shrinking:
https://lore.kernel.org/all/20230530232435.3097106-1-nphamcs@gmail.com/T/#u
But this solution leaves a lot to be desired, as we still do not
have an avenue for an memcg to free up its own memory locked up in
the zswap pool.
2. We only shrink the zswap pool when the user-defined limit is hit.
This means that if we set the limit too high, cold data that are
unlikely to be used again will reside in the pool, wasting precious
memory. It is hard to predict how much zswap space will be needed
ahead of time, as this depends on the workload (specifically, on
factors such as memory access patterns and compressibility of the
memory pages).
This patch series solves these issues by separating the global zswap LRU
into per-memcg and per-NUMA LRUs, and performs workload-specific (i.e
memcg- and NUMA-aware) zswap writeback under memory pressure. The new
shrinker does not have any parameter that must be tuned by the user, and
can be opted in or out on a per-memcg basis.
As a proof of concept, we ran the following synthetic benchmark: build the
linux kernel in a memory-limited cgroup, and allocate some cold data in
tmpfs to see if the shrinker could write them out and improved the overall
performance. Depending on the amount of cold data generated, we observe
from 14% to 35% reduction in kernel CPU time used in the kernel builds.
This patch (of 6):
The interface of list_lru is based on the assumption that the list node
and the data it represents belong to the same allocated on the correct
node/memcg. While this assumption is valid for existing slab objects LRU
such as dentries and inodes, it is undocumented, and rather inflexible for
certain potential list_lru users (such as the upcoming zswap shrinker and
the THP shrinker). It has caused us a lot of issues during our
development.
This patch changes list_lru interface so that the caller must explicitly
specify numa node and memcg when adding and removing objects. The old
list_lru_add() and list_lru_del() are renamed to list_lru_add_obj() and
list_lru_del_obj(), respectively.
It also extends the list_lru API with a new function, list_lru_putback,
which undoes a previous list_lru_isolate call. Unlike list_lru_add, it
does not increment the LRU node count (as list_lru_isolate does not
decrement the node count). list_lru_putback also allows for explicit
memcg and NUMA node selection.
Link: https://lkml.kernel.org/r/20231130194023.4102148-1-nphamcs@gmail.com
Link: https://lkml.kernel.org/r/20231130194023.4102148-2-nphamcs@gmail.com
Signed-off-by: Nhat Pham <nphamcs@gmail.com>
Suggested-by: Johannes Weiner <hannes@cmpxchg.org>
Acked-by: Johannes Weiner <hannes@cmpxchg.org>
Tested-by: Bagas Sanjaya <bagasdotme@gmail.com>
Cc: Chris Li <chrisl@kernel.org>
Cc: Dan Streetman <ddstreet@ieee.org>
Cc: Domenico Cerasuolo <cerasuolodomenico@gmail.com>
Cc: Michal Hocko <mhocko@kernel.org>
Cc: Muchun Song <muchun.song@linux.dev>
Cc: Roman Gushchin <roman.gushchin@linux.dev>
Cc: Seth Jennings <sjenning@redhat.com>
Cc: Shakeel Butt <shakeelb@google.com>
Cc: Shuah Khan <shuah@kernel.org>
Cc: Vitaly Wool <vitaly.wool@konsulko.com>
Cc: Yosry Ahmed <yosryahmed@google.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Rafael Aquini <aquini@redhat.com>
2024-06-28 16:23:39 +00:00
|
|
|
on_lru = list_lru_del_obj(&binder_alloc_lru, &page->lru);
|
2017-08-23 15:46:42 +00:00
|
|
|
WARN_ON(!on_lru);
|
2017-08-23 15:46:43 +00:00
|
|
|
|
|
|
|
trace_binder_alloc_lru_end(alloc, index);
|
2017-08-23 15:46:42 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (WARN_ON(!vma))
|
|
|
|
goto err_page_ptr_cleared;
|
|
|
|
|
2017-08-23 15:46:43 +00:00
|
|
|
trace_binder_alloc_page_start(alloc, index);
|
2017-08-23 15:46:42 +00:00
|
|
|
page->page_ptr = alloc_page(GFP_KERNEL |
|
|
|
|
__GFP_HIGHMEM |
|
|
|
|
__GFP_ZERO);
|
|
|
|
if (!page->page_ptr) {
|
2017-06-29 19:01:41 +00:00
|
|
|
pr_err("%d: binder_alloc_buf failed for page at %pK\n",
|
|
|
|
alloc->pid, page_addr);
|
|
|
|
goto err_alloc_page_failed;
|
|
|
|
}
|
2017-08-23 15:46:42 +00:00
|
|
|
page->alloc = alloc;
|
|
|
|
INIT_LIST_HEAD(&page->lru);
|
|
|
|
|
2019-02-08 18:35:19 +00:00
|
|
|
user_page_addr = (uintptr_t)page_addr;
|
2017-08-23 15:46:42 +00:00
|
|
|
ret = vm_insert_page(vma, user_page_addr, page[0].page_ptr);
|
2017-06-29 19:01:41 +00:00
|
|
|
if (ret) {
|
|
|
|
pr_err("%d: binder_alloc_buf failed to map page at %lx in userspace\n",
|
|
|
|
alloc->pid, user_page_addr);
|
|
|
|
goto err_vm_insert_page_failed;
|
|
|
|
}
|
2017-08-23 15:46:43 +00:00
|
|
|
|
2017-11-13 09:06:56 +00:00
|
|
|
if (index + 1 > alloc->pages_high)
|
|
|
|
alloc->pages_high = index + 1;
|
|
|
|
|
2017-08-23 15:46:43 +00:00
|
|
|
trace_binder_alloc_page_end(alloc, index);
|
2017-06-29 19:01:41 +00:00
|
|
|
}
|
|
|
|
if (mm) {
|
2020-06-09 04:33:25 +00:00
|
|
|
mmap_read_unlock(mm);
|
2017-06-29 19:01:41 +00:00
|
|
|
mmput(mm);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
free_range:
|
2019-10-18 20:56:31 +00:00
|
|
|
for (page_addr = end - PAGE_SIZE; 1; page_addr -= PAGE_SIZE) {
|
2017-08-23 15:46:42 +00:00
|
|
|
bool ret;
|
2017-08-23 15:46:43 +00:00
|
|
|
size_t index;
|
2017-08-23 15:46:42 +00:00
|
|
|
|
2017-08-23 15:46:43 +00:00
|
|
|
index = (page_addr - alloc->buffer) / PAGE_SIZE;
|
|
|
|
page = &alloc->pages[index];
|
|
|
|
|
|
|
|
trace_binder_free_lru_start(alloc, index);
|
2017-08-23 15:46:42 +00:00
|
|
|
|
list_lru: allow explicit memcg and NUMA node selection
JIRA: https://issues.redhat.com/browse/RHEL-40684
Conflicts:
* include/linux/list_lru.h: minor context differences due to missing
upstream v6.8 commit 7679e14098c9 ("mm: list_lru: Update kernel
documentation to follow the requirements")
This patch is a backport of the following upstream commit:
commit 0a97c01cd20bb96359d8c9dedad92a061ed34e0b
Author: Nhat Pham <nphamcs@gmail.com>
Date: Thu Nov 30 11:40:18 2023 -0800
list_lru: allow explicit memcg and NUMA node selection
Patch series "workload-specific and memory pressure-driven zswap
writeback", v8.
There are currently several issues with zswap writeback:
1. There is only a single global LRU for zswap, making it impossible to
perform worload-specific shrinking - an memcg under memory pressure
cannot determine which pages in the pool it owns, and often ends up
writing pages from other memcgs. This issue has been previously
observed in practice and mitigated by simply disabling
memcg-initiated shrinking:
https://lore.kernel.org/all/20230530232435.3097106-1-nphamcs@gmail.com/T/#u
But this solution leaves a lot to be desired, as we still do not
have an avenue for an memcg to free up its own memory locked up in
the zswap pool.
2. We only shrink the zswap pool when the user-defined limit is hit.
This means that if we set the limit too high, cold data that are
unlikely to be used again will reside in the pool, wasting precious
memory. It is hard to predict how much zswap space will be needed
ahead of time, as this depends on the workload (specifically, on
factors such as memory access patterns and compressibility of the
memory pages).
This patch series solves these issues by separating the global zswap LRU
into per-memcg and per-NUMA LRUs, and performs workload-specific (i.e
memcg- and NUMA-aware) zswap writeback under memory pressure. The new
shrinker does not have any parameter that must be tuned by the user, and
can be opted in or out on a per-memcg basis.
As a proof of concept, we ran the following synthetic benchmark: build the
linux kernel in a memory-limited cgroup, and allocate some cold data in
tmpfs to see if the shrinker could write them out and improved the overall
performance. Depending on the amount of cold data generated, we observe
from 14% to 35% reduction in kernel CPU time used in the kernel builds.
This patch (of 6):
The interface of list_lru is based on the assumption that the list node
and the data it represents belong to the same allocated on the correct
node/memcg. While this assumption is valid for existing slab objects LRU
such as dentries and inodes, it is undocumented, and rather inflexible for
certain potential list_lru users (such as the upcoming zswap shrinker and
the THP shrinker). It has caused us a lot of issues during our
development.
This patch changes list_lru interface so that the caller must explicitly
specify numa node and memcg when adding and removing objects. The old
list_lru_add() and list_lru_del() are renamed to list_lru_add_obj() and
list_lru_del_obj(), respectively.
It also extends the list_lru API with a new function, list_lru_putback,
which undoes a previous list_lru_isolate call. Unlike list_lru_add, it
does not increment the LRU node count (as list_lru_isolate does not
decrement the node count). list_lru_putback also allows for explicit
memcg and NUMA node selection.
Link: https://lkml.kernel.org/r/20231130194023.4102148-1-nphamcs@gmail.com
Link: https://lkml.kernel.org/r/20231130194023.4102148-2-nphamcs@gmail.com
Signed-off-by: Nhat Pham <nphamcs@gmail.com>
Suggested-by: Johannes Weiner <hannes@cmpxchg.org>
Acked-by: Johannes Weiner <hannes@cmpxchg.org>
Tested-by: Bagas Sanjaya <bagasdotme@gmail.com>
Cc: Chris Li <chrisl@kernel.org>
Cc: Dan Streetman <ddstreet@ieee.org>
Cc: Domenico Cerasuolo <cerasuolodomenico@gmail.com>
Cc: Michal Hocko <mhocko@kernel.org>
Cc: Muchun Song <muchun.song@linux.dev>
Cc: Roman Gushchin <roman.gushchin@linux.dev>
Cc: Seth Jennings <sjenning@redhat.com>
Cc: Shakeel Butt <shakeelb@google.com>
Cc: Shuah Khan <shuah@kernel.org>
Cc: Vitaly Wool <vitaly.wool@konsulko.com>
Cc: Yosry Ahmed <yosryahmed@google.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Rafael Aquini <aquini@redhat.com>
2024-06-28 16:23:39 +00:00
|
|
|
ret = list_lru_add_obj(&binder_alloc_lru, &page->lru);
|
2017-08-23 15:46:42 +00:00
|
|
|
WARN_ON(!ret);
|
2017-08-23 15:46:43 +00:00
|
|
|
|
|
|
|
trace_binder_free_lru_end(alloc, index);
|
2019-10-18 20:56:31 +00:00
|
|
|
if (page_addr == start)
|
|
|
|
break;
|
2017-08-23 15:46:42 +00:00
|
|
|
continue;
|
|
|
|
|
2017-06-29 19:01:41 +00:00
|
|
|
err_vm_insert_page_failed:
|
2017-08-23 15:46:42 +00:00
|
|
|
__free_page(page->page_ptr);
|
|
|
|
page->page_ptr = NULL;
|
2017-06-29 19:01:41 +00:00
|
|
|
err_alloc_page_failed:
|
2017-08-23 15:46:42 +00:00
|
|
|
err_page_ptr_cleared:
|
2019-10-18 20:56:31 +00:00
|
|
|
if (page_addr == start)
|
|
|
|
break;
|
2017-06-29 19:01:41 +00:00
|
|
|
}
|
|
|
|
err_no_vma:
|
|
|
|
if (mm) {
|
2020-06-09 04:33:25 +00:00
|
|
|
mmap_read_unlock(mm);
|
2017-06-29 19:01:41 +00:00
|
|
|
mmput(mm);
|
|
|
|
}
|
2017-06-29 19:01:46 +00:00
|
|
|
return vma ? -ENOMEM : -ESRCH;
|
2017-06-29 19:01:41 +00:00
|
|
|
}
|
|
|
|
|
2018-08-23 05:29:56 +00:00
|
|
|
|
|
|
|
static inline void binder_alloc_set_vma(struct binder_alloc *alloc,
|
|
|
|
struct vm_area_struct *vma)
|
|
|
|
{
|
|
|
|
if (vma)
|
|
|
|
alloc->vma_vm_mm = vma->vm_mm;
|
|
|
|
/*
|
|
|
|
* If we see alloc->vma is not NULL, buffer data structures set up
|
|
|
|
* completely. Look at smp_rmb side binder_alloc_get_vma.
|
|
|
|
* We also want to guarantee new alloc->vma_vm_mm is always visible
|
|
|
|
* if alloc->vma is set.
|
|
|
|
*/
|
|
|
|
smp_wmb();
|
|
|
|
alloc->vma = vma;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline struct vm_area_struct *binder_alloc_get_vma(
|
|
|
|
struct binder_alloc *alloc)
|
|
|
|
{
|
|
|
|
struct vm_area_struct *vma = NULL;
|
|
|
|
|
|
|
|
if (alloc->vma) {
|
|
|
|
/* Look at description in binder_alloc_set_vma */
|
|
|
|
smp_rmb();
|
|
|
|
vma = alloc->vma;
|
|
|
|
}
|
|
|
|
return vma;
|
|
|
|
}
|
|
|
|
|
2021-04-09 09:40:46 +00:00
|
|
|
static bool debug_low_async_space_locked(struct binder_alloc *alloc, int pid)
|
2020-08-21 12:25:44 +00:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
* Find the amount and size of buffers allocated by the current caller;
|
|
|
|
* The idea is that once we cross the threshold, whoever is responsible
|
|
|
|
* for the low async space is likely to try to send another async txn,
|
|
|
|
* and at some point we'll catch them in the act. This is more efficient
|
|
|
|
* than keeping a map per pid.
|
|
|
|
*/
|
2020-09-10 15:12:21 +00:00
|
|
|
struct rb_node *n;
|
2020-08-21 12:25:44 +00:00
|
|
|
struct binder_buffer *buffer;
|
|
|
|
size_t total_alloc_size = 0;
|
|
|
|
size_t num_buffers = 0;
|
|
|
|
|
|
|
|
for (n = rb_first(&alloc->allocated_buffers); n != NULL;
|
|
|
|
n = rb_next(n)) {
|
|
|
|
buffer = rb_entry(n, struct binder_buffer, rb_node);
|
|
|
|
if (buffer->pid != pid)
|
|
|
|
continue;
|
|
|
|
if (!buffer->async_transaction)
|
|
|
|
continue;
|
|
|
|
total_alloc_size += binder_alloc_buffer_size(alloc, buffer)
|
|
|
|
+ sizeof(struct binder_buffer);
|
|
|
|
num_buffers++;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Warn if this pid has more than 50 transactions, or more than 50% of
|
2021-04-09 09:40:46 +00:00
|
|
|
* async space (which is 25% of total buffer size). Oneway spam is only
|
|
|
|
* detected when the threshold is exceeded.
|
2020-08-21 12:25:44 +00:00
|
|
|
*/
|
|
|
|
if (num_buffers > 50 || total_alloc_size > alloc->buffer_size / 4) {
|
|
|
|
binder_alloc_debug(BINDER_DEBUG_USER_ERROR,
|
|
|
|
"%d: pid %d spamming oneway? %zd buffers allocated for a total size of %zd\n",
|
|
|
|
alloc->pid, pid, num_buffers, total_alloc_size);
|
2021-04-09 09:40:46 +00:00
|
|
|
if (!alloc->oneway_spam_detected) {
|
|
|
|
alloc->oneway_spam_detected = true;
|
|
|
|
return true;
|
|
|
|
}
|
2020-08-21 12:25:44 +00:00
|
|
|
}
|
2021-04-09 09:40:46 +00:00
|
|
|
return false;
|
2020-08-21 12:25:44 +00:00
|
|
|
}
|
|
|
|
|
2017-12-14 04:15:42 +00:00
|
|
|
static struct binder_buffer *binder_alloc_new_buf_locked(
|
|
|
|
struct binder_alloc *alloc,
|
|
|
|
size_t data_size,
|
|
|
|
size_t offsets_size,
|
|
|
|
size_t extra_buffers_size,
|
2020-08-21 12:25:44 +00:00
|
|
|
int is_async,
|
|
|
|
int pid)
|
2017-06-29 19:01:41 +00:00
|
|
|
{
|
|
|
|
struct rb_node *n = alloc->free_buffers.rb_node;
|
|
|
|
struct binder_buffer *buffer;
|
|
|
|
size_t buffer_size;
|
|
|
|
struct rb_node *best_fit = NULL;
|
2019-02-08 18:35:20 +00:00
|
|
|
void __user *has_page_addr;
|
|
|
|
void __user *end_page_addr;
|
2017-06-29 19:01:41 +00:00
|
|
|
size_t size, data_offsets_size;
|
2017-06-29 19:01:46 +00:00
|
|
|
int ret;
|
2017-06-29 19:01:41 +00:00
|
|
|
|
2018-08-23 05:29:56 +00:00
|
|
|
if (!binder_alloc_get_vma(alloc)) {
|
2018-08-07 19:57:13 +00:00
|
|
|
binder_alloc_debug(BINDER_DEBUG_USER_ERROR,
|
|
|
|
"%d: binder_alloc_buf, no vma\n",
|
|
|
|
alloc->pid);
|
2017-06-29 19:01:46 +00:00
|
|
|
return ERR_PTR(-ESRCH);
|
2017-06-29 19:01:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
data_offsets_size = ALIGN(data_size, sizeof(void *)) +
|
|
|
|
ALIGN(offsets_size, sizeof(void *));
|
|
|
|
|
|
|
|
if (data_offsets_size < data_size || data_offsets_size < offsets_size) {
|
|
|
|
binder_alloc_debug(BINDER_DEBUG_BUFFER_ALLOC,
|
|
|
|
"%d: got transaction with invalid size %zd-%zd\n",
|
|
|
|
alloc->pid, data_size, offsets_size);
|
2017-06-29 19:01:46 +00:00
|
|
|
return ERR_PTR(-EINVAL);
|
2017-06-29 19:01:41 +00:00
|
|
|
}
|
|
|
|
size = data_offsets_size + ALIGN(extra_buffers_size, sizeof(void *));
|
|
|
|
if (size < data_offsets_size || size < extra_buffers_size) {
|
|
|
|
binder_alloc_debug(BINDER_DEBUG_BUFFER_ALLOC,
|
|
|
|
"%d: got transaction with invalid extra_buffers_size %zd\n",
|
|
|
|
alloc->pid, extra_buffers_size);
|
2017-06-29 19:01:46 +00:00
|
|
|
return ERR_PTR(-EINVAL);
|
2017-06-29 19:01:41 +00:00
|
|
|
}
|
|
|
|
if (is_async &&
|
|
|
|
alloc->free_async_space < size + sizeof(struct binder_buffer)) {
|
|
|
|
binder_alloc_debug(BINDER_DEBUG_BUFFER_ALLOC,
|
|
|
|
"%d: binder_alloc_buf size %zd failed, no async space left\n",
|
|
|
|
alloc->pid, size);
|
2017-06-29 19:01:46 +00:00
|
|
|
return ERR_PTR(-ENOSPC);
|
2017-06-29 19:01:41 +00:00
|
|
|
}
|
|
|
|
|
2017-08-23 15:46:41 +00:00
|
|
|
/* Pad 0-size buffers so they get assigned unique addresses */
|
|
|
|
size = max(size, sizeof(void *));
|
|
|
|
|
2017-06-29 19:01:41 +00:00
|
|
|
while (n) {
|
|
|
|
buffer = rb_entry(n, struct binder_buffer, rb_node);
|
|
|
|
BUG_ON(!buffer->free);
|
|
|
|
buffer_size = binder_alloc_buffer_size(alloc, buffer);
|
|
|
|
|
|
|
|
if (size < buffer_size) {
|
|
|
|
best_fit = n;
|
|
|
|
n = n->rb_left;
|
|
|
|
} else if (size > buffer_size)
|
|
|
|
n = n->rb_right;
|
|
|
|
else {
|
|
|
|
best_fit = n;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (best_fit == NULL) {
|
2017-06-29 19:01:52 +00:00
|
|
|
size_t allocated_buffers = 0;
|
|
|
|
size_t largest_alloc_size = 0;
|
|
|
|
size_t total_alloc_size = 0;
|
|
|
|
size_t free_buffers = 0;
|
|
|
|
size_t largest_free_size = 0;
|
|
|
|
size_t total_free_size = 0;
|
|
|
|
|
|
|
|
for (n = rb_first(&alloc->allocated_buffers); n != NULL;
|
|
|
|
n = rb_next(n)) {
|
|
|
|
buffer = rb_entry(n, struct binder_buffer, rb_node);
|
|
|
|
buffer_size = binder_alloc_buffer_size(alloc, buffer);
|
|
|
|
allocated_buffers++;
|
|
|
|
total_alloc_size += buffer_size;
|
|
|
|
if (buffer_size > largest_alloc_size)
|
|
|
|
largest_alloc_size = buffer_size;
|
|
|
|
}
|
|
|
|
for (n = rb_first(&alloc->free_buffers); n != NULL;
|
|
|
|
n = rb_next(n)) {
|
|
|
|
buffer = rb_entry(n, struct binder_buffer, rb_node);
|
|
|
|
buffer_size = binder_alloc_buffer_size(alloc, buffer);
|
|
|
|
free_buffers++;
|
|
|
|
total_free_size += buffer_size;
|
|
|
|
if (buffer_size > largest_free_size)
|
|
|
|
largest_free_size = buffer_size;
|
|
|
|
}
|
2018-08-07 19:57:13 +00:00
|
|
|
binder_alloc_debug(BINDER_DEBUG_USER_ERROR,
|
|
|
|
"%d: binder_alloc_buf size %zd failed, no address space\n",
|
|
|
|
alloc->pid, size);
|
|
|
|
binder_alloc_debug(BINDER_DEBUG_USER_ERROR,
|
|
|
|
"allocated: %zd (num: %zd largest: %zd), free: %zd (num: %zd largest: %zd)\n",
|
|
|
|
total_alloc_size, allocated_buffers,
|
|
|
|
largest_alloc_size, total_free_size,
|
|
|
|
free_buffers, largest_free_size);
|
2017-06-29 19:01:46 +00:00
|
|
|
return ERR_PTR(-ENOSPC);
|
2017-06-29 19:01:41 +00:00
|
|
|
}
|
|
|
|
if (n == NULL) {
|
|
|
|
buffer = rb_entry(best_fit, struct binder_buffer, rb_node);
|
|
|
|
buffer_size = binder_alloc_buffer_size(alloc, buffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
binder_alloc_debug(BINDER_DEBUG_BUFFER_ALLOC,
|
|
|
|
"%d: binder_alloc_buf size %zd got buffer %pK size %zd\n",
|
|
|
|
alloc->pid, size, buffer, buffer_size);
|
|
|
|
|
2019-02-08 18:35:20 +00:00
|
|
|
has_page_addr = (void __user *)
|
|
|
|
(((uintptr_t)buffer->user_data + buffer_size) & PAGE_MASK);
|
2017-08-23 15:46:41 +00:00
|
|
|
WARN_ON(n && buffer_size != size);
|
2017-06-29 19:01:41 +00:00
|
|
|
end_page_addr =
|
2019-02-08 18:35:20 +00:00
|
|
|
(void __user *)PAGE_ALIGN((uintptr_t)buffer->user_data + size);
|
2017-06-29 19:01:41 +00:00
|
|
|
if (end_page_addr > has_page_addr)
|
|
|
|
end_page_addr = has_page_addr;
|
2019-02-08 18:35:20 +00:00
|
|
|
ret = binder_update_page_range(alloc, 1, (void __user *)
|
|
|
|
PAGE_ALIGN((uintptr_t)buffer->user_data), end_page_addr);
|
2017-06-29 19:01:46 +00:00
|
|
|
if (ret)
|
|
|
|
return ERR_PTR(ret);
|
2017-06-29 19:01:41 +00:00
|
|
|
|
|
|
|
if (buffer_size != size) {
|
2017-08-23 15:46:41 +00:00
|
|
|
struct binder_buffer *new_buffer;
|
2017-06-29 19:01:41 +00:00
|
|
|
|
2017-08-23 15:46:41 +00:00
|
|
|
new_buffer = kzalloc(sizeof(*buffer), GFP_KERNEL);
|
|
|
|
if (!new_buffer) {
|
|
|
|
pr_err("%s: %d failed to alloc new buffer struct\n",
|
|
|
|
__func__, alloc->pid);
|
|
|
|
goto err_alloc_buf_struct_failed;
|
|
|
|
}
|
2019-02-08 18:35:20 +00:00
|
|
|
new_buffer->user_data = (u8 __user *)buffer->user_data + size;
|
2017-06-29 19:01:41 +00:00
|
|
|
list_add(&new_buffer->entry, &buffer->entry);
|
|
|
|
new_buffer->free = 1;
|
|
|
|
binder_insert_free_buffer(alloc, new_buffer);
|
|
|
|
}
|
2017-08-23 15:46:41 +00:00
|
|
|
|
|
|
|
rb_erase(best_fit, &alloc->free_buffers);
|
|
|
|
buffer->free = 0;
|
2018-11-06 23:55:32 +00:00
|
|
|
buffer->allow_user_free = 0;
|
2017-08-23 15:46:41 +00:00
|
|
|
binder_insert_allocated_buffer_locked(alloc, buffer);
|
2017-06-29 19:01:41 +00:00
|
|
|
binder_alloc_debug(BINDER_DEBUG_BUFFER_ALLOC,
|
|
|
|
"%d: binder_alloc_buf size %zd got %pK\n",
|
|
|
|
alloc->pid, size, buffer);
|
|
|
|
buffer->data_size = data_size;
|
|
|
|
buffer->offsets_size = offsets_size;
|
|
|
|
buffer->async_transaction = is_async;
|
|
|
|
buffer->extra_buffers_size = extra_buffers_size;
|
2020-08-21 12:25:44 +00:00
|
|
|
buffer->pid = pid;
|
2021-04-09 09:40:46 +00:00
|
|
|
buffer->oneway_spam_suspect = false;
|
2017-06-29 19:01:41 +00:00
|
|
|
if (is_async) {
|
|
|
|
alloc->free_async_space -= size + sizeof(struct binder_buffer);
|
|
|
|
binder_alloc_debug(BINDER_DEBUG_BUFFER_ALLOC_ASYNC,
|
|
|
|
"%d: binder_alloc_buf size %zd async free %zd\n",
|
|
|
|
alloc->pid, size, alloc->free_async_space);
|
2020-08-21 12:25:44 +00:00
|
|
|
if (alloc->free_async_space < alloc->buffer_size / 10) {
|
|
|
|
/*
|
|
|
|
* Start detecting spammers once we have less than 20%
|
|
|
|
* of async space left (which is less than 10% of total
|
|
|
|
* buffer size).
|
|
|
|
*/
|
2021-04-09 09:40:46 +00:00
|
|
|
buffer->oneway_spam_suspect = debug_low_async_space_locked(alloc, pid);
|
|
|
|
} else {
|
|
|
|
alloc->oneway_spam_detected = false;
|
2020-08-21 12:25:44 +00:00
|
|
|
}
|
2017-06-29 19:01:41 +00:00
|
|
|
}
|
|
|
|
return buffer;
|
2017-08-23 15:46:41 +00:00
|
|
|
|
|
|
|
err_alloc_buf_struct_failed:
|
2019-02-08 18:35:20 +00:00
|
|
|
binder_update_page_range(alloc, 0, (void __user *)
|
|
|
|
PAGE_ALIGN((uintptr_t)buffer->user_data),
|
2017-09-16 05:11:56 +00:00
|
|
|
end_page_addr);
|
2017-08-23 15:46:41 +00:00
|
|
|
return ERR_PTR(-ENOMEM);
|
2017-06-29 19:01:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* binder_alloc_new_buf() - Allocate a new binder buffer
|
|
|
|
* @alloc: binder_alloc for this proc
|
|
|
|
* @data_size: size of user data buffer
|
|
|
|
* @offsets_size: user specified buffer offset
|
|
|
|
* @extra_buffers_size: size of extra space for meta-data (eg, security context)
|
|
|
|
* @is_async: buffer for async transaction
|
2020-08-21 12:25:44 +00:00
|
|
|
* @pid: pid to attribute allocation to (used for debugging)
|
2017-06-29 19:01:41 +00:00
|
|
|
*
|
|
|
|
* Allocate a new buffer given the requested sizes. Returns
|
|
|
|
* the kernel version of the buffer pointer. The size allocated
|
|
|
|
* is the sum of the three given sizes (each rounded up to
|
|
|
|
* pointer-sized boundary)
|
|
|
|
*
|
|
|
|
* Return: The allocated buffer or %NULL if error
|
|
|
|
*/
|
|
|
|
struct binder_buffer *binder_alloc_new_buf(struct binder_alloc *alloc,
|
|
|
|
size_t data_size,
|
|
|
|
size_t offsets_size,
|
|
|
|
size_t extra_buffers_size,
|
2020-08-21 12:25:44 +00:00
|
|
|
int is_async,
|
|
|
|
int pid)
|
2017-06-29 19:01:41 +00:00
|
|
|
{
|
|
|
|
struct binder_buffer *buffer;
|
|
|
|
|
|
|
|
mutex_lock(&alloc->mutex);
|
|
|
|
buffer = binder_alloc_new_buf_locked(alloc, data_size, offsets_size,
|
2020-08-21 12:25:44 +00:00
|
|
|
extra_buffers_size, is_async, pid);
|
2017-06-29 19:01:41 +00:00
|
|
|
mutex_unlock(&alloc->mutex);
|
|
|
|
return buffer;
|
|
|
|
}
|
|
|
|
|
2019-02-08 18:35:20 +00:00
|
|
|
static void __user *buffer_start_page(struct binder_buffer *buffer)
|
2017-06-29 19:01:41 +00:00
|
|
|
{
|
2019-02-08 18:35:20 +00:00
|
|
|
return (void __user *)((uintptr_t)buffer->user_data & PAGE_MASK);
|
2017-06-29 19:01:41 +00:00
|
|
|
}
|
|
|
|
|
2019-02-08 18:35:20 +00:00
|
|
|
static void __user *prev_buffer_end_page(struct binder_buffer *buffer)
|
2017-06-29 19:01:41 +00:00
|
|
|
{
|
2019-02-08 18:35:20 +00:00
|
|
|
return (void __user *)
|
|
|
|
(((uintptr_t)(buffer->user_data) - 1) & PAGE_MASK);
|
2017-06-29 19:01:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void binder_delete_free_buffer(struct binder_alloc *alloc,
|
|
|
|
struct binder_buffer *buffer)
|
|
|
|
{
|
|
|
|
struct binder_buffer *prev, *next = NULL;
|
2017-08-23 15:46:41 +00:00
|
|
|
bool to_free = true;
|
2020-07-24 13:12:54 +00:00
|
|
|
|
2017-06-29 19:01:41 +00:00
|
|
|
BUG_ON(alloc->buffers.next == &buffer->entry);
|
2017-08-23 15:46:39 +00:00
|
|
|
prev = binder_buffer_prev(buffer);
|
2017-06-29 19:01:41 +00:00
|
|
|
BUG_ON(!prev->free);
|
2017-08-23 15:46:41 +00:00
|
|
|
if (prev_buffer_end_page(prev) == buffer_start_page(buffer)) {
|
|
|
|
to_free = false;
|
2017-06-29 19:01:41 +00:00
|
|
|
binder_alloc_debug(BINDER_DEBUG_BUFFER_ALLOC,
|
2017-08-23 15:46:41 +00:00
|
|
|
"%d: merge free, buffer %pK share page with %pK\n",
|
2019-02-08 18:35:20 +00:00
|
|
|
alloc->pid, buffer->user_data,
|
|
|
|
prev->user_data);
|
2017-06-29 19:01:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!list_is_last(&buffer->entry, &alloc->buffers)) {
|
2017-08-23 15:46:39 +00:00
|
|
|
next = binder_buffer_next(buffer);
|
2017-08-23 15:46:41 +00:00
|
|
|
if (buffer_start_page(next) == buffer_start_page(buffer)) {
|
|
|
|
to_free = false;
|
2017-06-29 19:01:41 +00:00
|
|
|
binder_alloc_debug(BINDER_DEBUG_BUFFER_ALLOC,
|
2017-08-23 15:46:41 +00:00
|
|
|
"%d: merge free, buffer %pK share page with %pK\n",
|
|
|
|
alloc->pid,
|
2019-02-08 18:35:20 +00:00
|
|
|
buffer->user_data,
|
|
|
|
next->user_data);
|
2017-06-29 19:01:41 +00:00
|
|
|
}
|
|
|
|
}
|
2017-08-23 15:46:41 +00:00
|
|
|
|
2019-02-08 18:35:20 +00:00
|
|
|
if (PAGE_ALIGNED(buffer->user_data)) {
|
2017-08-23 15:46:41 +00:00
|
|
|
binder_alloc_debug(BINDER_DEBUG_BUFFER_ALLOC,
|
|
|
|
"%d: merge free, buffer start %pK is page aligned\n",
|
2019-02-08 18:35:20 +00:00
|
|
|
alloc->pid, buffer->user_data);
|
2017-08-23 15:46:41 +00:00
|
|
|
to_free = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (to_free) {
|
2017-06-29 19:01:41 +00:00
|
|
|
binder_alloc_debug(BINDER_DEBUG_BUFFER_ALLOC,
|
2017-08-23 15:46:41 +00:00
|
|
|
"%d: merge free, buffer %pK do not share page with %pK or %pK\n",
|
2019-02-08 18:35:20 +00:00
|
|
|
alloc->pid, buffer->user_data,
|
|
|
|
prev->user_data,
|
|
|
|
next ? next->user_data : NULL);
|
2017-08-23 15:46:41 +00:00
|
|
|
binder_update_page_range(alloc, 0, buffer_start_page(buffer),
|
2017-09-16 05:11:56 +00:00
|
|
|
buffer_start_page(buffer) + PAGE_SIZE);
|
2017-06-29 19:01:41 +00:00
|
|
|
}
|
2017-08-23 15:46:41 +00:00
|
|
|
list_del(&buffer->entry);
|
|
|
|
kfree(buffer);
|
2017-06-29 19:01:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void binder_free_buf_locked(struct binder_alloc *alloc,
|
|
|
|
struct binder_buffer *buffer)
|
|
|
|
{
|
|
|
|
size_t size, buffer_size;
|
|
|
|
|
|
|
|
buffer_size = binder_alloc_buffer_size(alloc, buffer);
|
|
|
|
|
|
|
|
size = ALIGN(buffer->data_size, sizeof(void *)) +
|
|
|
|
ALIGN(buffer->offsets_size, sizeof(void *)) +
|
|
|
|
ALIGN(buffer->extra_buffers_size, sizeof(void *));
|
|
|
|
|
|
|
|
binder_alloc_debug(BINDER_DEBUG_BUFFER_ALLOC,
|
|
|
|
"%d: binder_free_buf %pK size %zd buffer_size %zd\n",
|
|
|
|
alloc->pid, buffer, size, buffer_size);
|
|
|
|
|
|
|
|
BUG_ON(buffer->free);
|
|
|
|
BUG_ON(size > buffer_size);
|
|
|
|
BUG_ON(buffer->transaction != NULL);
|
2019-02-08 18:35:20 +00:00
|
|
|
BUG_ON(buffer->user_data < alloc->buffer);
|
|
|
|
BUG_ON(buffer->user_data > alloc->buffer + alloc->buffer_size);
|
2017-06-29 19:01:41 +00:00
|
|
|
|
|
|
|
if (buffer->async_transaction) {
|
|
|
|
alloc->free_async_space += size + sizeof(struct binder_buffer);
|
|
|
|
|
|
|
|
binder_alloc_debug(BINDER_DEBUG_BUFFER_ALLOC_ASYNC,
|
|
|
|
"%d: binder_free_buf size %zd async free %zd\n",
|
|
|
|
alloc->pid, size, alloc->free_async_space);
|
|
|
|
}
|
|
|
|
|
|
|
|
binder_update_page_range(alloc, 0,
|
2019-02-08 18:35:20 +00:00
|
|
|
(void __user *)PAGE_ALIGN((uintptr_t)buffer->user_data),
|
|
|
|
(void __user *)(((uintptr_t)
|
|
|
|
buffer->user_data + buffer_size) & PAGE_MASK));
|
2017-06-29 19:01:41 +00:00
|
|
|
|
|
|
|
rb_erase(&buffer->rb_node, &alloc->allocated_buffers);
|
|
|
|
buffer->free = 1;
|
|
|
|
if (!list_is_last(&buffer->entry, &alloc->buffers)) {
|
2017-08-23 15:46:39 +00:00
|
|
|
struct binder_buffer *next = binder_buffer_next(buffer);
|
2017-06-29 19:01:41 +00:00
|
|
|
|
|
|
|
if (next->free) {
|
|
|
|
rb_erase(&next->rb_node, &alloc->free_buffers);
|
|
|
|
binder_delete_free_buffer(alloc, next);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (alloc->buffers.next != &buffer->entry) {
|
2017-08-23 15:46:39 +00:00
|
|
|
struct binder_buffer *prev = binder_buffer_prev(buffer);
|
2017-06-29 19:01:41 +00:00
|
|
|
|
|
|
|
if (prev->free) {
|
|
|
|
binder_delete_free_buffer(alloc, buffer);
|
|
|
|
rb_erase(&prev->rb_node, &alloc->free_buffers);
|
|
|
|
buffer = prev;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
binder_insert_free_buffer(alloc, buffer);
|
|
|
|
}
|
|
|
|
|
2020-11-20 23:37:43 +00:00
|
|
|
static void binder_alloc_clear_buf(struct binder_alloc *alloc,
|
|
|
|
struct binder_buffer *buffer);
|
2017-06-29 19:01:41 +00:00
|
|
|
/**
|
|
|
|
* binder_alloc_free_buf() - free a binder buffer
|
|
|
|
* @alloc: binder_alloc for this proc
|
|
|
|
* @buffer: kernel pointer to buffer
|
|
|
|
*
|
2020-08-18 01:34:04 +00:00
|
|
|
* Free the buffer allocated via binder_alloc_new_buf()
|
2017-06-29 19:01:41 +00:00
|
|
|
*/
|
|
|
|
void binder_alloc_free_buf(struct binder_alloc *alloc,
|
|
|
|
struct binder_buffer *buffer)
|
|
|
|
{
|
2020-11-20 23:37:43 +00:00
|
|
|
/*
|
|
|
|
* We could eliminate the call to binder_alloc_clear_buf()
|
|
|
|
* from binder_alloc_deferred_release() by moving this to
|
|
|
|
* binder_alloc_free_buf_locked(). However, that could
|
|
|
|
* increase contention for the alloc mutex if clear_on_free
|
|
|
|
* is used frequently for large buffers. The mutex is not
|
|
|
|
* needed for correctness here.
|
|
|
|
*/
|
|
|
|
if (buffer->clear_on_free) {
|
|
|
|
binder_alloc_clear_buf(alloc, buffer);
|
|
|
|
buffer->clear_on_free = false;
|
|
|
|
}
|
2017-06-29 19:01:41 +00:00
|
|
|
mutex_lock(&alloc->mutex);
|
|
|
|
binder_free_buf_locked(alloc, buffer);
|
|
|
|
mutex_unlock(&alloc->mutex);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* binder_alloc_mmap_handler() - map virtual address space for proc
|
|
|
|
* @alloc: alloc structure for this proc
|
|
|
|
* @vma: vma passed to mmap()
|
|
|
|
*
|
|
|
|
* Called by binder_mmap() to initialize the space specified in
|
|
|
|
* vma for allocating binder buffers
|
|
|
|
*
|
|
|
|
* Return:
|
|
|
|
* 0 = success
|
|
|
|
* -EBUSY = address space already mapped
|
|
|
|
* -ENOMEM = failed to map memory to given address space
|
|
|
|
*/
|
|
|
|
int binder_alloc_mmap_handler(struct binder_alloc *alloc,
|
|
|
|
struct vm_area_struct *vma)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
const char *failure_string;
|
|
|
|
struct binder_buffer *buffer;
|
|
|
|
|
|
|
|
mutex_lock(&binder_alloc_mmap_lock);
|
2019-10-18 20:56:30 +00:00
|
|
|
if (alloc->buffer_size) {
|
2017-06-29 19:01:41 +00:00
|
|
|
ret = -EBUSY;
|
|
|
|
failure_string = "already mapped";
|
|
|
|
goto err_already_mapped;
|
|
|
|
}
|
2019-10-18 20:56:30 +00:00
|
|
|
alloc->buffer_size = min_t(unsigned long, vma->vm_end - vma->vm_start,
|
|
|
|
SZ_4M);
|
|
|
|
mutex_unlock(&binder_alloc_mmap_lock);
|
2017-06-29 19:01:41 +00:00
|
|
|
|
2019-02-08 18:35:20 +00:00
|
|
|
alloc->buffer = (void __user *)vma->vm_start;
|
2017-06-29 19:01:41 +00:00
|
|
|
|
binder: Don't modify VMA bounds in ->mmap handler
binder_mmap() tries to prevent the creation of overly big binder mappings
by silently truncating the size of the VMA to 4MiB. However, this violates
the API contract of mmap(). If userspace attempts to create a large binder
VMA, and later attempts to unmap that VMA, it will call munmap() on a range
beyond the end of the VMA, which may have been allocated to another VMA in
the meantime. This can lead to userspace memory corruption.
The following sequence of calls leads to a segfault without this commit:
int main(void) {
int binder_fd = open("/dev/binder", O_RDWR);
if (binder_fd == -1) err(1, "open binder");
void *binder_mapping = mmap(NULL, 0x800000UL, PROT_READ, MAP_SHARED,
binder_fd, 0);
if (binder_mapping == MAP_FAILED) err(1, "mmap binder");
void *data_mapping = mmap(NULL, 0x400000UL, PROT_READ|PROT_WRITE,
MAP_PRIVATE|MAP_ANONYMOUS, -1, 0);
if (data_mapping == MAP_FAILED) err(1, "mmap data");
munmap(binder_mapping, 0x800000UL);
*(char*)data_mapping = 1;
return 0;
}
Cc: stable@vger.kernel.org
Signed-off-by: Jann Horn <jannh@google.com>
Acked-by: Todd Kjos <tkjos@google.com>
Acked-by: Christian Brauner <christian.brauner@ubuntu.com>
Link: https://lore.kernel.org/r/20191016150119.154756-1-jannh@google.com
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2019-10-16 15:01:18 +00:00
|
|
|
alloc->pages = kcalloc(alloc->buffer_size / PAGE_SIZE,
|
treewide: kzalloc() -> kcalloc()
The kzalloc() function has a 2-factor argument form, kcalloc(). This
patch replaces cases of:
kzalloc(a * b, gfp)
with:
kcalloc(a * b, gfp)
as well as handling cases of:
kzalloc(a * b * c, gfp)
with:
kzalloc(array3_size(a, b, c), gfp)
as it's slightly less ugly than:
kzalloc_array(array_size(a, b), c, gfp)
This does, however, attempt to ignore constant size factors like:
kzalloc(4 * 1024, gfp)
though any constants defined via macros get caught up in the conversion.
Any factors with a sizeof() of "unsigned char", "char", and "u8" were
dropped, since they're redundant.
The Coccinelle script used for this was:
// Fix redundant parens around sizeof().
@@
type TYPE;
expression THING, E;
@@
(
kzalloc(
- (sizeof(TYPE)) * E
+ sizeof(TYPE) * E
, ...)
|
kzalloc(
- (sizeof(THING)) * E
+ sizeof(THING) * E
, ...)
)
// Drop single-byte sizes and redundant parens.
@@
expression COUNT;
typedef u8;
typedef __u8;
@@
(
kzalloc(
- sizeof(u8) * (COUNT)
+ COUNT
, ...)
|
kzalloc(
- sizeof(__u8) * (COUNT)
+ COUNT
, ...)
|
kzalloc(
- sizeof(char) * (COUNT)
+ COUNT
, ...)
|
kzalloc(
- sizeof(unsigned char) * (COUNT)
+ COUNT
, ...)
|
kzalloc(
- sizeof(u8) * COUNT
+ COUNT
, ...)
|
kzalloc(
- sizeof(__u8) * COUNT
+ COUNT
, ...)
|
kzalloc(
- sizeof(char) * COUNT
+ COUNT
, ...)
|
kzalloc(
- sizeof(unsigned char) * COUNT
+ COUNT
, ...)
)
// 2-factor product with sizeof(type/expression) and identifier or constant.
@@
type TYPE;
expression THING;
identifier COUNT_ID;
constant COUNT_CONST;
@@
(
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * (COUNT_ID)
+ COUNT_ID, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * COUNT_ID
+ COUNT_ID, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * (COUNT_CONST)
+ COUNT_CONST, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * COUNT_CONST
+ COUNT_CONST, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * (COUNT_ID)
+ COUNT_ID, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * COUNT_ID
+ COUNT_ID, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * (COUNT_CONST)
+ COUNT_CONST, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * COUNT_CONST
+ COUNT_CONST, sizeof(THING)
, ...)
)
// 2-factor product, only identifiers.
@@
identifier SIZE, COUNT;
@@
- kzalloc
+ kcalloc
(
- SIZE * COUNT
+ COUNT, SIZE
, ...)
// 3-factor product with 1 sizeof(type) or sizeof(expression), with
// redundant parens removed.
@@
expression THING;
identifier STRIDE, COUNT;
type TYPE;
@@
(
kzalloc(
- sizeof(TYPE) * (COUNT) * (STRIDE)
+ array3_size(COUNT, STRIDE, sizeof(TYPE))
, ...)
|
kzalloc(
- sizeof(TYPE) * (COUNT) * STRIDE
+ array3_size(COUNT, STRIDE, sizeof(TYPE))
, ...)
|
kzalloc(
- sizeof(TYPE) * COUNT * (STRIDE)
+ array3_size(COUNT, STRIDE, sizeof(TYPE))
, ...)
|
kzalloc(
- sizeof(TYPE) * COUNT * STRIDE
+ array3_size(COUNT, STRIDE, sizeof(TYPE))
, ...)
|
kzalloc(
- sizeof(THING) * (COUNT) * (STRIDE)
+ array3_size(COUNT, STRIDE, sizeof(THING))
, ...)
|
kzalloc(
- sizeof(THING) * (COUNT) * STRIDE
+ array3_size(COUNT, STRIDE, sizeof(THING))
, ...)
|
kzalloc(
- sizeof(THING) * COUNT * (STRIDE)
+ array3_size(COUNT, STRIDE, sizeof(THING))
, ...)
|
kzalloc(
- sizeof(THING) * COUNT * STRIDE
+ array3_size(COUNT, STRIDE, sizeof(THING))
, ...)
)
// 3-factor product with 2 sizeof(variable), with redundant parens removed.
@@
expression THING1, THING2;
identifier COUNT;
type TYPE1, TYPE2;
@@
(
kzalloc(
- sizeof(TYPE1) * sizeof(TYPE2) * COUNT
+ array3_size(COUNT, sizeof(TYPE1), sizeof(TYPE2))
, ...)
|
kzalloc(
- sizeof(TYPE1) * sizeof(THING2) * (COUNT)
+ array3_size(COUNT, sizeof(TYPE1), sizeof(TYPE2))
, ...)
|
kzalloc(
- sizeof(THING1) * sizeof(THING2) * COUNT
+ array3_size(COUNT, sizeof(THING1), sizeof(THING2))
, ...)
|
kzalloc(
- sizeof(THING1) * sizeof(THING2) * (COUNT)
+ array3_size(COUNT, sizeof(THING1), sizeof(THING2))
, ...)
|
kzalloc(
- sizeof(TYPE1) * sizeof(THING2) * COUNT
+ array3_size(COUNT, sizeof(TYPE1), sizeof(THING2))
, ...)
|
kzalloc(
- sizeof(TYPE1) * sizeof(THING2) * (COUNT)
+ array3_size(COUNT, sizeof(TYPE1), sizeof(THING2))
, ...)
)
// 3-factor product, only identifiers, with redundant parens removed.
@@
identifier STRIDE, SIZE, COUNT;
@@
(
kzalloc(
- (COUNT) * STRIDE * SIZE
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- COUNT * (STRIDE) * SIZE
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- COUNT * STRIDE * (SIZE)
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- (COUNT) * (STRIDE) * SIZE
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- COUNT * (STRIDE) * (SIZE)
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- (COUNT) * STRIDE * (SIZE)
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- (COUNT) * (STRIDE) * (SIZE)
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- COUNT * STRIDE * SIZE
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
)
// Any remaining multi-factor products, first at least 3-factor products,
// when they're not all constants...
@@
expression E1, E2, E3;
constant C1, C2, C3;
@@
(
kzalloc(C1 * C2 * C3, ...)
|
kzalloc(
- (E1) * E2 * E3
+ array3_size(E1, E2, E3)
, ...)
|
kzalloc(
- (E1) * (E2) * E3
+ array3_size(E1, E2, E3)
, ...)
|
kzalloc(
- (E1) * (E2) * (E3)
+ array3_size(E1, E2, E3)
, ...)
|
kzalloc(
- E1 * E2 * E3
+ array3_size(E1, E2, E3)
, ...)
)
// And then all remaining 2 factors products when they're not all constants,
// keeping sizeof() as the second factor argument.
@@
expression THING, E1, E2;
type TYPE;
constant C1, C2, C3;
@@
(
kzalloc(sizeof(THING) * C2, ...)
|
kzalloc(sizeof(TYPE) * C2, ...)
|
kzalloc(C1 * C2 * C3, ...)
|
kzalloc(C1 * C2, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * (E2)
+ E2, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * E2
+ E2, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * (E2)
+ E2, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * E2
+ E2, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- (E1) * E2
+ E1, E2
, ...)
|
- kzalloc
+ kcalloc
(
- (E1) * (E2)
+ E1, E2
, ...)
|
- kzalloc
+ kcalloc
(
- E1 * E2
+ E1, E2
, ...)
)
Signed-off-by: Kees Cook <keescook@chromium.org>
2018-06-12 21:03:40 +00:00
|
|
|
sizeof(alloc->pages[0]),
|
2017-06-29 19:01:41 +00:00
|
|
|
GFP_KERNEL);
|
|
|
|
if (alloc->pages == NULL) {
|
|
|
|
ret = -ENOMEM;
|
|
|
|
failure_string = "alloc page array";
|
|
|
|
goto err_alloc_pages_failed;
|
|
|
|
}
|
|
|
|
|
2017-08-23 15:46:41 +00:00
|
|
|
buffer = kzalloc(sizeof(*buffer), GFP_KERNEL);
|
|
|
|
if (!buffer) {
|
2017-06-29 19:01:41 +00:00
|
|
|
ret = -ENOMEM;
|
2017-08-23 15:46:41 +00:00
|
|
|
failure_string = "alloc buffer struct";
|
|
|
|
goto err_alloc_buf_struct_failed;
|
2017-06-29 19:01:41 +00:00
|
|
|
}
|
2017-08-23 15:46:41 +00:00
|
|
|
|
2019-02-08 18:35:20 +00:00
|
|
|
buffer->user_data = alloc->buffer;
|
2017-06-29 19:01:41 +00:00
|
|
|
list_add(&buffer->entry, &alloc->buffers);
|
|
|
|
buffer->free = 1;
|
|
|
|
binder_insert_free_buffer(alloc, buffer);
|
|
|
|
alloc->free_async_space = alloc->buffer_size / 2;
|
2018-08-23 05:29:56 +00:00
|
|
|
binder_alloc_set_vma(alloc, vma);
|
2017-10-21 00:58:58 +00:00
|
|
|
mmgrab(alloc->vma_vm_mm);
|
2017-06-29 19:01:41 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
2017-08-23 15:46:41 +00:00
|
|
|
err_alloc_buf_struct_failed:
|
2017-06-29 19:01:41 +00:00
|
|
|
kfree(alloc->pages);
|
|
|
|
alloc->pages = NULL;
|
|
|
|
err_alloc_pages_failed:
|
|
|
|
alloc->buffer = NULL;
|
2019-10-18 20:56:30 +00:00
|
|
|
mutex_lock(&binder_alloc_mmap_lock);
|
|
|
|
alloc->buffer_size = 0;
|
2017-06-29 19:01:41 +00:00
|
|
|
err_already_mapped:
|
|
|
|
mutex_unlock(&binder_alloc_mmap_lock);
|
2018-08-07 19:57:13 +00:00
|
|
|
binder_alloc_debug(BINDER_DEBUG_USER_ERROR,
|
|
|
|
"%s: %d %lx-%lx %s failed %d\n", __func__,
|
|
|
|
alloc->pid, vma->vm_start, vma->vm_end,
|
|
|
|
failure_string, ret);
|
2017-06-29 19:01:41 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void binder_alloc_deferred_release(struct binder_alloc *alloc)
|
|
|
|
{
|
|
|
|
struct rb_node *n;
|
|
|
|
int buffers, page_count;
|
2017-08-23 15:46:41 +00:00
|
|
|
struct binder_buffer *buffer;
|
2017-06-29 19:01:41 +00:00
|
|
|
|
|
|
|
buffers = 0;
|
|
|
|
mutex_lock(&alloc->mutex);
|
2018-08-23 05:29:56 +00:00
|
|
|
BUG_ON(alloc->vma);
|
|
|
|
|
2017-06-29 19:01:41 +00:00
|
|
|
while ((n = rb_first(&alloc->allocated_buffers))) {
|
|
|
|
buffer = rb_entry(n, struct binder_buffer, rb_node);
|
|
|
|
|
|
|
|
/* Transaction should already have been freed */
|
|
|
|
BUG_ON(buffer->transaction);
|
|
|
|
|
2020-11-20 23:37:43 +00:00
|
|
|
if (buffer->clear_on_free) {
|
|
|
|
binder_alloc_clear_buf(alloc, buffer);
|
|
|
|
buffer->clear_on_free = false;
|
|
|
|
}
|
2017-06-29 19:01:41 +00:00
|
|
|
binder_free_buf_locked(alloc, buffer);
|
|
|
|
buffers++;
|
|
|
|
}
|
|
|
|
|
2017-08-23 15:46:41 +00:00
|
|
|
while (!list_empty(&alloc->buffers)) {
|
|
|
|
buffer = list_first_entry(&alloc->buffers,
|
|
|
|
struct binder_buffer, entry);
|
|
|
|
WARN_ON(!buffer->free);
|
|
|
|
|
|
|
|
list_del(&buffer->entry);
|
|
|
|
WARN_ON_ONCE(!list_empty(&alloc->buffers));
|
|
|
|
kfree(buffer);
|
|
|
|
}
|
|
|
|
|
2017-06-29 19:01:41 +00:00
|
|
|
page_count = 0;
|
|
|
|
if (alloc->pages) {
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < alloc->buffer_size / PAGE_SIZE; i++) {
|
2019-02-08 18:35:20 +00:00
|
|
|
void __user *page_addr;
|
2017-08-23 15:46:42 +00:00
|
|
|
bool on_lru;
|
2017-06-29 19:01:41 +00:00
|
|
|
|
2017-08-23 15:46:42 +00:00
|
|
|
if (!alloc->pages[i].page_ptr)
|
2017-06-29 19:01:41 +00:00
|
|
|
continue;
|
|
|
|
|
list_lru: allow explicit memcg and NUMA node selection
JIRA: https://issues.redhat.com/browse/RHEL-40684
Conflicts:
* include/linux/list_lru.h: minor context differences due to missing
upstream v6.8 commit 7679e14098c9 ("mm: list_lru: Update kernel
documentation to follow the requirements")
This patch is a backport of the following upstream commit:
commit 0a97c01cd20bb96359d8c9dedad92a061ed34e0b
Author: Nhat Pham <nphamcs@gmail.com>
Date: Thu Nov 30 11:40:18 2023 -0800
list_lru: allow explicit memcg and NUMA node selection
Patch series "workload-specific and memory pressure-driven zswap
writeback", v8.
There are currently several issues with zswap writeback:
1. There is only a single global LRU for zswap, making it impossible to
perform worload-specific shrinking - an memcg under memory pressure
cannot determine which pages in the pool it owns, and often ends up
writing pages from other memcgs. This issue has been previously
observed in practice and mitigated by simply disabling
memcg-initiated shrinking:
https://lore.kernel.org/all/20230530232435.3097106-1-nphamcs@gmail.com/T/#u
But this solution leaves a lot to be desired, as we still do not
have an avenue for an memcg to free up its own memory locked up in
the zswap pool.
2. We only shrink the zswap pool when the user-defined limit is hit.
This means that if we set the limit too high, cold data that are
unlikely to be used again will reside in the pool, wasting precious
memory. It is hard to predict how much zswap space will be needed
ahead of time, as this depends on the workload (specifically, on
factors such as memory access patterns and compressibility of the
memory pages).
This patch series solves these issues by separating the global zswap LRU
into per-memcg and per-NUMA LRUs, and performs workload-specific (i.e
memcg- and NUMA-aware) zswap writeback under memory pressure. The new
shrinker does not have any parameter that must be tuned by the user, and
can be opted in or out on a per-memcg basis.
As a proof of concept, we ran the following synthetic benchmark: build the
linux kernel in a memory-limited cgroup, and allocate some cold data in
tmpfs to see if the shrinker could write them out and improved the overall
performance. Depending on the amount of cold data generated, we observe
from 14% to 35% reduction in kernel CPU time used in the kernel builds.
This patch (of 6):
The interface of list_lru is based on the assumption that the list node
and the data it represents belong to the same allocated on the correct
node/memcg. While this assumption is valid for existing slab objects LRU
such as dentries and inodes, it is undocumented, and rather inflexible for
certain potential list_lru users (such as the upcoming zswap shrinker and
the THP shrinker). It has caused us a lot of issues during our
development.
This patch changes list_lru interface so that the caller must explicitly
specify numa node and memcg when adding and removing objects. The old
list_lru_add() and list_lru_del() are renamed to list_lru_add_obj() and
list_lru_del_obj(), respectively.
It also extends the list_lru API with a new function, list_lru_putback,
which undoes a previous list_lru_isolate call. Unlike list_lru_add, it
does not increment the LRU node count (as list_lru_isolate does not
decrement the node count). list_lru_putback also allows for explicit
memcg and NUMA node selection.
Link: https://lkml.kernel.org/r/20231130194023.4102148-1-nphamcs@gmail.com
Link: https://lkml.kernel.org/r/20231130194023.4102148-2-nphamcs@gmail.com
Signed-off-by: Nhat Pham <nphamcs@gmail.com>
Suggested-by: Johannes Weiner <hannes@cmpxchg.org>
Acked-by: Johannes Weiner <hannes@cmpxchg.org>
Tested-by: Bagas Sanjaya <bagasdotme@gmail.com>
Cc: Chris Li <chrisl@kernel.org>
Cc: Dan Streetman <ddstreet@ieee.org>
Cc: Domenico Cerasuolo <cerasuolodomenico@gmail.com>
Cc: Michal Hocko <mhocko@kernel.org>
Cc: Muchun Song <muchun.song@linux.dev>
Cc: Roman Gushchin <roman.gushchin@linux.dev>
Cc: Seth Jennings <sjenning@redhat.com>
Cc: Shakeel Butt <shakeelb@google.com>
Cc: Shuah Khan <shuah@kernel.org>
Cc: Vitaly Wool <vitaly.wool@konsulko.com>
Cc: Yosry Ahmed <yosryahmed@google.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Rafael Aquini <aquini@redhat.com>
2024-06-28 16:23:39 +00:00
|
|
|
on_lru = list_lru_del_obj(&binder_alloc_lru,
|
2017-08-23 15:46:42 +00:00
|
|
|
&alloc->pages[i].lru);
|
2017-06-29 19:01:41 +00:00
|
|
|
page_addr = alloc->buffer + i * PAGE_SIZE;
|
|
|
|
binder_alloc_debug(BINDER_DEBUG_BUFFER_ALLOC,
|
2017-08-23 15:46:42 +00:00
|
|
|
"%s: %d: page %d at %pK %s\n",
|
|
|
|
__func__, alloc->pid, i, page_addr,
|
|
|
|
on_lru ? "on lru" : "active");
|
|
|
|
__free_page(alloc->pages[i].page_ptr);
|
2017-06-29 19:01:41 +00:00
|
|
|
page_count++;
|
|
|
|
}
|
|
|
|
kfree(alloc->pages);
|
|
|
|
}
|
|
|
|
mutex_unlock(&alloc->mutex);
|
2017-10-21 00:58:58 +00:00
|
|
|
if (alloc->vma_vm_mm)
|
|
|
|
mmdrop(alloc->vma_vm_mm);
|
2017-06-29 19:01:41 +00:00
|
|
|
|
|
|
|
binder_alloc_debug(BINDER_DEBUG_OPEN_CLOSE,
|
|
|
|
"%s: %d buffers %d, pages %d\n",
|
|
|
|
__func__, alloc->pid, buffers, page_count);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void print_binder_buffer(struct seq_file *m, const char *prefix,
|
|
|
|
struct binder_buffer *buffer)
|
|
|
|
{
|
2017-06-29 19:01:52 +00:00
|
|
|
seq_printf(m, "%s %d: %pK size %zd:%zd:%zd %s\n",
|
2019-02-08 18:35:20 +00:00
|
|
|
prefix, buffer->debug_id, buffer->user_data,
|
2017-06-29 19:01:41 +00:00
|
|
|
buffer->data_size, buffer->offsets_size,
|
2017-06-29 19:01:52 +00:00
|
|
|
buffer->extra_buffers_size,
|
2017-06-29 19:01:41 +00:00
|
|
|
buffer->transaction ? "active" : "delivered");
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* binder_alloc_print_allocated() - print buffer info
|
|
|
|
* @m: seq_file for output via seq_printf()
|
|
|
|
* @alloc: binder_alloc for this proc
|
|
|
|
*
|
|
|
|
* Prints information about every buffer associated with
|
|
|
|
* the binder_alloc state to the given seq_file
|
|
|
|
*/
|
|
|
|
void binder_alloc_print_allocated(struct seq_file *m,
|
|
|
|
struct binder_alloc *alloc)
|
|
|
|
{
|
|
|
|
struct rb_node *n;
|
|
|
|
|
|
|
|
mutex_lock(&alloc->mutex);
|
|
|
|
for (n = rb_first(&alloc->allocated_buffers); n != NULL; n = rb_next(n))
|
|
|
|
print_binder_buffer(m, " buffer",
|
|
|
|
rb_entry(n, struct binder_buffer, rb_node));
|
|
|
|
mutex_unlock(&alloc->mutex);
|
|
|
|
}
|
|
|
|
|
2017-08-31 18:56:36 +00:00
|
|
|
/**
|
|
|
|
* binder_alloc_print_pages() - print page usage
|
|
|
|
* @m: seq_file for output via seq_printf()
|
|
|
|
* @alloc: binder_alloc for this proc
|
|
|
|
*/
|
|
|
|
void binder_alloc_print_pages(struct seq_file *m,
|
|
|
|
struct binder_alloc *alloc)
|
|
|
|
{
|
|
|
|
struct binder_lru_page *page;
|
|
|
|
int i;
|
|
|
|
int active = 0;
|
|
|
|
int lru = 0;
|
|
|
|
int free = 0;
|
|
|
|
|
|
|
|
mutex_lock(&alloc->mutex);
|
2019-10-18 20:56:29 +00:00
|
|
|
/*
|
|
|
|
* Make sure the binder_alloc is fully initialized, otherwise we might
|
|
|
|
* read inconsistent state.
|
|
|
|
*/
|
|
|
|
if (binder_alloc_get_vma(alloc) != NULL) {
|
|
|
|
for (i = 0; i < alloc->buffer_size / PAGE_SIZE; i++) {
|
|
|
|
page = &alloc->pages[i];
|
|
|
|
if (!page->page_ptr)
|
|
|
|
free++;
|
|
|
|
else if (list_empty(&page->lru))
|
|
|
|
active++;
|
|
|
|
else
|
|
|
|
lru++;
|
|
|
|
}
|
2017-08-31 18:56:36 +00:00
|
|
|
}
|
|
|
|
mutex_unlock(&alloc->mutex);
|
|
|
|
seq_printf(m, " pages: %d:%d:%d\n", active, lru, free);
|
2017-11-13 09:06:56 +00:00
|
|
|
seq_printf(m, " pages high watermark: %zu\n", alloc->pages_high);
|
2017-08-31 18:56:36 +00:00
|
|
|
}
|
|
|
|
|
2017-06-29 19:01:41 +00:00
|
|
|
/**
|
|
|
|
* binder_alloc_get_allocated_count() - return count of buffers
|
|
|
|
* @alloc: binder_alloc for this proc
|
|
|
|
*
|
|
|
|
* Return: count of allocated buffers
|
|
|
|
*/
|
|
|
|
int binder_alloc_get_allocated_count(struct binder_alloc *alloc)
|
|
|
|
{
|
|
|
|
struct rb_node *n;
|
|
|
|
int count = 0;
|
|
|
|
|
|
|
|
mutex_lock(&alloc->mutex);
|
|
|
|
for (n = rb_first(&alloc->allocated_buffers); n != NULL; n = rb_next(n))
|
|
|
|
count++;
|
|
|
|
mutex_unlock(&alloc->mutex);
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* binder_alloc_vma_close() - invalidate address space
|
|
|
|
* @alloc: binder_alloc for this proc
|
|
|
|
*
|
|
|
|
* Called from binder_vma_close() when releasing address space.
|
|
|
|
* Clears alloc->vma to prevent new incoming transactions from
|
|
|
|
* allocating more buffers.
|
|
|
|
*/
|
|
|
|
void binder_alloc_vma_close(struct binder_alloc *alloc)
|
|
|
|
{
|
2018-08-23 05:29:56 +00:00
|
|
|
binder_alloc_set_vma(alloc, NULL);
|
2017-06-29 19:01:41 +00:00
|
|
|
}
|
|
|
|
|
2017-08-23 15:46:42 +00:00
|
|
|
/**
|
|
|
|
* binder_alloc_free_page() - shrinker callback to free pages
|
|
|
|
* @item: item to free
|
|
|
|
* @lock: lock protecting the item
|
|
|
|
* @cb_arg: callback argument
|
|
|
|
*
|
|
|
|
* Called from list_lru_walk() in binder_shrink_scan() to free
|
|
|
|
* up pages when the system is under memory pressure.
|
|
|
|
*/
|
|
|
|
enum lru_status binder_alloc_free_page(struct list_head *item,
|
|
|
|
struct list_lru_one *lru,
|
|
|
|
spinlock_t *lock,
|
|
|
|
void *cb_arg)
|
2018-11-06 23:56:31 +00:00
|
|
|
__must_hold(lock)
|
2017-08-23 15:46:42 +00:00
|
|
|
{
|
|
|
|
struct mm_struct *mm = NULL;
|
|
|
|
struct binder_lru_page *page = container_of(item,
|
|
|
|
struct binder_lru_page,
|
|
|
|
lru);
|
|
|
|
struct binder_alloc *alloc;
|
|
|
|
uintptr_t page_addr;
|
|
|
|
size_t index;
|
2017-10-03 23:15:00 +00:00
|
|
|
struct vm_area_struct *vma;
|
2017-08-23 15:46:42 +00:00
|
|
|
|
|
|
|
alloc = page->alloc;
|
|
|
|
if (!mutex_trylock(&alloc->mutex))
|
|
|
|
goto err_get_alloc_mutex_failed;
|
|
|
|
|
|
|
|
if (!page->page_ptr)
|
|
|
|
goto err_page_already_freed;
|
|
|
|
|
|
|
|
index = page - alloc->pages;
|
|
|
|
page_addr = (uintptr_t)alloc->buffer + index * PAGE_SIZE;
|
2019-03-01 23:06:06 +00:00
|
|
|
|
|
|
|
mm = alloc->vma_vm_mm;
|
|
|
|
if (!mmget_not_zero(mm))
|
|
|
|
goto err_mmget;
|
2020-06-09 04:33:25 +00:00
|
|
|
if (!mmap_read_trylock(mm))
|
2020-06-09 04:33:51 +00:00
|
|
|
goto err_mmap_read_lock_failed;
|
2018-08-23 05:29:56 +00:00
|
|
|
vma = binder_alloc_get_vma(alloc);
|
2017-10-03 23:15:00 +00:00
|
|
|
|
|
|
|
list_lru_isolate(lru, item);
|
|
|
|
spin_unlock(lock);
|
2017-08-23 15:46:42 +00:00
|
|
|
|
2017-10-03 23:15:00 +00:00
|
|
|
if (vma) {
|
2017-08-23 15:46:43 +00:00
|
|
|
trace_binder_unmap_user_start(alloc, index);
|
|
|
|
|
2024-04-29 18:32:56 +00:00
|
|
|
zap_page_range_single(vma, page_addr, PAGE_SIZE, NULL);
|
2017-08-23 15:46:42 +00:00
|
|
|
|
2017-08-23 15:46:43 +00:00
|
|
|
trace_binder_unmap_user_end(alloc, index);
|
2017-08-23 15:46:42 +00:00
|
|
|
}
|
2020-06-09 04:33:25 +00:00
|
|
|
mmap_read_unlock(mm);
|
2020-07-16 15:12:15 +00:00
|
|
|
mmput_async(mm);
|
2017-08-23 15:46:42 +00:00
|
|
|
|
2017-08-23 15:46:43 +00:00
|
|
|
trace_binder_unmap_kernel_start(alloc, index);
|
|
|
|
|
2017-08-23 15:46:42 +00:00
|
|
|
__free_page(page->page_ptr);
|
|
|
|
page->page_ptr = NULL;
|
|
|
|
|
2017-08-23 15:46:43 +00:00
|
|
|
trace_binder_unmap_kernel_end(alloc, index);
|
|
|
|
|
2017-10-03 23:15:00 +00:00
|
|
|
spin_lock(lock);
|
2017-08-23 15:46:42 +00:00
|
|
|
mutex_unlock(&alloc->mutex);
|
2017-10-03 23:15:00 +00:00
|
|
|
return LRU_REMOVED_RETRY;
|
2017-08-23 15:46:42 +00:00
|
|
|
|
2020-06-09 04:33:51 +00:00
|
|
|
err_mmap_read_lock_failed:
|
2017-10-03 23:15:00 +00:00
|
|
|
mmput_async(mm);
|
2017-10-21 00:58:58 +00:00
|
|
|
err_mmget:
|
2017-08-23 15:46:42 +00:00
|
|
|
err_page_already_freed:
|
|
|
|
mutex_unlock(&alloc->mutex);
|
|
|
|
err_get_alloc_mutex_failed:
|
|
|
|
return LRU_SKIP;
|
|
|
|
}
|
|
|
|
|
|
|
|
static unsigned long
|
|
|
|
binder_shrink_count(struct shrinker *shrink, struct shrink_control *sc)
|
|
|
|
{
|
|
|
|
unsigned long ret = list_lru_count(&binder_alloc_lru);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static unsigned long
|
|
|
|
binder_shrink_scan(struct shrinker *shrink, struct shrink_control *sc)
|
|
|
|
{
|
|
|
|
unsigned long ret;
|
|
|
|
|
|
|
|
ret = list_lru_walk(&binder_alloc_lru, binder_alloc_free_page,
|
|
|
|
NULL, sc->nr_to_scan);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2017-10-06 20:12:05 +00:00
|
|
|
static struct shrinker binder_shrinker = {
|
2017-08-23 15:46:42 +00:00
|
|
|
.count_objects = binder_shrink_count,
|
|
|
|
.scan_objects = binder_shrink_scan,
|
|
|
|
.seeks = DEFAULT_SEEKS,
|
|
|
|
};
|
|
|
|
|
2017-06-29 19:01:41 +00:00
|
|
|
/**
|
|
|
|
* binder_alloc_init() - called by binder_open() for per-proc initialization
|
|
|
|
* @alloc: binder_alloc for this proc
|
|
|
|
*
|
|
|
|
* Called from binder_open() to initialize binder_alloc fields for
|
|
|
|
* new binder proc
|
|
|
|
*/
|
|
|
|
void binder_alloc_init(struct binder_alloc *alloc)
|
|
|
|
{
|
|
|
|
alloc->pid = current->group_leader->pid;
|
|
|
|
mutex_init(&alloc->mutex);
|
2017-08-31 17:26:06 +00:00
|
|
|
INIT_LIST_HEAD(&alloc->buffers);
|
2017-06-29 19:01:41 +00:00
|
|
|
}
|
|
|
|
|
2017-11-29 13:29:47 +00:00
|
|
|
int binder_alloc_shrinker_init(void)
|
2017-08-23 15:46:42 +00:00
|
|
|
{
|
2017-11-29 13:29:47 +00:00
|
|
|
int ret = list_lru_init(&binder_alloc_lru);
|
|
|
|
|
|
|
|
if (ret == 0) {
|
2023-03-24 11:44:19 +00:00
|
|
|
ret = register_shrinker(&binder_shrinker, "android-binder");
|
2017-11-29 13:29:47 +00:00
|
|
|
if (ret)
|
|
|
|
list_lru_destroy(&binder_alloc_lru);
|
|
|
|
}
|
|
|
|
return ret;
|
2017-08-23 15:46:42 +00:00
|
|
|
}
|
2019-02-08 18:35:14 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* check_buffer() - verify that buffer/offset is safe to access
|
|
|
|
* @alloc: binder_alloc for this proc
|
|
|
|
* @buffer: binder buffer to be accessed
|
|
|
|
* @offset: offset into @buffer data
|
|
|
|
* @bytes: bytes to access from offset
|
|
|
|
*
|
|
|
|
* Check that the @offset/@bytes are within the size of the given
|
|
|
|
* @buffer and that the buffer is currently active and not freeable.
|
|
|
|
* Offsets must also be multiples of sizeof(u32). The kernel is
|
|
|
|
* allowed to touch the buffer in two cases:
|
|
|
|
*
|
|
|
|
* 1) when the buffer is being created:
|
|
|
|
* (buffer->free == 0 && buffer->allow_user_free == 0)
|
|
|
|
* 2) when the buffer is being torn down:
|
|
|
|
* (buffer->free == 0 && buffer->transaction == NULL).
|
|
|
|
*
|
|
|
|
* Return: true if the buffer is safe to access
|
|
|
|
*/
|
|
|
|
static inline bool check_buffer(struct binder_alloc *alloc,
|
|
|
|
struct binder_buffer *buffer,
|
|
|
|
binder_size_t offset, size_t bytes)
|
|
|
|
{
|
|
|
|
size_t buffer_size = binder_alloc_buffer_size(alloc, buffer);
|
|
|
|
|
|
|
|
return buffer_size >= bytes &&
|
|
|
|
offset <= buffer_size - bytes &&
|
|
|
|
IS_ALIGNED(offset, sizeof(u32)) &&
|
|
|
|
!buffer->free &&
|
|
|
|
(!buffer->allow_user_free || !buffer->transaction);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* binder_alloc_get_page() - get kernel pointer for given buffer offset
|
|
|
|
* @alloc: binder_alloc for this proc
|
|
|
|
* @buffer: binder buffer to be accessed
|
|
|
|
* @buffer_offset: offset into @buffer data
|
|
|
|
* @pgoffp: address to copy final page offset to
|
|
|
|
*
|
|
|
|
* Lookup the struct page corresponding to the address
|
2019-02-08 18:35:20 +00:00
|
|
|
* at @buffer_offset into @buffer->user_data. If @pgoffp is not
|
2019-02-08 18:35:14 +00:00
|
|
|
* NULL, the byte-offset into the page is written there.
|
|
|
|
*
|
|
|
|
* The caller is responsible to ensure that the offset points
|
|
|
|
* to a valid address within the @buffer and that @buffer is
|
|
|
|
* not freeable by the user. Since it can't be freed, we are
|
|
|
|
* guaranteed that the corresponding elements of @alloc->pages[]
|
|
|
|
* cannot change.
|
|
|
|
*
|
|
|
|
* Return: struct page
|
|
|
|
*/
|
|
|
|
static struct page *binder_alloc_get_page(struct binder_alloc *alloc,
|
|
|
|
struct binder_buffer *buffer,
|
|
|
|
binder_size_t buffer_offset,
|
|
|
|
pgoff_t *pgoffp)
|
|
|
|
{
|
|
|
|
binder_size_t buffer_space_offset = buffer_offset +
|
2019-02-08 18:35:20 +00:00
|
|
|
(buffer->user_data - alloc->buffer);
|
2019-02-08 18:35:14 +00:00
|
|
|
pgoff_t pgoff = buffer_space_offset & ~PAGE_MASK;
|
|
|
|
size_t index = buffer_space_offset >> PAGE_SHIFT;
|
|
|
|
struct binder_lru_page *lru_page;
|
|
|
|
|
|
|
|
lru_page = &alloc->pages[index];
|
|
|
|
*pgoffp = pgoff;
|
|
|
|
return lru_page->page_ptr;
|
|
|
|
}
|
|
|
|
|
2020-11-20 23:37:43 +00:00
|
|
|
/**
|
|
|
|
* binder_alloc_clear_buf() - zero out buffer
|
|
|
|
* @alloc: binder_alloc for this proc
|
|
|
|
* @buffer: binder buffer to be cleared
|
|
|
|
*
|
|
|
|
* memset the given buffer to 0
|
|
|
|
*/
|
|
|
|
static void binder_alloc_clear_buf(struct binder_alloc *alloc,
|
|
|
|
struct binder_buffer *buffer)
|
|
|
|
{
|
|
|
|
size_t bytes = binder_alloc_buffer_size(alloc, buffer);
|
|
|
|
binder_size_t buffer_offset = 0;
|
|
|
|
|
|
|
|
while (bytes) {
|
|
|
|
unsigned long size;
|
|
|
|
struct page *page;
|
|
|
|
pgoff_t pgoff;
|
|
|
|
void *kptr;
|
|
|
|
|
|
|
|
page = binder_alloc_get_page(alloc, buffer,
|
|
|
|
buffer_offset, &pgoff);
|
|
|
|
size = min_t(size_t, bytes, PAGE_SIZE - pgoff);
|
|
|
|
kptr = kmap(page) + pgoff;
|
|
|
|
memset(kptr, 0, size);
|
|
|
|
kunmap(page);
|
|
|
|
bytes -= size;
|
|
|
|
buffer_offset += size;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-08 18:35:14 +00:00
|
|
|
/**
|
|
|
|
* binder_alloc_copy_user_to_buffer() - copy src user to tgt user
|
|
|
|
* @alloc: binder_alloc for this proc
|
|
|
|
* @buffer: binder buffer to be accessed
|
|
|
|
* @buffer_offset: offset into @buffer data
|
|
|
|
* @from: userspace pointer to source buffer
|
|
|
|
* @bytes: bytes to copy
|
|
|
|
*
|
|
|
|
* Copy bytes from source userspace to target buffer.
|
|
|
|
*
|
|
|
|
* Return: bytes remaining to be copied
|
|
|
|
*/
|
|
|
|
unsigned long
|
|
|
|
binder_alloc_copy_user_to_buffer(struct binder_alloc *alloc,
|
|
|
|
struct binder_buffer *buffer,
|
|
|
|
binder_size_t buffer_offset,
|
|
|
|
const void __user *from,
|
|
|
|
size_t bytes)
|
|
|
|
{
|
|
|
|
if (!check_buffer(alloc, buffer, buffer_offset, bytes))
|
|
|
|
return bytes;
|
|
|
|
|
|
|
|
while (bytes) {
|
|
|
|
unsigned long size;
|
|
|
|
unsigned long ret;
|
|
|
|
struct page *page;
|
|
|
|
pgoff_t pgoff;
|
|
|
|
void *kptr;
|
|
|
|
|
|
|
|
page = binder_alloc_get_page(alloc, buffer,
|
|
|
|
buffer_offset, &pgoff);
|
|
|
|
size = min_t(size_t, bytes, PAGE_SIZE - pgoff);
|
|
|
|
kptr = kmap(page) + pgoff;
|
|
|
|
ret = copy_from_user(kptr, from, size);
|
|
|
|
kunmap(page);
|
|
|
|
if (ret)
|
|
|
|
return bytes - size + ret;
|
|
|
|
bytes -= size;
|
|
|
|
from += size;
|
|
|
|
buffer_offset += size;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
2019-02-08 18:35:15 +00:00
|
|
|
|
2019-06-28 16:50:12 +00:00
|
|
|
static int binder_alloc_do_buffer_copy(struct binder_alloc *alloc,
|
|
|
|
bool to_buffer,
|
|
|
|
struct binder_buffer *buffer,
|
|
|
|
binder_size_t buffer_offset,
|
|
|
|
void *ptr,
|
|
|
|
size_t bytes)
|
2019-02-08 18:35:15 +00:00
|
|
|
{
|
|
|
|
/* All copies must be 32-bit aligned and 32-bit size */
|
2019-06-28 16:50:12 +00:00
|
|
|
if (!check_buffer(alloc, buffer, buffer_offset, bytes))
|
|
|
|
return -EINVAL;
|
2019-02-08 18:35:15 +00:00
|
|
|
|
|
|
|
while (bytes) {
|
|
|
|
unsigned long size;
|
|
|
|
struct page *page;
|
|
|
|
pgoff_t pgoff;
|
|
|
|
void *tmpptr;
|
|
|
|
void *base_ptr;
|
|
|
|
|
|
|
|
page = binder_alloc_get_page(alloc, buffer,
|
|
|
|
buffer_offset, &pgoff);
|
|
|
|
size = min_t(size_t, bytes, PAGE_SIZE - pgoff);
|
|
|
|
base_ptr = kmap_atomic(page);
|
|
|
|
tmpptr = base_ptr + pgoff;
|
|
|
|
if (to_buffer)
|
|
|
|
memcpy(tmpptr, ptr, size);
|
|
|
|
else
|
|
|
|
memcpy(ptr, tmpptr, size);
|
|
|
|
/*
|
|
|
|
* kunmap_atomic() takes care of flushing the cache
|
|
|
|
* if this device has VIVT cache arch
|
|
|
|
*/
|
|
|
|
kunmap_atomic(base_ptr);
|
|
|
|
bytes -= size;
|
|
|
|
pgoff = 0;
|
|
|
|
ptr = ptr + size;
|
|
|
|
buffer_offset += size;
|
|
|
|
}
|
2019-06-28 16:50:12 +00:00
|
|
|
return 0;
|
2019-02-08 18:35:15 +00:00
|
|
|
}
|
|
|
|
|
2019-06-28 16:50:12 +00:00
|
|
|
int binder_alloc_copy_to_buffer(struct binder_alloc *alloc,
|
|
|
|
struct binder_buffer *buffer,
|
|
|
|
binder_size_t buffer_offset,
|
|
|
|
void *src,
|
|
|
|
size_t bytes)
|
2019-02-08 18:35:15 +00:00
|
|
|
{
|
2019-06-28 16:50:12 +00:00
|
|
|
return binder_alloc_do_buffer_copy(alloc, true, buffer, buffer_offset,
|
|
|
|
src, bytes);
|
2019-02-08 18:35:15 +00:00
|
|
|
}
|
|
|
|
|
2019-06-28 16:50:12 +00:00
|
|
|
int binder_alloc_copy_from_buffer(struct binder_alloc *alloc,
|
|
|
|
void *dest,
|
|
|
|
struct binder_buffer *buffer,
|
|
|
|
binder_size_t buffer_offset,
|
|
|
|
size_t bytes)
|
2019-02-08 18:35:15 +00:00
|
|
|
{
|
2019-06-28 16:50:12 +00:00
|
|
|
return binder_alloc_do_buffer_copy(alloc, false, buffer, buffer_offset,
|
|
|
|
dest, bytes);
|
2019-02-08 18:35:15 +00:00
|
|
|
}
|