2021-01-26 16:33:47 +08:00
|
|
|
// SPDX-License-Identifier: GPL-2.0
|
|
|
|
|
|
|
|
#include <linux/slab.h>
|
|
|
|
#include "ctree.h"
|
|
|
|
#include "subpage.h"
|
|
|
|
|
|
|
|
int btrfs_attach_subpage(const struct btrfs_fs_info *fs_info,
|
|
|
|
struct page *page, enum btrfs_subpage_type type)
|
|
|
|
{
|
2021-01-26 16:33:48 +08:00
|
|
|
struct btrfs_subpage *subpage = NULL;
|
|
|
|
int ret;
|
2021-01-26 16:33:47 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* We have cases like a dummy extent buffer page, which is not mappped
|
|
|
|
* and doesn't need to be locked.
|
|
|
|
*/
|
|
|
|
if (page->mapping)
|
|
|
|
ASSERT(PageLocked(page));
|
|
|
|
/* Either not subpage, or the page already has private attached */
|
|
|
|
if (fs_info->sectorsize == PAGE_SIZE || PagePrivate(page))
|
|
|
|
return 0;
|
|
|
|
|
2021-01-26 16:33:48 +08:00
|
|
|
ret = btrfs_alloc_subpage(fs_info, &subpage, type);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
2021-01-26 16:33:47 +08:00
|
|
|
attach_page_private(page, subpage);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void btrfs_detach_subpage(const struct btrfs_fs_info *fs_info,
|
|
|
|
struct page *page)
|
|
|
|
{
|
|
|
|
struct btrfs_subpage *subpage;
|
|
|
|
|
|
|
|
/* Either not subpage, or already detached */
|
|
|
|
if (fs_info->sectorsize == PAGE_SIZE || !PagePrivate(page))
|
|
|
|
return;
|
|
|
|
|
|
|
|
subpage = (struct btrfs_subpage *)detach_page_private(page);
|
|
|
|
ASSERT(subpage);
|
2021-01-26 16:33:48 +08:00
|
|
|
btrfs_free_subpage(subpage);
|
|
|
|
}
|
|
|
|
|
|
|
|
int btrfs_alloc_subpage(const struct btrfs_fs_info *fs_info,
|
|
|
|
struct btrfs_subpage **ret,
|
|
|
|
enum btrfs_subpage_type type)
|
|
|
|
{
|
|
|
|
if (fs_info->sectorsize == PAGE_SIZE)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
*ret = kzalloc(sizeof(struct btrfs_subpage), GFP_NOFS);
|
|
|
|
if (!*ret)
|
|
|
|
return -ENOMEM;
|
|
|
|
spin_lock_init(&(*ret)->lock);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void btrfs_free_subpage(struct btrfs_subpage *subpage)
|
|
|
|
{
|
2021-01-26 16:33:47 +08:00
|
|
|
kfree(subpage);
|
|
|
|
}
|