代码拉取完成,页面将自动刷新
// SPDX-License-Identifier: GPL-2.0
/*
* FDEION Memory Allocator page pool helpers
*
* Copyright (C) 2011 Google, Inc.
*/
#include <linux/list.h>
#include <linux/slab.h>
#include <linux/swap.h>
#include <linux/sched/signal.h>
#include "fdeion.h"
static inline struct page *fdeion_page_pool_alloc_pages(struct fdeion_page_pool *pool)
{
if (fatal_signal_pending(current))
return NULL;
return alloc_pages(pool->gfp_mask, pool->order);
}
static void fdeion_page_pool_free_pages(struct fdeion_page_pool *pool,
struct page *page)
{
__free_pages(page, pool->order);
}
static void fdeion_page_pool_add(struct fdeion_page_pool *pool, struct page *page)
{
mutex_lock(&pool->mutex);
if (PageHighMem(page))
{
list_add_tail(&page->lru, &pool->high_items);
pool->high_count++;
}
else
{
list_add_tail(&page->lru, &pool->low_items);
pool->low_count++;
}
mod_node_page_state(page_pgdat(page), NR_KERNEL_MISC_RECLAIMABLE,
1 << pool->order);
mutex_unlock(&pool->mutex);
}
static struct page *fdeion_page_pool_remove(struct fdeion_page_pool *pool, bool high)
{
struct page *page;
if (high)
{
BUG_ON(!pool->high_count);
page = list_first_entry(&pool->high_items, struct page, lru);
pool->high_count--;
}
else
{
BUG_ON(!pool->low_count);
page = list_first_entry(&pool->low_items, struct page, lru);
pool->low_count--;
}
list_del(&page->lru);
mod_node_page_state(page_pgdat(page), NR_KERNEL_MISC_RECLAIMABLE,
-(1 << pool->order));
return page;
}
struct page *fdeion_page_pool_alloc(struct fdeion_page_pool *pool)
{
struct page *page = NULL;
BUG_ON(!pool);
mutex_lock(&pool->mutex);
if (pool->high_count)
page = fdeion_page_pool_remove(pool, true);
else if (pool->low_count)
page = fdeion_page_pool_remove(pool, false);
mutex_unlock(&pool->mutex);
if (!page)
page = fdeion_page_pool_alloc_pages(pool);
return page;
}
void fdeion_page_pool_free(struct fdeion_page_pool *pool, struct page *page)
{
BUG_ON(pool->order != compound_order(page));
fdeion_page_pool_add(pool, page);
}
static int fdeion_page_pool_total(struct fdeion_page_pool *pool, bool high)
{
int count = pool->low_count;
if (high)
count += pool->high_count;
return count << pool->order;
}
int fdeion_page_pool_shrink(struct fdeion_page_pool *pool, gfp_t gfp_mask,
int nr_to_scan)
{
int freed = 0;
bool high;
if (current_is_kswapd())
high = true;
else
high = !!(gfp_mask & __GFP_HIGHMEM);
if (nr_to_scan == 0)
return fdeion_page_pool_total(pool, high);
while (freed < nr_to_scan)
{
struct page *page;
mutex_lock(&pool->mutex);
if (pool->low_count)
{
page = fdeion_page_pool_remove(pool, false);
}
else if (high && pool->high_count)
{
page = fdeion_page_pool_remove(pool, true);
}
else
{
mutex_unlock(&pool->mutex);
break;
}
mutex_unlock(&pool->mutex);
fdeion_page_pool_free_pages(pool, page);
freed += (1 << pool->order);
}
return freed;
}
struct fdeion_page_pool *fdeion_page_pool_create(gfp_t gfp_mask, unsigned int order)
{
struct fdeion_page_pool *pool = kmalloc(sizeof(*pool), GFP_KERNEL);
if (!pool)
return NULL;
pool->high_count = 0;
pool->low_count = 0;
INIT_LIST_HEAD(&pool->low_items);
INIT_LIST_HEAD(&pool->high_items);
pool->gfp_mask = gfp_mask | __GFP_COMP;
pool->order = order;
mutex_init(&pool->mutex);
plist_node_init(&pool->list, order);
return pool;
}
void fdeion_page_pool_destroy(struct fdeion_page_pool *pool)
{
kfree(pool);
}
此处可能存在不合适展示的内容,页面不予展示。您可通过相关编辑功能自查并修改。
如您确认内容无涉及 不当用语 / 纯广告导流 / 暴力 / 低俗色情 / 侵权 / 盗版 / 虚假 / 无价值内容或违法国家有关法律法规的内容,可点击提交进行申诉,我们将尽快为您处理。