mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/next/linux-next.git
synced 2025-01-09 07:23:14 +00:00
for-6.8-tag
-----BEGIN PGP SIGNATURE----- iQIzBAABCgAdFiEE8rQSAMVO+zA4DBdWxWXV+ddtWDsFAmWYTmMACgkQxWXV+ddt WDvPRg/+KgS5LV3nNC0MguYcTMQxmgeutIgXZIMfeA3v6EnFS7nj8leP4EPc6+bj JPSkwj4u2vHVwpnTVuEAuJUXnmFY+Qu70nVy6bM2uOHOYTVBQ8zRVK4cErNNLWCp OekDaADR53RrZ/xprlQ7b7Ph0Ch2uq9OrpH50IcyquEsH1ffkxlqwyrvth4/8dxC 6zgsFHWrbtVKJf0DYoQPpjEPz5tpdQ+xHZwtmf1cNlUgI1objODr/ZTqXtZqTfw4 /GwrtDPbEri53K/qjgr0dDH7pBVqD6PtnbgoHfYkiizZ0G7UkmlaK6rZIurtATJb Yk/RCqCUp9tPC4yeFSewFMm1Y8Ae3rkUBG7rnYkvMmBspMqyh/kQAWSBimF5yk/y vFEdFTe9AbdvP19Nw0CqovLzaO6RrOXCL1usnFvCmBgvF5gZAv63ZW1njP3ZoNta wB8Rs6hxdRkph8Dk7yvYf54uUR+JyKqjHY6egg2qkKTjz0CSf6qQFyFZXpr81m97 gK4WN5SeP/P2ukRbBKKyzZ5IljUxZuVatvJa0tktd7kAbU26WLzofOJ7pX+iqimM F2G7gKGJZykLY1WPntXBp9Dg97Ras2O5iViQ7ZKwRdOx1yZS5zzTYlIznHBAmXbL UgXfVnpJH1xFdkvedNTn+Fz9BHNV1K2a2AT7VITj7sxz23z3aJA= =4sw3 -----END PGP SIGNATURE----- Merge tag 'for-6.8-tag' of git://git.kernel.org/pub/scm/linux/kernel/git/kdave/linux Pull btrfs updates from David Sterba: "There are no exciting changes for users, it's been mostly API conversions and some fixes or refactoring. The mount API conversion is a base for future improvements that would come with VFS. Metadata processing has been converted to folios, not yet enabling the large folios but it's one patch away once everything gets tested enough. Core changes: - convert extent buffers to folios: - direct API conversion where possible - performance can drop by a few percent on metadata heavy workloads, the folio sizes are not constant and the calculations add up in the item helpers - both regular and subpage modes - data cannot be converted yet, we need to port that to iomap and there are some other generic changes required - convert mount to the new API, should not be user visible: - options deprecated long time ago have been removed: inode_cache, recovery - the new logic that splits mount to two phases slightly changes timing of device scanning for multi-device filesystems - LSM options will now work (like for selinux) - convert delayed nodes radix tree to xarray, preserving the preload-like logic that still allows to allocate with GFP_NOFS - more validation of sysfs value of scrub_speed_max - refactor chunk map structure, reduce size and improve performance - extent map refactoring, smaller data structures, improved performance - reduce size of struct extent_io_tree, embedded in several structures - temporary pages used for compression are cached and attached to a shrinker, this may slightly improve performance - in zoned mode, remove redirty extent buffer tracking, zeros are written in case an out-of-order is detected and proper data are written to the actual write pointer - cleanups, refactoring, error message improvements, updated tests - verify and update branch name or tag - remove unwanted text" * tag 'for-6.8-tag' of git://git.kernel.org/pub/scm/linux/kernel/git/kdave/linux: (89 commits) btrfs: pass btrfs_io_geometry into btrfs_max_io_len btrfs: pass struct btrfs_io_geometry to set_io_stripe btrfs: open code set_io_stripe for RAID56 btrfs: change block mapping to switch/case in btrfs_map_block btrfs: factor out block mapping for single profiles btrfs: factor out block mapping for RAID5/6 btrfs: reduce scope of data_stripes in btrfs_map_block btrfs: factor out block mapping for RAID10 btrfs: factor out block mapping for DUP profiles btrfs: factor out RAID1 block mapping btrfs: factor out block-mapping for RAID0 btrfs: re-introduce struct btrfs_io_geometry btrfs: factor out helper for single device IO check btrfs: migrate btrfs_repair_io_failure() to folio interfaces btrfs: migrate eb_bitmap_offset() to folio interfaces btrfs: migrate various end io functions to folios btrfs: migrate subpage code to folio interfaces btrfs: migrate get_eb_page_index() and get_eb_offset_in_page() to folios btrfs: don't double put our subpage reference in alloc_extent_buffer btrfs: cleanup metadata page pointer usage ...
This commit is contained in:
commit
affc5af36b
@ -27,7 +27,7 @@ static bool check_setget_bounds(const struct extent_buffer *eb,
|
||||
void btrfs_init_map_token(struct btrfs_map_token *token, struct extent_buffer *eb)
|
||||
{
|
||||
token->eb = eb;
|
||||
token->kaddr = page_address(eb->pages[0]);
|
||||
token->kaddr = folio_address(eb->folios[0]);
|
||||
token->offset = 0;
|
||||
}
|
||||
|
||||
@ -50,7 +50,7 @@ void btrfs_init_map_token(struct btrfs_map_token *token, struct extent_buffer *e
|
||||
* an offset into the extent buffer page array, cast to a specific type. This
|
||||
* gives us all the type checking.
|
||||
*
|
||||
* The extent buffer pages stored in the array pages do not form a contiguous
|
||||
* The extent buffer pages stored in the array folios may not form a contiguous
|
||||
* phyusical range, but the API functions assume the linear offset to the range
|
||||
* from 0 to metadata node size.
|
||||
*/
|
||||
@ -60,28 +60,30 @@ u##bits btrfs_get_token_##bits(struct btrfs_map_token *token, \
|
||||
const void *ptr, unsigned long off) \
|
||||
{ \
|
||||
const unsigned long member_offset = (unsigned long)ptr + off; \
|
||||
const unsigned long idx = get_eb_page_index(member_offset); \
|
||||
const unsigned long oip = get_eb_offset_in_page(token->eb, \
|
||||
member_offset); \
|
||||
const unsigned long idx = get_eb_folio_index(token->eb, member_offset); \
|
||||
const unsigned long oil = get_eb_offset_in_folio(token->eb, \
|
||||
member_offset);\
|
||||
const int unit_size = folio_size(token->eb->folios[0]); \
|
||||
const int unit_shift = folio_shift(token->eb->folios[0]); \
|
||||
const int size = sizeof(u##bits); \
|
||||
u8 lebytes[sizeof(u##bits)]; \
|
||||
const int part = PAGE_SIZE - oip; \
|
||||
const int part = unit_size - oil; \
|
||||
\
|
||||
ASSERT(token); \
|
||||
ASSERT(token->kaddr); \
|
||||
ASSERT(check_setget_bounds(token->eb, ptr, off, size)); \
|
||||
if (token->offset <= member_offset && \
|
||||
member_offset + size <= token->offset + PAGE_SIZE) { \
|
||||
return get_unaligned_le##bits(token->kaddr + oip); \
|
||||
member_offset + size <= token->offset + unit_size) { \
|
||||
return get_unaligned_le##bits(token->kaddr + oil); \
|
||||
} \
|
||||
token->kaddr = page_address(token->eb->pages[idx]); \
|
||||
token->offset = idx << PAGE_SHIFT; \
|
||||
if (INLINE_EXTENT_BUFFER_PAGES == 1 || oip + size <= PAGE_SIZE ) \
|
||||
return get_unaligned_le##bits(token->kaddr + oip); \
|
||||
token->kaddr = folio_address(token->eb->folios[idx]); \
|
||||
token->offset = idx << unit_shift; \
|
||||
if (INLINE_EXTENT_BUFFER_PAGES == 1 || oil + size <= unit_size) \
|
||||
return get_unaligned_le##bits(token->kaddr + oil); \
|
||||
\
|
||||
memcpy(lebytes, token->kaddr + oip, part); \
|
||||
token->kaddr = page_address(token->eb->pages[idx + 1]); \
|
||||
token->offset = (idx + 1) << PAGE_SHIFT; \
|
||||
memcpy(lebytes, token->kaddr + oil, part); \
|
||||
token->kaddr = folio_address(token->eb->folios[idx + 1]); \
|
||||
token->offset = (idx + 1) << unit_shift; \
|
||||
memcpy(lebytes + part, token->kaddr, size - part); \
|
||||
return get_unaligned_le##bits(lebytes); \
|
||||
} \
|
||||
@ -89,19 +91,21 @@ u##bits btrfs_get_##bits(const struct extent_buffer *eb, \
|
||||
const void *ptr, unsigned long off) \
|
||||
{ \
|
||||
const unsigned long member_offset = (unsigned long)ptr + off; \
|
||||
const unsigned long oip = get_eb_offset_in_page(eb, member_offset); \
|
||||
const unsigned long idx = get_eb_page_index(member_offset); \
|
||||
char *kaddr = page_address(eb->pages[idx]); \
|
||||
const unsigned long idx = get_eb_folio_index(eb, member_offset);\
|
||||
const unsigned long oil = get_eb_offset_in_folio(eb, \
|
||||
member_offset);\
|
||||
const int unit_size = folio_size(eb->folios[0]); \
|
||||
char *kaddr = folio_address(eb->folios[idx]); \
|
||||
const int size = sizeof(u##bits); \
|
||||
const int part = PAGE_SIZE - oip; \
|
||||
const int part = unit_size - oil; \
|
||||
u8 lebytes[sizeof(u##bits)]; \
|
||||
\
|
||||
ASSERT(check_setget_bounds(eb, ptr, off, size)); \
|
||||
if (INLINE_EXTENT_BUFFER_PAGES == 1 || oip + size <= PAGE_SIZE) \
|
||||
return get_unaligned_le##bits(kaddr + oip); \
|
||||
if (INLINE_EXTENT_BUFFER_PAGES == 1 || oil + size <= unit_size) \
|
||||
return get_unaligned_le##bits(kaddr + oil); \
|
||||
\
|
||||
memcpy(lebytes, kaddr + oip, part); \
|
||||
kaddr = page_address(eb->pages[idx + 1]); \
|
||||
memcpy(lebytes, kaddr + oil, part); \
|
||||
kaddr = folio_address(eb->folios[idx + 1]); \
|
||||
memcpy(lebytes + part, kaddr, size - part); \
|
||||
return get_unaligned_le##bits(lebytes); \
|
||||
} \
|
||||
@ -110,53 +114,59 @@ void btrfs_set_token_##bits(struct btrfs_map_token *token, \
|
||||
u##bits val) \
|
||||
{ \
|
||||
const unsigned long member_offset = (unsigned long)ptr + off; \
|
||||
const unsigned long idx = get_eb_page_index(member_offset); \
|
||||
const unsigned long oip = get_eb_offset_in_page(token->eb, \
|
||||
member_offset); \
|
||||
const unsigned long idx = get_eb_folio_index(token->eb, member_offset); \
|
||||
const unsigned long oil = get_eb_offset_in_folio(token->eb, \
|
||||
member_offset);\
|
||||
const int unit_size = folio_size(token->eb->folios[0]); \
|
||||
const int unit_shift = folio_shift(token->eb->folios[0]); \
|
||||
const int size = sizeof(u##bits); \
|
||||
u8 lebytes[sizeof(u##bits)]; \
|
||||
const int part = PAGE_SIZE - oip; \
|
||||
const int part = unit_size - oil; \
|
||||
\
|
||||
ASSERT(token); \
|
||||
ASSERT(token->kaddr); \
|
||||
ASSERT(check_setget_bounds(token->eb, ptr, off, size)); \
|
||||
if (token->offset <= member_offset && \
|
||||
member_offset + size <= token->offset + PAGE_SIZE) { \
|
||||
put_unaligned_le##bits(val, token->kaddr + oip); \
|
||||
member_offset + size <= token->offset + unit_size) { \
|
||||
put_unaligned_le##bits(val, token->kaddr + oil); \
|
||||
return; \
|
||||
} \
|
||||
token->kaddr = page_address(token->eb->pages[idx]); \
|
||||
token->offset = idx << PAGE_SHIFT; \
|
||||
if (INLINE_EXTENT_BUFFER_PAGES == 1 || oip + size <= PAGE_SIZE) { \
|
||||
put_unaligned_le##bits(val, token->kaddr + oip); \
|
||||
token->kaddr = folio_address(token->eb->folios[idx]); \
|
||||
token->offset = idx << unit_shift; \
|
||||
if (INLINE_EXTENT_BUFFER_PAGES == 1 || \
|
||||
oil + size <= unit_size) { \
|
||||
put_unaligned_le##bits(val, token->kaddr + oil); \
|
||||
return; \
|
||||
} \
|
||||
put_unaligned_le##bits(val, lebytes); \
|
||||
memcpy(token->kaddr + oip, lebytes, part); \
|
||||
token->kaddr = page_address(token->eb->pages[idx + 1]); \
|
||||
token->offset = (idx + 1) << PAGE_SHIFT; \
|
||||
memcpy(token->kaddr + oil, lebytes, part); \
|
||||
token->kaddr = folio_address(token->eb->folios[idx + 1]); \
|
||||
token->offset = (idx + 1) << unit_shift; \
|
||||
memcpy(token->kaddr, lebytes + part, size - part); \
|
||||
} \
|
||||
void btrfs_set_##bits(const struct extent_buffer *eb, void *ptr, \
|
||||
unsigned long off, u##bits val) \
|
||||
{ \
|
||||
const unsigned long member_offset = (unsigned long)ptr + off; \
|
||||
const unsigned long oip = get_eb_offset_in_page(eb, member_offset); \
|
||||
const unsigned long idx = get_eb_page_index(member_offset); \
|
||||
char *kaddr = page_address(eb->pages[idx]); \
|
||||
const unsigned long idx = get_eb_folio_index(eb, member_offset);\
|
||||
const unsigned long oil = get_eb_offset_in_folio(eb, \
|
||||
member_offset);\
|
||||
const int unit_size = folio_size(eb->folios[0]); \
|
||||
char *kaddr = folio_address(eb->folios[idx]); \
|
||||
const int size = sizeof(u##bits); \
|
||||
const int part = PAGE_SIZE - oip; \
|
||||
const int part = unit_size - oil; \
|
||||
u8 lebytes[sizeof(u##bits)]; \
|
||||
\
|
||||
ASSERT(check_setget_bounds(eb, ptr, off, size)); \
|
||||
if (INLINE_EXTENT_BUFFER_PAGES == 1 || oip + size <= PAGE_SIZE) { \
|
||||
put_unaligned_le##bits(val, kaddr + oip); \
|
||||
if (INLINE_EXTENT_BUFFER_PAGES == 1 || \
|
||||
oil + size <= unit_size) { \
|
||||
put_unaligned_le##bits(val, kaddr + oil); \
|
||||
return; \
|
||||
} \
|
||||
\
|
||||
put_unaligned_le##bits(val, lebytes); \
|
||||
memcpy(kaddr + oip, lebytes, part); \
|
||||
kaddr = page_address(eb->pages[idx + 1]); \
|
||||
memcpy(kaddr + oil, lebytes, part); \
|
||||
kaddr = folio_address(eb->folios[idx + 1]); \
|
||||
memcpy(kaddr, lebytes + part, size - part); \
|
||||
}
|
||||
|
||||
|
@ -90,14 +90,14 @@ static inline void btrfs_set_token_##name(struct btrfs_map_token *token,\
|
||||
#define BTRFS_SETGET_HEADER_FUNCS(name, type, member, bits) \
|
||||
static inline u##bits btrfs_##name(const struct extent_buffer *eb) \
|
||||
{ \
|
||||
const type *p = page_address(eb->pages[0]) + \
|
||||
const type *p = folio_address(eb->folios[0]) + \
|
||||
offset_in_page(eb->start); \
|
||||
return get_unaligned_le##bits(&p->member); \
|
||||
} \
|
||||
static inline void btrfs_set_##name(const struct extent_buffer *eb, \
|
||||
u##bits val) \
|
||||
{ \
|
||||
type *p = page_address(eb->pages[0]) + offset_in_page(eb->start); \
|
||||
type *p = folio_address(eb->folios[0]) + offset_in_page(eb->start); \
|
||||
put_unaligned_le##bits(val, &p->member); \
|
||||
}
|
||||
|
||||
|
@ -194,6 +194,12 @@ static void btrfs_end_repair_bio(struct btrfs_bio *repair_bbio,
|
||||
struct bio_vec *bv = bio_first_bvec_all(&repair_bbio->bio);
|
||||
int mirror = repair_bbio->mirror_num;
|
||||
|
||||
/*
|
||||
* We can only trigger this for data bio, which doesn't support larger
|
||||
* folios yet.
|
||||
*/
|
||||
ASSERT(folio_order(page_folio(bv->bv_page)) == 0);
|
||||
|
||||
if (repair_bbio->bio.bi_status ||
|
||||
!btrfs_data_csum_ok(repair_bbio, dev, 0, bv)) {
|
||||
bio_reset(&repair_bbio->bio, NULL, REQ_OP_READ);
|
||||
@ -215,7 +221,7 @@ static void btrfs_end_repair_bio(struct btrfs_bio *repair_bbio,
|
||||
btrfs_repair_io_failure(fs_info, btrfs_ino(inode),
|
||||
repair_bbio->file_offset, fs_info->sectorsize,
|
||||
repair_bbio->saved_iter.bi_sector << SECTOR_SHIFT,
|
||||
bv->bv_page, bv->bv_offset, mirror);
|
||||
page_folio(bv->bv_page), bv->bv_offset, mirror);
|
||||
} while (mirror != fbio->bbio->mirror_num);
|
||||
|
||||
done:
|
||||
@ -626,7 +632,7 @@ static bool should_async_write(struct btrfs_bio *bbio)
|
||||
/*
|
||||
* Submit bio to an async queue.
|
||||
*
|
||||
* Return true if the work has been succesfuly submitted, else false.
|
||||
* Return true if the work has been successfully submitted, else false.
|
||||
*/
|
||||
static bool btrfs_wq_submit_bio(struct btrfs_bio *bbio,
|
||||
struct btrfs_io_context *bioc,
|
||||
@ -767,8 +773,8 @@ void btrfs_submit_bio(struct btrfs_bio *bbio, int mirror_num)
|
||||
* freeing the bio.
|
||||
*/
|
||||
int btrfs_repair_io_failure(struct btrfs_fs_info *fs_info, u64 ino, u64 start,
|
||||
u64 length, u64 logical, struct page *page,
|
||||
unsigned int pg_offset, int mirror_num)
|
||||
u64 length, u64 logical, struct folio *folio,
|
||||
unsigned int folio_offset, int mirror_num)
|
||||
{
|
||||
struct btrfs_io_stripe smap = { 0 };
|
||||
struct bio_vec bvec;
|
||||
@ -799,7 +805,8 @@ int btrfs_repair_io_failure(struct btrfs_fs_info *fs_info, u64 ino, u64 start,
|
||||
|
||||
bio_init(&bio, smap.dev->bdev, &bvec, 1, REQ_OP_WRITE | REQ_SYNC);
|
||||
bio.bi_iter.bi_sector = smap.physical >> SECTOR_SHIFT;
|
||||
__bio_add_page(&bio, page, length, pg_offset);
|
||||
ret = bio_add_folio(&bio, folio, length, folio_offset);
|
||||
ASSERT(ret);
|
||||
ret = submit_bio_wait(&bio);
|
||||
if (ret) {
|
||||
/* try to remap that extent elsewhere? */
|
||||
|
@ -105,7 +105,7 @@ void btrfs_bio_end_io(struct btrfs_bio *bbio, blk_status_t status);
|
||||
void btrfs_submit_bio(struct btrfs_bio *bbio, int mirror_num);
|
||||
void btrfs_submit_repair_write(struct btrfs_bio *bbio, int mirror_num, bool dev_replace);
|
||||
int btrfs_repair_io_failure(struct btrfs_fs_info *fs_info, u64 ino, u64 start,
|
||||
u64 length, u64 logical, struct page *page,
|
||||
unsigned int pg_offset, int mirror_num);
|
||||
u64 length, u64 logical, struct folio *folio,
|
||||
unsigned int folio_offset, int mirror_num);
|
||||
|
||||
#endif
|
||||
|
@ -168,7 +168,7 @@ void btrfs_put_block_group(struct btrfs_block_group *cache)
|
||||
cache);
|
||||
|
||||
kfree(cache->free_space_ctl);
|
||||
kfree(cache->physical_map);
|
||||
btrfs_free_chunk_map(cache->physical_map);
|
||||
kfree(cache);
|
||||
}
|
||||
}
|
||||
@ -1047,7 +1047,7 @@ static int remove_block_group_item(struct btrfs_trans_handle *trans,
|
||||
}
|
||||
|
||||
int btrfs_remove_block_group(struct btrfs_trans_handle *trans,
|
||||
u64 group_start, struct extent_map *em)
|
||||
struct btrfs_chunk_map *map)
|
||||
{
|
||||
struct btrfs_fs_info *fs_info = trans->fs_info;
|
||||
struct btrfs_path *path;
|
||||
@ -1059,10 +1059,10 @@ int btrfs_remove_block_group(struct btrfs_trans_handle *trans,
|
||||
int index;
|
||||
int factor;
|
||||
struct btrfs_caching_control *caching_ctl = NULL;
|
||||
bool remove_em;
|
||||
bool remove_map;
|
||||
bool remove_rsv = false;
|
||||
|
||||
block_group = btrfs_lookup_block_group(fs_info, group_start);
|
||||
block_group = btrfs_lookup_block_group(fs_info, map->start);
|
||||
BUG_ON(!block_group);
|
||||
BUG_ON(!block_group->ro);
|
||||
|
||||
@ -1252,7 +1252,7 @@ int btrfs_remove_block_group(struct btrfs_trans_handle *trans,
|
||||
* entries because we already removed them all when we called
|
||||
* btrfs_remove_free_space_cache().
|
||||
*
|
||||
* And we must not remove the extent map from the fs_info->mapping_tree
|
||||
* And we must not remove the chunk map from the fs_info->mapping_tree
|
||||
* to prevent the same logical address range and physical device space
|
||||
* ranges from being reused for a new block group. This is needed to
|
||||
* avoid races with trimming and scrub.
|
||||
@ -1268,19 +1268,11 @@ int btrfs_remove_block_group(struct btrfs_trans_handle *trans,
|
||||
* in place until the extents have been discarded completely when
|
||||
* the transaction commit has completed.
|
||||
*/
|
||||
remove_em = (atomic_read(&block_group->frozen) == 0);
|
||||
remove_map = (atomic_read(&block_group->frozen) == 0);
|
||||
spin_unlock(&block_group->lock);
|
||||
|
||||
if (remove_em) {
|
||||
struct extent_map_tree *em_tree;
|
||||
|
||||
em_tree = &fs_info->mapping_tree;
|
||||
write_lock(&em_tree->lock);
|
||||
remove_extent_mapping(em_tree, em);
|
||||
write_unlock(&em_tree->lock);
|
||||
/* once for the tree */
|
||||
free_extent_map(em);
|
||||
}
|
||||
if (remove_map)
|
||||
btrfs_remove_chunk_map(fs_info, map);
|
||||
|
||||
out:
|
||||
/* Once for the lookup reference */
|
||||
@ -1295,15 +1287,12 @@ struct btrfs_trans_handle *btrfs_start_trans_remove_block_group(
|
||||
struct btrfs_fs_info *fs_info, const u64 chunk_offset)
|
||||
{
|
||||
struct btrfs_root *root = btrfs_block_group_root(fs_info);
|
||||
struct extent_map_tree *em_tree = &fs_info->mapping_tree;
|
||||
struct extent_map *em;
|
||||
struct map_lookup *map;
|
||||
struct btrfs_chunk_map *map;
|
||||
unsigned int num_items;
|
||||
|
||||
read_lock(&em_tree->lock);
|
||||
em = lookup_extent_mapping(em_tree, chunk_offset, 1);
|
||||
read_unlock(&em_tree->lock);
|
||||
ASSERT(em && em->start == chunk_offset);
|
||||
map = btrfs_find_chunk_map(fs_info, chunk_offset, 1);
|
||||
ASSERT(map != NULL);
|
||||
ASSERT(map->start == chunk_offset);
|
||||
|
||||
/*
|
||||
* We need to reserve 3 + N units from the metadata space info in order
|
||||
@ -1324,9 +1313,8 @@ struct btrfs_trans_handle *btrfs_start_trans_remove_block_group(
|
||||
* more device items and remove one chunk item), but this is done at
|
||||
* btrfs_remove_chunk() through a call to check_system_chunk().
|
||||
*/
|
||||
map = em->map_lookup;
|
||||
num_items = 3 + map->num_stripes;
|
||||
free_extent_map(em);
|
||||
btrfs_free_chunk_map(map);
|
||||
|
||||
return btrfs_start_transaction_fallback_global_rsv(root, num_items);
|
||||
}
|
||||
@ -1927,8 +1915,7 @@ void btrfs_mark_bg_to_reclaim(struct btrfs_block_group *bg)
|
||||
static int read_bg_from_eb(struct btrfs_fs_info *fs_info, struct btrfs_key *key,
|
||||
struct btrfs_path *path)
|
||||
{
|
||||
struct extent_map_tree *em_tree;
|
||||
struct extent_map *em;
|
||||
struct btrfs_chunk_map *map;
|
||||
struct btrfs_block_group_item bg;
|
||||
struct extent_buffer *leaf;
|
||||
int slot;
|
||||
@ -1938,23 +1925,20 @@ static int read_bg_from_eb(struct btrfs_fs_info *fs_info, struct btrfs_key *key,
|
||||
slot = path->slots[0];
|
||||
leaf = path->nodes[0];
|
||||
|
||||
em_tree = &fs_info->mapping_tree;
|
||||
read_lock(&em_tree->lock);
|
||||
em = lookup_extent_mapping(em_tree, key->objectid, key->offset);
|
||||
read_unlock(&em_tree->lock);
|
||||
if (!em) {
|
||||
map = btrfs_find_chunk_map(fs_info, key->objectid, key->offset);
|
||||
if (!map) {
|
||||
btrfs_err(fs_info,
|
||||
"logical %llu len %llu found bg but no related chunk",
|
||||
key->objectid, key->offset);
|
||||
return -ENOENT;
|
||||
}
|
||||
|
||||
if (em->start != key->objectid || em->len != key->offset) {
|
||||
if (map->start != key->objectid || map->chunk_len != key->offset) {
|
||||
btrfs_err(fs_info,
|
||||
"block group %llu len %llu mismatch with chunk %llu len %llu",
|
||||
key->objectid, key->offset, em->start, em->len);
|
||||
key->objectid, key->offset, map->start, map->chunk_len);
|
||||
ret = -EUCLEAN;
|
||||
goto out_free_em;
|
||||
goto out_free_map;
|
||||
}
|
||||
|
||||
read_extent_buffer(leaf, &bg, btrfs_item_ptr_offset(leaf, slot),
|
||||
@ -1962,16 +1946,16 @@ static int read_bg_from_eb(struct btrfs_fs_info *fs_info, struct btrfs_key *key,
|
||||
flags = btrfs_stack_block_group_flags(&bg) &
|
||||
BTRFS_BLOCK_GROUP_TYPE_MASK;
|
||||
|
||||
if (flags != (em->map_lookup->type & BTRFS_BLOCK_GROUP_TYPE_MASK)) {
|
||||
if (flags != (map->type & BTRFS_BLOCK_GROUP_TYPE_MASK)) {
|
||||
btrfs_err(fs_info,
|
||||
"block group %llu len %llu type flags 0x%llx mismatch with chunk type flags 0x%llx",
|
||||
key->objectid, key->offset, flags,
|
||||
(BTRFS_BLOCK_GROUP_TYPE_MASK & em->map_lookup->type));
|
||||
(BTRFS_BLOCK_GROUP_TYPE_MASK & map->type));
|
||||
ret = -EUCLEAN;
|
||||
}
|
||||
|
||||
out_free_em:
|
||||
free_extent_map(em);
|
||||
out_free_map:
|
||||
btrfs_free_chunk_map(map);
|
||||
return ret;
|
||||
}
|
||||
|
||||
@ -2024,8 +2008,7 @@ static void set_avail_alloc_bits(struct btrfs_fs_info *fs_info, u64 flags)
|
||||
int btrfs_rmap_block(struct btrfs_fs_info *fs_info, u64 chunk_start,
|
||||
u64 physical, u64 **logical, int *naddrs, int *stripe_len)
|
||||
{
|
||||
struct extent_map *em;
|
||||
struct map_lookup *map;
|
||||
struct btrfs_chunk_map *map;
|
||||
u64 *buf;
|
||||
u64 bytenr;
|
||||
u64 data_stripe_length;
|
||||
@ -2033,14 +2016,13 @@ int btrfs_rmap_block(struct btrfs_fs_info *fs_info, u64 chunk_start,
|
||||
int i, nr = 0;
|
||||
int ret = 0;
|
||||
|
||||
em = btrfs_get_chunk_map(fs_info, chunk_start, 1);
|
||||
if (IS_ERR(em))
|
||||
map = btrfs_get_chunk_map(fs_info, chunk_start, 1);
|
||||
if (IS_ERR(map))
|
||||
return -EIO;
|
||||
|
||||
map = em->map_lookup;
|
||||
data_stripe_length = em->orig_block_len;
|
||||
data_stripe_length = map->stripe_size;
|
||||
io_stripe_size = BTRFS_STRIPE_LEN;
|
||||
chunk_start = em->start;
|
||||
chunk_start = map->start;
|
||||
|
||||
/* For RAID5/6 adjust to a full IO stripe length */
|
||||
if (map->type & BTRFS_BLOCK_GROUP_RAID56_MASK)
|
||||
@ -2094,7 +2076,7 @@ int btrfs_rmap_block(struct btrfs_fs_info *fs_info, u64 chunk_start,
|
||||
*naddrs = nr;
|
||||
*stripe_len = io_stripe_size;
|
||||
out:
|
||||
free_extent_map(em);
|
||||
btrfs_free_chunk_map(map);
|
||||
return ret;
|
||||
}
|
||||
|
||||
@ -2199,49 +2181,47 @@ static struct btrfs_block_group *btrfs_create_block_group_cache(
|
||||
*/
|
||||
static int check_chunk_block_group_mappings(struct btrfs_fs_info *fs_info)
|
||||
{
|
||||
struct extent_map_tree *map_tree = &fs_info->mapping_tree;
|
||||
struct extent_map *em;
|
||||
struct btrfs_block_group *bg;
|
||||
u64 start = 0;
|
||||
int ret = 0;
|
||||
|
||||
while (1) {
|
||||
read_lock(&map_tree->lock);
|
||||
struct btrfs_chunk_map *map;
|
||||
struct btrfs_block_group *bg;
|
||||
|
||||
/*
|
||||
* lookup_extent_mapping will return the first extent map
|
||||
* intersecting the range, so setting @len to 1 is enough to
|
||||
* btrfs_find_chunk_map() will return the first chunk map
|
||||
* intersecting the range, so setting @length to 1 is enough to
|
||||
* get the first chunk.
|
||||
*/
|
||||
em = lookup_extent_mapping(map_tree, start, 1);
|
||||
read_unlock(&map_tree->lock);
|
||||
if (!em)
|
||||
map = btrfs_find_chunk_map(fs_info, start, 1);
|
||||
if (!map)
|
||||
break;
|
||||
|
||||
bg = btrfs_lookup_block_group(fs_info, em->start);
|
||||
bg = btrfs_lookup_block_group(fs_info, map->start);
|
||||
if (!bg) {
|
||||
btrfs_err(fs_info,
|
||||
"chunk start=%llu len=%llu doesn't have corresponding block group",
|
||||
em->start, em->len);
|
||||
map->start, map->chunk_len);
|
||||
ret = -EUCLEAN;
|
||||
free_extent_map(em);
|
||||
btrfs_free_chunk_map(map);
|
||||
break;
|
||||
}
|
||||
if (bg->start != em->start || bg->length != em->len ||
|
||||
if (bg->start != map->start || bg->length != map->chunk_len ||
|
||||
(bg->flags & BTRFS_BLOCK_GROUP_TYPE_MASK) !=
|
||||
(em->map_lookup->type & BTRFS_BLOCK_GROUP_TYPE_MASK)) {
|
||||
(map->type & BTRFS_BLOCK_GROUP_TYPE_MASK)) {
|
||||
btrfs_err(fs_info,
|
||||
"chunk start=%llu len=%llu flags=0x%llx doesn't match block group start=%llu len=%llu flags=0x%llx",
|
||||
em->start, em->len,
|
||||
em->map_lookup->type & BTRFS_BLOCK_GROUP_TYPE_MASK,
|
||||
map->start, map->chunk_len,
|
||||
map->type & BTRFS_BLOCK_GROUP_TYPE_MASK,
|
||||
bg->start, bg->length,
|
||||
bg->flags & BTRFS_BLOCK_GROUP_TYPE_MASK);
|
||||
ret = -EUCLEAN;
|
||||
free_extent_map(em);
|
||||
btrfs_free_chunk_map(map);
|
||||
btrfs_put_block_group(bg);
|
||||
break;
|
||||
}
|
||||
start = em->start + em->len;
|
||||
free_extent_map(em);
|
||||
start = map->start + map->chunk_len;
|
||||
btrfs_free_chunk_map(map);
|
||||
btrfs_put_block_group(bg);
|
||||
}
|
||||
return ret;
|
||||
@ -2369,28 +2349,25 @@ error:
|
||||
|
||||
static int fill_dummy_bgs(struct btrfs_fs_info *fs_info)
|
||||
{
|
||||
struct extent_map_tree *em_tree = &fs_info->mapping_tree;
|
||||
struct rb_node *node;
|
||||
int ret = 0;
|
||||
|
||||
for (node = rb_first_cached(&em_tree->map); node; node = rb_next(node)) {
|
||||
struct extent_map *em;
|
||||
struct map_lookup *map;
|
||||
for (node = rb_first_cached(&fs_info->mapping_tree); node; node = rb_next(node)) {
|
||||
struct btrfs_chunk_map *map;
|
||||
struct btrfs_block_group *bg;
|
||||
|
||||
em = rb_entry(node, struct extent_map, rb_node);
|
||||
map = em->map_lookup;
|
||||
bg = btrfs_create_block_group_cache(fs_info, em->start);
|
||||
map = rb_entry(node, struct btrfs_chunk_map, rb_node);
|
||||
bg = btrfs_create_block_group_cache(fs_info, map->start);
|
||||
if (!bg) {
|
||||
ret = -ENOMEM;
|
||||
break;
|
||||
}
|
||||
|
||||
/* Fill dummy cache as FULL */
|
||||
bg->length = em->len;
|
||||
bg->length = map->chunk_len;
|
||||
bg->flags = map->type;
|
||||
bg->cached = BTRFS_CACHE_FINISHED;
|
||||
bg->used = em->len;
|
||||
bg->used = map->chunk_len;
|
||||
bg->flags = map->type;
|
||||
ret = btrfs_add_block_group_cache(fs_info, bg);
|
||||
/*
|
||||
@ -2618,19 +2595,14 @@ static int insert_dev_extents(struct btrfs_trans_handle *trans,
|
||||
{
|
||||
struct btrfs_fs_info *fs_info = trans->fs_info;
|
||||
struct btrfs_device *device;
|
||||
struct extent_map *em;
|
||||
struct map_lookup *map;
|
||||
struct btrfs_chunk_map *map;
|
||||
u64 dev_offset;
|
||||
u64 stripe_size;
|
||||
int i;
|
||||
int ret = 0;
|
||||
|
||||
em = btrfs_get_chunk_map(fs_info, chunk_offset, chunk_size);
|
||||
if (IS_ERR(em))
|
||||
return PTR_ERR(em);
|
||||
|
||||
map = em->map_lookup;
|
||||
stripe_size = em->orig_block_len;
|
||||
map = btrfs_get_chunk_map(fs_info, chunk_offset, chunk_size);
|
||||
if (IS_ERR(map))
|
||||
return PTR_ERR(map);
|
||||
|
||||
/*
|
||||
* Take the device list mutex to prevent races with the final phase of
|
||||
@ -2647,13 +2619,13 @@ static int insert_dev_extents(struct btrfs_trans_handle *trans,
|
||||
dev_offset = map->stripes[i].physical;
|
||||
|
||||
ret = insert_dev_extent(trans, device, chunk_offset, dev_offset,
|
||||
stripe_size);
|
||||
map->stripe_size);
|
||||
if (ret)
|
||||
break;
|
||||
}
|
||||
mutex_unlock(&fs_info->fs_devices->device_list_mutex);
|
||||
|
||||
free_extent_map(em);
|
||||
btrfs_free_chunk_map(map);
|
||||
return ret;
|
||||
}
|
||||
|
||||
@ -2910,7 +2882,7 @@ int btrfs_inc_block_group_ro(struct btrfs_block_group *cache,
|
||||
goto unlock_out;
|
||||
|
||||
/*
|
||||
* Skip chunk alloction if the bg is SYSTEM, this is to avoid system
|
||||
* Skip chunk allocation if the bg is SYSTEM, this is to avoid system
|
||||
* chunk allocation storm to exhaust the system chunk array. Otherwise
|
||||
* we still want to try our best to mark the block group read-only.
|
||||
*/
|
||||
@ -4406,8 +4378,6 @@ void btrfs_freeze_block_group(struct btrfs_block_group *cache)
|
||||
void btrfs_unfreeze_block_group(struct btrfs_block_group *block_group)
|
||||
{
|
||||
struct btrfs_fs_info *fs_info = block_group->fs_info;
|
||||
struct extent_map_tree *em_tree;
|
||||
struct extent_map *em;
|
||||
bool cleanup;
|
||||
|
||||
spin_lock(&block_group->lock);
|
||||
@ -4416,17 +4386,16 @@ void btrfs_unfreeze_block_group(struct btrfs_block_group *block_group)
|
||||
spin_unlock(&block_group->lock);
|
||||
|
||||
if (cleanup) {
|
||||
em_tree = &fs_info->mapping_tree;
|
||||
write_lock(&em_tree->lock);
|
||||
em = lookup_extent_mapping(em_tree, block_group->start,
|
||||
1);
|
||||
BUG_ON(!em); /* logic error, can't happen */
|
||||
remove_extent_mapping(em_tree, em);
|
||||
write_unlock(&em_tree->lock);
|
||||
struct btrfs_chunk_map *map;
|
||||
|
||||
/* once for us and once for the tree */
|
||||
free_extent_map(em);
|
||||
free_extent_map(em);
|
||||
map = btrfs_find_chunk_map(fs_info, block_group->start, 1);
|
||||
/* Logic error, can't happen. */
|
||||
ASSERT(map);
|
||||
|
||||
btrfs_remove_chunk_map(fs_info, map);
|
||||
|
||||
/* Once for our lookup reference. */
|
||||
btrfs_free_chunk_map(map);
|
||||
|
||||
/*
|
||||
* We may have left one free space entry and other possible
|
||||
|
@ -5,6 +5,8 @@
|
||||
|
||||
#include "free-space-cache.h"
|
||||
|
||||
struct btrfs_chunk_map;
|
||||
|
||||
enum btrfs_disk_cache_state {
|
||||
BTRFS_DC_WRITTEN,
|
||||
BTRFS_DC_ERROR,
|
||||
@ -243,7 +245,7 @@ struct btrfs_block_group {
|
||||
u64 zone_unusable;
|
||||
u64 zone_capacity;
|
||||
u64 meta_write_pointer;
|
||||
struct map_lookup *physical_map;
|
||||
struct btrfs_chunk_map *physical_map;
|
||||
struct list_head active_bg_list;
|
||||
struct work_struct zone_finish_work;
|
||||
struct extent_buffer *last_eb;
|
||||
@ -297,7 +299,7 @@ struct btrfs_trans_handle *btrfs_start_trans_remove_block_group(
|
||||
struct btrfs_fs_info *fs_info,
|
||||
const u64 chunk_offset);
|
||||
int btrfs_remove_block_group(struct btrfs_trans_handle *trans,
|
||||
u64 group_start, struct extent_map *em);
|
||||
struct btrfs_chunk_map *map);
|
||||
void btrfs_delete_unused_bgs(struct btrfs_fs_info *fs_info);
|
||||
void btrfs_mark_bg_unused(struct btrfs_block_group *bg);
|
||||
void btrfs_reclaim_bgs_work(struct work_struct *work);
|
||||
|
@ -69,6 +69,8 @@ enum {
|
||||
BTRFS_INODE_VERITY_IN_PROGRESS,
|
||||
/* Set when this inode is a free space inode. */
|
||||
BTRFS_INODE_FREE_SPACE_INODE,
|
||||
/* Set when there are no capabilities in XATTs for the inode. */
|
||||
BTRFS_INODE_NO_CAP_XATTR,
|
||||
};
|
||||
|
||||
/* in memory btrfs inode */
|
||||
@ -107,9 +109,11 @@ struct btrfs_inode {
|
||||
|
||||
/*
|
||||
* Keep track of where the inode has extent items mapped in order to
|
||||
* make sure the i_size adjustments are accurate
|
||||
* make sure the i_size adjustments are accurate. Not required when the
|
||||
* filesystem is NO_HOLES, the status can't be set while mounted as
|
||||
* it's a mkfs-time feature.
|
||||
*/
|
||||
struct extent_io_tree file_extent_tree;
|
||||
struct extent_io_tree *file_extent_tree;
|
||||
|
||||
/* held while logging the inode in tree-log.c */
|
||||
struct mutex log_mutex;
|
||||
@ -487,7 +491,7 @@ struct inode *btrfs_iget_path(struct super_block *s, u64 ino,
|
||||
struct inode *btrfs_iget(struct super_block *s, u64 ino, struct btrfs_root *root);
|
||||
struct extent_map *btrfs_get_extent(struct btrfs_inode *inode,
|
||||
struct page *page, size_t pg_offset,
|
||||
u64 start, u64 end);
|
||||
u64 start, u64 len);
|
||||
int btrfs_update_inode(struct btrfs_trans_handle *trans,
|
||||
struct btrfs_inode *inode);
|
||||
int btrfs_update_inode_fallback(struct btrfs_trans_handle *trans,
|
||||
|
@ -20,6 +20,7 @@
|
||||
#include <linux/slab.h>
|
||||
#include <linux/sched/mm.h>
|
||||
#include <linux/log2.h>
|
||||
#include <linux/shrinker.h>
|
||||
#include <crypto/hash.h>
|
||||
#include "misc.h"
|
||||
#include "ctree.h"
|
||||
@ -163,13 +164,107 @@ static int compression_decompress(int type, struct list_head *ws,
|
||||
static void btrfs_free_compressed_pages(struct compressed_bio *cb)
|
||||
{
|
||||
for (unsigned int i = 0; i < cb->nr_pages; i++)
|
||||
put_page(cb->compressed_pages[i]);
|
||||
btrfs_free_compr_page(cb->compressed_pages[i]);
|
||||
kfree(cb->compressed_pages);
|
||||
}
|
||||
|
||||
static int btrfs_decompress_bio(struct compressed_bio *cb);
|
||||
|
||||
static void end_compressed_bio_read(struct btrfs_bio *bbio)
|
||||
/*
|
||||
* Global cache of last unused pages for compression/decompression.
|
||||
*/
|
||||
static struct btrfs_compr_pool {
|
||||
struct shrinker *shrinker;
|
||||
spinlock_t lock;
|
||||
struct list_head list;
|
||||
int count;
|
||||
int thresh;
|
||||
} compr_pool;
|
||||
|
||||
static unsigned long btrfs_compr_pool_count(struct shrinker *sh, struct shrink_control *sc)
|
||||
{
|
||||
int ret;
|
||||
|
||||
/*
|
||||
* We must not read the values more than once if 'ret' gets expanded in
|
||||
* the return statement so we don't accidentally return a negative
|
||||
* number, even if the first condition finds it positive.
|
||||
*/
|
||||
ret = READ_ONCE(compr_pool.count) - READ_ONCE(compr_pool.thresh);
|
||||
|
||||
return ret > 0 ? ret : 0;
|
||||
}
|
||||
|
||||
static unsigned long btrfs_compr_pool_scan(struct shrinker *sh, struct shrink_control *sc)
|
||||
{
|
||||
struct list_head remove;
|
||||
struct list_head *tmp, *next;
|
||||
int freed;
|
||||
|
||||
if (compr_pool.count == 0)
|
||||
return SHRINK_STOP;
|
||||
|
||||
INIT_LIST_HEAD(&remove);
|
||||
|
||||
/* For now, just simply drain the whole list. */
|
||||
spin_lock(&compr_pool.lock);
|
||||
list_splice_init(&compr_pool.list, &remove);
|
||||
freed = compr_pool.count;
|
||||
compr_pool.count = 0;
|
||||
spin_unlock(&compr_pool.lock);
|
||||
|
||||
list_for_each_safe(tmp, next, &remove) {
|
||||
struct page *page = list_entry(tmp, struct page, lru);
|
||||
|
||||
ASSERT(page_ref_count(page) == 1);
|
||||
put_page(page);
|
||||
}
|
||||
|
||||
return freed;
|
||||
}
|
||||
|
||||
/*
|
||||
* Common wrappers for page allocation from compression wrappers
|
||||
*/
|
||||
struct page *btrfs_alloc_compr_page(void)
|
||||
{
|
||||
struct page *page = NULL;
|
||||
|
||||
spin_lock(&compr_pool.lock);
|
||||
if (compr_pool.count > 0) {
|
||||
page = list_first_entry(&compr_pool.list, struct page, lru);
|
||||
list_del_init(&page->lru);
|
||||
compr_pool.count--;
|
||||
}
|
||||
spin_unlock(&compr_pool.lock);
|
||||
|
||||
if (page)
|
||||
return page;
|
||||
|
||||
return alloc_page(GFP_NOFS);
|
||||
}
|
||||
|
||||
void btrfs_free_compr_page(struct page *page)
|
||||
{
|
||||
bool do_free = false;
|
||||
|
||||
spin_lock(&compr_pool.lock);
|
||||
if (compr_pool.count > compr_pool.thresh) {
|
||||
do_free = true;
|
||||
} else {
|
||||
list_add(&page->lru, &compr_pool.list);
|
||||
compr_pool.count++;
|
||||
}
|
||||
spin_unlock(&compr_pool.lock);
|
||||
|
||||
if (!do_free)
|
||||
return;
|
||||
|
||||
ASSERT(page_ref_count(page) == 1);
|
||||
put_page(page);
|
||||
}
|
||||
|
||||
static void end_bbio_comprssed_read(struct btrfs_bio *bbio)
|
||||
{
|
||||
struct compressed_bio *cb = to_compressed_bio(bbio);
|
||||
blk_status_t status = bbio->bio.bi_status;
|
||||
@ -211,8 +306,8 @@ static noinline void end_compressed_writeback(const struct compressed_bio *cb)
|
||||
for (i = 0; i < ret; i++) {
|
||||
struct folio *folio = fbatch.folios[i];
|
||||
|
||||
btrfs_page_clamp_clear_writeback(fs_info, &folio->page,
|
||||
cb->start, cb->len);
|
||||
btrfs_folio_clamp_clear_writeback(fs_info, folio,
|
||||
cb->start, cb->len);
|
||||
}
|
||||
folio_batch_release(&fbatch);
|
||||
}
|
||||
@ -242,7 +337,7 @@ static void btrfs_finish_compressed_write_work(struct work_struct *work)
|
||||
* This also calls the writeback end hooks for the file pages so that metadata
|
||||
* and checksums can be updated in the file.
|
||||
*/
|
||||
static void end_compressed_bio_write(struct btrfs_bio *bbio)
|
||||
static void end_bbio_comprssed_write(struct btrfs_bio *bbio)
|
||||
{
|
||||
struct compressed_bio *cb = to_compressed_bio(bbio);
|
||||
struct btrfs_fs_info *fs_info = bbio->inode->root->fs_info;
|
||||
@ -289,7 +384,7 @@ void btrfs_submit_compressed_write(struct btrfs_ordered_extent *ordered,
|
||||
|
||||
cb = alloc_compressed_bio(inode, ordered->file_offset,
|
||||
REQ_OP_WRITE | write_flags,
|
||||
end_compressed_bio_write);
|
||||
end_bbio_comprssed_write);
|
||||
cb->start = ordered->file_offset;
|
||||
cb->len = ordered->num_bytes;
|
||||
cb->compressed_pages = compressed_pages;
|
||||
@ -446,7 +541,8 @@ static noinline int add_ra_bio_pages(struct inode *inode,
|
||||
* subpage::readers and to unlock the page.
|
||||
*/
|
||||
if (fs_info->sectorsize < PAGE_SIZE)
|
||||
btrfs_subpage_start_reader(fs_info, page, cur, add_size);
|
||||
btrfs_subpage_start_reader(fs_info, page_folio(page),
|
||||
cur, add_size);
|
||||
put_page(page);
|
||||
cur += add_size;
|
||||
}
|
||||
@ -489,11 +585,11 @@ void btrfs_submit_compressed_read(struct btrfs_bio *bbio)
|
||||
goto out;
|
||||
}
|
||||
|
||||
ASSERT(em->compress_type != BTRFS_COMPRESS_NONE);
|
||||
ASSERT(extent_map_is_compressed(em));
|
||||
compressed_len = em->block_len;
|
||||
|
||||
cb = alloc_compressed_bio(inode, file_offset, REQ_OP_READ,
|
||||
end_compressed_bio_read);
|
||||
end_bbio_comprssed_read);
|
||||
|
||||
cb->start = em->orig_start;
|
||||
em_len = em->len;
|
||||
@ -501,7 +597,7 @@ void btrfs_submit_compressed_read(struct btrfs_bio *bbio)
|
||||
|
||||
cb->len = bbio->bio.bi_iter.bi_size;
|
||||
cb->compressed_len = compressed_len;
|
||||
cb->compress_type = em->compress_type;
|
||||
cb->compress_type = extent_map_compression(em);
|
||||
cb->orig_bbio = bbio;
|
||||
|
||||
free_extent_map(em);
|
||||
@ -513,7 +609,7 @@ void btrfs_submit_compressed_read(struct btrfs_bio *bbio)
|
||||
goto out_free_bio;
|
||||
}
|
||||
|
||||
ret2 = btrfs_alloc_page_array(cb->nr_pages, cb->compressed_pages);
|
||||
ret2 = btrfs_alloc_page_array(cb->nr_pages, cb->compressed_pages, 0);
|
||||
if (ret2) {
|
||||
ret = BLK_STS_RESOURCE;
|
||||
goto out_free_compressed_pages;
|
||||
@ -960,15 +1056,36 @@ int __init btrfs_init_compress(void)
|
||||
offsetof(struct compressed_bio, bbio.bio),
|
||||
BIOSET_NEED_BVECS))
|
||||
return -ENOMEM;
|
||||
|
||||
compr_pool.shrinker = shrinker_alloc(SHRINKER_NONSLAB, "btrfs-compr-pages");
|
||||
if (!compr_pool.shrinker)
|
||||
return -ENOMEM;
|
||||
|
||||
btrfs_init_workspace_manager(BTRFS_COMPRESS_NONE);
|
||||
btrfs_init_workspace_manager(BTRFS_COMPRESS_ZLIB);
|
||||
btrfs_init_workspace_manager(BTRFS_COMPRESS_LZO);
|
||||
zstd_init_workspace_manager();
|
||||
|
||||
spin_lock_init(&compr_pool.lock);
|
||||
INIT_LIST_HEAD(&compr_pool.list);
|
||||
compr_pool.count = 0;
|
||||
/* 128K / 4K = 32, for 8 threads is 256 pages. */
|
||||
compr_pool.thresh = BTRFS_MAX_COMPRESSED / PAGE_SIZE * 8;
|
||||
compr_pool.shrinker->count_objects = btrfs_compr_pool_count;
|
||||
compr_pool.shrinker->scan_objects = btrfs_compr_pool_scan;
|
||||
compr_pool.shrinker->batch = 32;
|
||||
compr_pool.shrinker->seeks = DEFAULT_SEEKS;
|
||||
shrinker_register(compr_pool.shrinker);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
void __cold btrfs_exit_compress(void)
|
||||
{
|
||||
/* For now scan drains all pages and does not touch the parameters. */
|
||||
btrfs_compr_pool_scan(NULL, NULL);
|
||||
shrinker_free(compr_pool.shrinker);
|
||||
|
||||
btrfs_cleanup_workspace_manager(BTRFS_COMPRESS_NONE);
|
||||
btrfs_cleanup_workspace_manager(BTRFS_COMPRESS_ZLIB);
|
||||
btrfs_cleanup_workspace_manager(BTRFS_COMPRESS_LZO);
|
||||
|
@ -32,6 +32,8 @@ static_assert((BTRFS_MAX_COMPRESSED % PAGE_SIZE) == 0);
|
||||
|
||||
#define BTRFS_ZLIB_DEFAULT_LEVEL 3
|
||||
|
||||
struct page;
|
||||
|
||||
struct compressed_bio {
|
||||
/* Number of compressed pages in the array */
|
||||
unsigned int nr_pages;
|
||||
@ -96,6 +98,9 @@ void btrfs_submit_compressed_read(struct btrfs_bio *bbio);
|
||||
|
||||
unsigned int btrfs_compress_str2level(unsigned int type, const char *str);
|
||||
|
||||
struct page *btrfs_alloc_compr_page(void);
|
||||
void btrfs_free_compr_page(struct page *page);
|
||||
|
||||
enum btrfs_compression_type {
|
||||
BTRFS_COMPRESS_NONE = 0,
|
||||
BTRFS_COMPRESS_ZLIB = 1,
|
||||
|
@ -370,33 +370,41 @@ int btrfs_copy_root(struct btrfs_trans_handle *trans,
|
||||
/*
|
||||
* check if the tree block can be shared by multiple trees
|
||||
*/
|
||||
int btrfs_block_can_be_shared(struct btrfs_trans_handle *trans,
|
||||
struct btrfs_root *root,
|
||||
struct extent_buffer *buf)
|
||||
bool btrfs_block_can_be_shared(struct btrfs_trans_handle *trans,
|
||||
struct btrfs_root *root,
|
||||
struct extent_buffer *buf)
|
||||
{
|
||||
const u64 buf_gen = btrfs_header_generation(buf);
|
||||
|
||||
/*
|
||||
* Tree blocks not in shareable trees and tree roots are never shared.
|
||||
* If a block was allocated after the last snapshot and the block was
|
||||
* not allocated by tree relocation, we know the block is not shared.
|
||||
*/
|
||||
if (test_bit(BTRFS_ROOT_SHAREABLE, &root->state) &&
|
||||
buf != root->node &&
|
||||
(btrfs_header_generation(buf) <=
|
||||
btrfs_root_last_snapshot(&root->root_item) ||
|
||||
btrfs_header_flag(buf, BTRFS_HEADER_FLAG_RELOC))) {
|
||||
if (buf != root->commit_root)
|
||||
return 1;
|
||||
/*
|
||||
* An extent buffer that used to be the commit root may still be
|
||||
* shared because the tree height may have increased and it
|
||||
* became a child of a higher level root. This can happen when
|
||||
* snapshotting a subvolume created in the current transaction.
|
||||
*/
|
||||
if (btrfs_header_generation(buf) == trans->transid)
|
||||
return 1;
|
||||
}
|
||||
|
||||
return 0;
|
||||
if (!test_bit(BTRFS_ROOT_SHAREABLE, &root->state))
|
||||
return false;
|
||||
|
||||
if (buf == root->node)
|
||||
return false;
|
||||
|
||||
if (buf_gen > btrfs_root_last_snapshot(&root->root_item) &&
|
||||
!btrfs_header_flag(buf, BTRFS_HEADER_FLAG_RELOC))
|
||||
return false;
|
||||
|
||||
if (buf != root->commit_root)
|
||||
return true;
|
||||
|
||||
/*
|
||||
* An extent buffer that used to be the commit root may still be shared
|
||||
* because the tree height may have increased and it became a child of a
|
||||
* higher level root. This can happen when snapshotting a subvolume
|
||||
* created in the current transaction.
|
||||
*/
|
||||
if (buf_gen == trans->transid)
|
||||
return true;
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
static noinline int update_ref_for_cow(struct btrfs_trans_handle *trans,
|
||||
@ -812,7 +820,8 @@ int btrfs_bin_search(struct extent_buffer *eb, int first_slot,
|
||||
}
|
||||
|
||||
while (low < high) {
|
||||
unsigned long oip;
|
||||
const int unit_size = folio_size(eb->folios[0]);
|
||||
unsigned long oil;
|
||||
unsigned long offset;
|
||||
struct btrfs_disk_key *tmp;
|
||||
struct btrfs_disk_key unaligned;
|
||||
@ -820,14 +829,14 @@ int btrfs_bin_search(struct extent_buffer *eb, int first_slot,
|
||||
|
||||
mid = (low + high) / 2;
|
||||
offset = p + mid * item_size;
|
||||
oip = offset_in_page(offset);
|
||||
oil = get_eb_offset_in_folio(eb, offset);
|
||||
|
||||
if (oip + key_size <= PAGE_SIZE) {
|
||||
const unsigned long idx = get_eb_page_index(offset);
|
||||
char *kaddr = page_address(eb->pages[idx]);
|
||||
if (oil + key_size <= unit_size) {
|
||||
const unsigned long idx = get_eb_folio_index(eb, offset);
|
||||
char *kaddr = folio_address(eb->folios[idx]);
|
||||
|
||||
oip = get_eb_offset_in_page(eb, offset);
|
||||
tmp = (struct btrfs_disk_key *)(kaddr + oip);
|
||||
oil = get_eb_offset_in_folio(eb, offset);
|
||||
tmp = (struct btrfs_disk_key *)(kaddr + oil);
|
||||
} else {
|
||||
read_extent_buffer(eb, &unaligned, offset, key_size);
|
||||
tmp = &unaligned;
|
||||
|
@ -212,8 +212,6 @@ struct btrfs_root {
|
||||
|
||||
u64 last_trans;
|
||||
|
||||
u32 type;
|
||||
|
||||
u64 free_objectid;
|
||||
|
||||
struct btrfs_key defrag_progress;
|
||||
@ -224,18 +222,15 @@ struct btrfs_root {
|
||||
|
||||
struct list_head root_list;
|
||||
|
||||
spinlock_t log_extents_lock[2];
|
||||
struct list_head logged_list[2];
|
||||
|
||||
spinlock_t inode_lock;
|
||||
/* red-black tree that keeps track of in-memory inodes */
|
||||
struct rb_root inode_tree;
|
||||
|
||||
/*
|
||||
* radix tree that keeps track of delayed nodes of every inode,
|
||||
* protected by inode_lock
|
||||
* Xarray that keeps track of delayed nodes of every inode, protected
|
||||
* by @inode_lock.
|
||||
*/
|
||||
struct radix_tree_root delayed_nodes_tree;
|
||||
struct xarray delayed_nodes;
|
||||
/*
|
||||
* right now this just gets used so that a root has its own devid
|
||||
* for stat. It may be used for more later
|
||||
@ -561,9 +556,9 @@ int btrfs_copy_root(struct btrfs_trans_handle *trans,
|
||||
struct btrfs_root *root,
|
||||
struct extent_buffer *buf,
|
||||
struct extent_buffer **cow_ret, u64 new_root_objectid);
|
||||
int btrfs_block_can_be_shared(struct btrfs_trans_handle *trans,
|
||||
struct btrfs_root *root,
|
||||
struct extent_buffer *buf);
|
||||
bool btrfs_block_can_be_shared(struct btrfs_trans_handle *trans,
|
||||
struct btrfs_root *root,
|
||||
struct extent_buffer *buf);
|
||||
int btrfs_del_ptr(struct btrfs_trans_handle *trans, struct btrfs_root *root,
|
||||
struct btrfs_path *path, int level, int slot);
|
||||
void btrfs_extend_item(struct btrfs_trans_handle *trans,
|
||||
|
@ -775,7 +775,7 @@ static struct extent_map *defrag_lookup_extent(struct inode *inode, u64 start,
|
||||
* this em, as either we don't care about the generation, or the
|
||||
* merged extent map will be rejected anyway.
|
||||
*/
|
||||
if (em && test_bit(EXTENT_FLAG_MERGED, &em->flags) &&
|
||||
if (em && (em->flags & EXTENT_FLAG_MERGED) &&
|
||||
newer_than && em->generation >= newer_than) {
|
||||
free_extent_map(em);
|
||||
em = NULL;
|
||||
@ -802,7 +802,7 @@ static struct extent_map *defrag_lookup_extent(struct inode *inode, u64 start,
|
||||
static u32 get_extent_max_capacity(const struct btrfs_fs_info *fs_info,
|
||||
const struct extent_map *em)
|
||||
{
|
||||
if (test_bit(EXTENT_FLAG_COMPRESSED, &em->flags))
|
||||
if (extent_map_is_compressed(em))
|
||||
return BTRFS_MAX_COMPRESSED;
|
||||
return fs_info->max_extent_size;
|
||||
}
|
||||
@ -828,7 +828,7 @@ static bool defrag_check_next_extent(struct inode *inode, struct extent_map *em,
|
||||
/* No more em or hole */
|
||||
if (!next || next->block_start >= EXTENT_MAP_LAST_BYTE)
|
||||
goto out;
|
||||
if (test_bit(EXTENT_FLAG_PREALLOC, &next->flags))
|
||||
if (next->flags & EXTENT_FLAG_PREALLOC)
|
||||
goto out;
|
||||
/*
|
||||
* If the next extent is at its max capacity, defragging current extent
|
||||
@ -996,10 +996,9 @@ static int defrag_collect_targets(struct btrfs_inode *inode,
|
||||
em->len <= inode->root->fs_info->max_inline)
|
||||
goto next;
|
||||
|
||||
/* Skip hole/delalloc/preallocated extents */
|
||||
/* Skip holes and preallocated extents. */
|
||||
if (em->block_start == EXTENT_MAP_HOLE ||
|
||||
em->block_start == EXTENT_MAP_DELALLOC ||
|
||||
test_bit(EXTENT_FLAG_PREALLOC, &em->flags))
|
||||
(em->flags & EXTENT_FLAG_PREALLOC))
|
||||
goto next;
|
||||
|
||||
/* Skip older extent */
|
||||
@ -1190,7 +1189,7 @@ static int defrag_one_locked_target(struct btrfs_inode *inode,
|
||||
/* Update the page status */
|
||||
for (i = start_index - first_index; i <= last_index - first_index; i++) {
|
||||
ClearPageChecked(pages[i]);
|
||||
btrfs_page_clamp_set_dirty(fs_info, pages[i], start, len);
|
||||
btrfs_folio_clamp_set_dirty(fs_info, page_folio(pages[i]), start, len);
|
||||
}
|
||||
btrfs_delalloc_release_extents(inode, len);
|
||||
extent_changeset_free(data_reserved);
|
||||
|
@ -71,7 +71,7 @@ static struct btrfs_delayed_node *btrfs_get_delayed_node(
|
||||
}
|
||||
|
||||
spin_lock(&root->inode_lock);
|
||||
node = radix_tree_lookup(&root->delayed_nodes_tree, ino);
|
||||
node = xa_load(&root->delayed_nodes, ino);
|
||||
|
||||
if (node) {
|
||||
if (btrfs_inode->delayed_node) {
|
||||
@ -83,9 +83,9 @@ static struct btrfs_delayed_node *btrfs_get_delayed_node(
|
||||
|
||||
/*
|
||||
* It's possible that we're racing into the middle of removing
|
||||
* this node from the radix tree. In this case, the refcount
|
||||
* this node from the xarray. In this case, the refcount
|
||||
* was zero and it should never go back to one. Just return
|
||||
* NULL like it was never in the radix at all; our release
|
||||
* NULL like it was never in the xarray at all; our release
|
||||
* function is in the process of removing it.
|
||||
*
|
||||
* Some implementations of refcount_inc refuse to bump the
|
||||
@ -93,7 +93,7 @@ static struct btrfs_delayed_node *btrfs_get_delayed_node(
|
||||
* here, refcount_inc() may decide to just WARN_ONCE() instead
|
||||
* of actually bumping the refcount.
|
||||
*
|
||||
* If this node is properly in the radix, we want to bump the
|
||||
* If this node is properly in the xarray, we want to bump the
|
||||
* refcount twice, once for the inode and once for this get
|
||||
* operation.
|
||||
*/
|
||||
@ -120,6 +120,7 @@ static struct btrfs_delayed_node *btrfs_get_or_create_delayed_node(
|
||||
struct btrfs_root *root = btrfs_inode->root;
|
||||
u64 ino = btrfs_ino(btrfs_inode);
|
||||
int ret;
|
||||
void *ptr;
|
||||
|
||||
again:
|
||||
node = btrfs_get_delayed_node(btrfs_inode);
|
||||
@ -131,26 +132,30 @@ again:
|
||||
return ERR_PTR(-ENOMEM);
|
||||
btrfs_init_delayed_node(node, root, ino);
|
||||
|
||||
/* cached in the btrfs inode and can be accessed */
|
||||
/* Cached in the inode and can be accessed. */
|
||||
refcount_set(&node->refs, 2);
|
||||
|
||||
ret = radix_tree_preload(GFP_NOFS);
|
||||
if (ret) {
|
||||
/* Allocate and reserve the slot, from now it can return a NULL from xa_load(). */
|
||||
ret = xa_reserve(&root->delayed_nodes, ino, GFP_NOFS);
|
||||
if (ret == -ENOMEM) {
|
||||
kmem_cache_free(delayed_node_cache, node);
|
||||
return ERR_PTR(ret);
|
||||
return ERR_PTR(-ENOMEM);
|
||||
}
|
||||
|
||||
spin_lock(&root->inode_lock);
|
||||
ret = radix_tree_insert(&root->delayed_nodes_tree, ino, node);
|
||||
if (ret == -EEXIST) {
|
||||
ptr = xa_load(&root->delayed_nodes, ino);
|
||||
if (ptr) {
|
||||
/* Somebody inserted it, go back and read it. */
|
||||
spin_unlock(&root->inode_lock);
|
||||
kmem_cache_free(delayed_node_cache, node);
|
||||
radix_tree_preload_end();
|
||||
node = NULL;
|
||||
goto again;
|
||||
}
|
||||
ptr = xa_store(&root->delayed_nodes, ino, node, GFP_ATOMIC);
|
||||
ASSERT(xa_err(ptr) != -EINVAL);
|
||||
ASSERT(xa_err(ptr) != -ENOMEM);
|
||||
ASSERT(ptr == NULL);
|
||||
btrfs_inode->delayed_node = node;
|
||||
spin_unlock(&root->inode_lock);
|
||||
radix_tree_preload_end();
|
||||
|
||||
return node;
|
||||
}
|
||||
@ -269,8 +274,7 @@ static void __btrfs_release_delayed_node(
|
||||
* back up. We can delete it now.
|
||||
*/
|
||||
ASSERT(refcount_read(&delayed_node->refs) == 0);
|
||||
radix_tree_delete(&root->delayed_nodes_tree,
|
||||
delayed_node->inode_id);
|
||||
xa_erase(&root->delayed_nodes, delayed_node->inode_id);
|
||||
spin_unlock(&root->inode_lock);
|
||||
kmem_cache_free(delayed_node_cache, delayed_node);
|
||||
}
|
||||
@ -1036,14 +1040,33 @@ static int __btrfs_update_delayed_inode(struct btrfs_trans_handle *trans,
|
||||
if (!test_bit(BTRFS_DELAYED_NODE_DEL_IREF, &node->flags))
|
||||
goto out;
|
||||
|
||||
path->slots[0]++;
|
||||
if (path->slots[0] >= btrfs_header_nritems(leaf))
|
||||
goto search;
|
||||
again:
|
||||
/*
|
||||
* Now we're going to delete the INODE_REF/EXTREF, which should be the
|
||||
* only one ref left. Check if the next item is an INODE_REF/EXTREF.
|
||||
*
|
||||
* But if we're the last item already, release and search for the last
|
||||
* INODE_REF/EXTREF.
|
||||
*/
|
||||
if (path->slots[0] + 1 >= btrfs_header_nritems(leaf)) {
|
||||
key.objectid = node->inode_id;
|
||||
key.type = BTRFS_INODE_EXTREF_KEY;
|
||||
key.offset = (u64)-1;
|
||||
|
||||
btrfs_release_path(path);
|
||||
ret = btrfs_search_slot(trans, root, &key, path, -1, 1);
|
||||
if (ret < 0)
|
||||
goto err_out;
|
||||
ASSERT(ret > 0);
|
||||
ASSERT(path->slots[0] > 0);
|
||||
ret = 0;
|
||||
path->slots[0]--;
|
||||
leaf = path->nodes[0];
|
||||
} else {
|
||||
path->slots[0]++;
|
||||
}
|
||||
btrfs_item_key_to_cpu(leaf, &key, path->slots[0]);
|
||||
if (key.objectid != node->inode_id)
|
||||
goto out;
|
||||
|
||||
if (key.type != BTRFS_INODE_REF_KEY &&
|
||||
key.type != BTRFS_INODE_EXTREF_KEY)
|
||||
goto out;
|
||||
@ -1070,22 +1093,6 @@ err_out:
|
||||
btrfs_abort_transaction(trans, ret);
|
||||
|
||||
return ret;
|
||||
|
||||
search:
|
||||
btrfs_release_path(path);
|
||||
|
||||
key.type = BTRFS_INODE_EXTREF_KEY;
|
||||
key.offset = -1;
|
||||
|
||||
ret = btrfs_search_slot(trans, root, &key, path, -1, 1);
|
||||
if (ret < 0)
|
||||
goto err_out;
|
||||
ASSERT(ret);
|
||||
|
||||
ret = 0;
|
||||
leaf = path->nodes[0];
|
||||
path->slots[0]--;
|
||||
goto again;
|
||||
}
|
||||
|
||||
static inline int btrfs_update_delayed_inode(struct btrfs_trans_handle *trans,
|
||||
@ -2035,34 +2042,36 @@ void btrfs_kill_delayed_inode_items(struct btrfs_inode *inode)
|
||||
|
||||
void btrfs_kill_all_delayed_nodes(struct btrfs_root *root)
|
||||
{
|
||||
u64 inode_id = 0;
|
||||
unsigned long index = 0;
|
||||
struct btrfs_delayed_node *delayed_nodes[8];
|
||||
int i, n;
|
||||
|
||||
while (1) {
|
||||
struct btrfs_delayed_node *node;
|
||||
int count;
|
||||
|
||||
spin_lock(&root->inode_lock);
|
||||
n = radix_tree_gang_lookup(&root->delayed_nodes_tree,
|
||||
(void **)delayed_nodes, inode_id,
|
||||
ARRAY_SIZE(delayed_nodes));
|
||||
if (!n) {
|
||||
if (xa_empty(&root->delayed_nodes)) {
|
||||
spin_unlock(&root->inode_lock);
|
||||
break;
|
||||
return;
|
||||
}
|
||||
|
||||
inode_id = delayed_nodes[n - 1]->inode_id + 1;
|
||||
for (i = 0; i < n; i++) {
|
||||
count = 0;
|
||||
xa_for_each_start(&root->delayed_nodes, index, node, index) {
|
||||
/*
|
||||
* Don't increase refs in case the node is dead and
|
||||
* about to be removed from the tree in the loop below
|
||||
*/
|
||||
if (!refcount_inc_not_zero(&delayed_nodes[i]->refs))
|
||||
delayed_nodes[i] = NULL;
|
||||
if (refcount_inc_not_zero(&node->refs)) {
|
||||
delayed_nodes[count] = node;
|
||||
count++;
|
||||
}
|
||||
if (count >= ARRAY_SIZE(delayed_nodes))
|
||||
break;
|
||||
}
|
||||
spin_unlock(&root->inode_lock);
|
||||
index++;
|
||||
|
||||
for (i = 0; i < n; i++) {
|
||||
if (!delayed_nodes[i])
|
||||
continue;
|
||||
for (int i = 0; i < count; i++) {
|
||||
__btrfs_kill_delayed_node(delayed_nodes[i]);
|
||||
btrfs_release_delayed_node(delayed_nodes[i]);
|
||||
}
|
||||
|
@ -550,8 +550,7 @@ bool btrfs_finish_block_group_to_copy(struct btrfs_device *srcdev,
|
||||
u64 physical)
|
||||
{
|
||||
struct btrfs_fs_info *fs_info = cache->fs_info;
|
||||
struct extent_map *em;
|
||||
struct map_lookup *map;
|
||||
struct btrfs_chunk_map *map;
|
||||
u64 chunk_offset = cache->start;
|
||||
int num_extents, cur_extent;
|
||||
int i;
|
||||
@ -567,9 +566,8 @@ bool btrfs_finish_block_group_to_copy(struct btrfs_device *srcdev,
|
||||
}
|
||||
spin_unlock(&cache->lock);
|
||||
|
||||
em = btrfs_get_chunk_map(fs_info, chunk_offset, 1);
|
||||
ASSERT(!IS_ERR(em));
|
||||
map = em->map_lookup;
|
||||
map = btrfs_get_chunk_map(fs_info, chunk_offset, 1);
|
||||
ASSERT(!IS_ERR(map));
|
||||
|
||||
num_extents = 0;
|
||||
cur_extent = 0;
|
||||
@ -583,7 +581,7 @@ bool btrfs_finish_block_group_to_copy(struct btrfs_device *srcdev,
|
||||
cur_extent = i;
|
||||
}
|
||||
|
||||
free_extent_map(em);
|
||||
btrfs_free_chunk_map(map);
|
||||
|
||||
if (num_extents > 1 && cur_extent < num_extents - 1) {
|
||||
/*
|
||||
@ -812,25 +810,23 @@ static void btrfs_dev_replace_update_device_in_mapping_tree(
|
||||
struct btrfs_device *srcdev,
|
||||
struct btrfs_device *tgtdev)
|
||||
{
|
||||
struct extent_map_tree *em_tree = &fs_info->mapping_tree;
|
||||
struct extent_map *em;
|
||||
struct map_lookup *map;
|
||||
u64 start = 0;
|
||||
int i;
|
||||
|
||||
write_lock(&em_tree->lock);
|
||||
write_lock(&fs_info->mapping_tree_lock);
|
||||
do {
|
||||
em = lookup_extent_mapping(em_tree, start, (u64)-1);
|
||||
if (!em)
|
||||
struct btrfs_chunk_map *map;
|
||||
|
||||
map = btrfs_find_chunk_map_nolock(fs_info, start, U64_MAX);
|
||||
if (!map)
|
||||
break;
|
||||
map = em->map_lookup;
|
||||
for (i = 0; i < map->num_stripes; i++)
|
||||
if (srcdev == map->stripes[i].dev)
|
||||
map->stripes[i].dev = tgtdev;
|
||||
start = em->start + em->len;
|
||||
free_extent_map(em);
|
||||
start = map->start + map->chunk_len;
|
||||
btrfs_free_chunk_map(map);
|
||||
} while (start);
|
||||
write_unlock(&em_tree->lock);
|
||||
write_unlock(&fs_info->mapping_tree_lock);
|
||||
}
|
||||
|
||||
static int btrfs_dev_replace_finishing(struct btrfs_fs_info *fs_info,
|
||||
|
@ -74,20 +74,37 @@ static void btrfs_free_csum_hash(struct btrfs_fs_info *fs_info)
|
||||
static void csum_tree_block(struct extent_buffer *buf, u8 *result)
|
||||
{
|
||||
struct btrfs_fs_info *fs_info = buf->fs_info;
|
||||
const int num_pages = num_extent_pages(buf);
|
||||
const int first_page_part = min_t(u32, PAGE_SIZE, fs_info->nodesize);
|
||||
int num_pages;
|
||||
u32 first_page_part;
|
||||
SHASH_DESC_ON_STACK(shash, fs_info->csum_shash);
|
||||
char *kaddr;
|
||||
int i;
|
||||
|
||||
shash->tfm = fs_info->csum_shash;
|
||||
crypto_shash_init(shash);
|
||||
kaddr = page_address(buf->pages[0]) + offset_in_page(buf->start);
|
||||
|
||||
if (buf->addr) {
|
||||
/* Pages are contiguous, handle them as a big one. */
|
||||
kaddr = buf->addr;
|
||||
first_page_part = fs_info->nodesize;
|
||||
num_pages = 1;
|
||||
} else {
|
||||
kaddr = folio_address(buf->folios[0]);
|
||||
first_page_part = min_t(u32, PAGE_SIZE, fs_info->nodesize);
|
||||
num_pages = num_extent_pages(buf);
|
||||
}
|
||||
|
||||
crypto_shash_update(shash, kaddr + BTRFS_CSUM_SIZE,
|
||||
first_page_part - BTRFS_CSUM_SIZE);
|
||||
|
||||
/*
|
||||
* Multiple single-page folios case would reach here.
|
||||
*
|
||||
* nodesize <= PAGE_SIZE and large folio all handled by above
|
||||
* crypto_shash_update() already.
|
||||
*/
|
||||
for (i = 1; i < num_pages && INLINE_EXTENT_BUFFER_PAGES > 1; i++) {
|
||||
kaddr = page_address(buf->pages[i]);
|
||||
kaddr = folio_address(buf->folios[i]);
|
||||
crypto_shash_update(shash, kaddr, PAGE_SIZE);
|
||||
}
|
||||
memset(result, 0, BTRFS_CSUM_SIZE);
|
||||
@ -166,20 +183,22 @@ static int btrfs_repair_eb_io_failure(const struct extent_buffer *eb,
|
||||
int mirror_num)
|
||||
{
|
||||
struct btrfs_fs_info *fs_info = eb->fs_info;
|
||||
int i, num_pages = num_extent_pages(eb);
|
||||
int num_folios = num_extent_folios(eb);
|
||||
int ret = 0;
|
||||
|
||||
if (sb_rdonly(fs_info->sb))
|
||||
return -EROFS;
|
||||
|
||||
for (i = 0; i < num_pages; i++) {
|
||||
struct page *p = eb->pages[i];
|
||||
u64 start = max_t(u64, eb->start, page_offset(p));
|
||||
u64 end = min_t(u64, eb->start + eb->len, page_offset(p) + PAGE_SIZE);
|
||||
for (int i = 0; i < num_folios; i++) {
|
||||
struct folio *folio = eb->folios[i];
|
||||
u64 start = max_t(u64, eb->start, folio_pos(folio));
|
||||
u64 end = min_t(u64, eb->start + eb->len,
|
||||
folio_pos(folio) + folio_size(folio));
|
||||
u32 len = end - start;
|
||||
|
||||
ret = btrfs_repair_io_failure(fs_info, 0, start, len,
|
||||
start, p, offset_in_page(start), mirror_num);
|
||||
start, folio, offset_in_folio(folio, start),
|
||||
mirror_num);
|
||||
if (ret)
|
||||
break;
|
||||
}
|
||||
@ -254,15 +273,20 @@ blk_status_t btree_csum_one_bio(struct btrfs_bio *bbio)
|
||||
if (WARN_ON_ONCE(bbio->bio.bi_iter.bi_size != eb->len))
|
||||
return BLK_STS_IOERR;
|
||||
|
||||
if (test_bit(EXTENT_BUFFER_NO_CHECK, &eb->bflags)) {
|
||||
WARN_ON_ONCE(found_start != 0);
|
||||
/*
|
||||
* If an extent_buffer is marked as EXTENT_BUFFER_ZONED_ZEROOUT, don't
|
||||
* checksum it but zero-out its content. This is done to preserve
|
||||
* ordering of I/O without unnecessarily writing out data.
|
||||
*/
|
||||
if (test_bit(EXTENT_BUFFER_ZONED_ZEROOUT, &eb->bflags)) {
|
||||
memzero_extent_buffer(eb, 0, eb->len);
|
||||
return BLK_STS_OK;
|
||||
}
|
||||
|
||||
if (WARN_ON_ONCE(found_start != eb->start))
|
||||
return BLK_STS_IOERR;
|
||||
if (WARN_ON(!btrfs_page_test_uptodate(fs_info, eb->pages[0], eb->start,
|
||||
eb->len)))
|
||||
if (WARN_ON(!btrfs_folio_test_uptodate(fs_info, eb->folios[0],
|
||||
eb->start, eb->len)))
|
||||
return BLK_STS_IOERR;
|
||||
|
||||
ASSERT(memcmp_extent_buffer(eb, fs_info->fs_devices->metadata_uuid,
|
||||
@ -371,8 +395,8 @@ int btrfs_validate_extent_buffer(struct extent_buffer *eb,
|
||||
}
|
||||
|
||||
csum_tree_block(eb, result);
|
||||
header_csum = page_address(eb->pages[0]) +
|
||||
get_eb_offset_in_page(eb, offsetof(struct btrfs_header, csum));
|
||||
header_csum = folio_address(eb->folios[0]) +
|
||||
get_eb_offset_in_folio(eb, offsetof(struct btrfs_header, csum));
|
||||
|
||||
if (memcmp(result, header_csum, csum_size) != 0) {
|
||||
btrfs_warn_rl(fs_info,
|
||||
@ -639,7 +663,8 @@ static void __setup_root(struct btrfs_root *root, struct btrfs_fs_info *fs_info,
|
||||
root->nr_delalloc_inodes = 0;
|
||||
root->nr_ordered_extents = 0;
|
||||
root->inode_tree = RB_ROOT;
|
||||
INIT_RADIX_TREE(&root->delayed_nodes_tree, GFP_ATOMIC);
|
||||
/* GFP flags are compatible with XA_FLAGS_*. */
|
||||
xa_init_flags(&root->delayed_nodes, GFP_ATOMIC);
|
||||
|
||||
btrfs_init_root_block_rsv(root);
|
||||
|
||||
@ -650,14 +675,10 @@ static void __setup_root(struct btrfs_root *root, struct btrfs_fs_info *fs_info,
|
||||
INIT_LIST_HEAD(&root->ordered_extents);
|
||||
INIT_LIST_HEAD(&root->ordered_root);
|
||||
INIT_LIST_HEAD(&root->reloc_dirty_list);
|
||||
INIT_LIST_HEAD(&root->logged_list[0]);
|
||||
INIT_LIST_HEAD(&root->logged_list[1]);
|
||||
spin_lock_init(&root->inode_lock);
|
||||
spin_lock_init(&root->delalloc_lock);
|
||||
spin_lock_init(&root->ordered_extent_lock);
|
||||
spin_lock_init(&root->accounting_lock);
|
||||
spin_lock_init(&root->log_extents_lock[0]);
|
||||
spin_lock_init(&root->log_extents_lock[1]);
|
||||
spin_lock_init(&root->qgroup_meta_rsv_lock);
|
||||
mutex_init(&root->objectid_mutex);
|
||||
mutex_init(&root->log_mutex);
|
||||
@ -2618,9 +2639,6 @@ static int __cold init_tree_roots(struct btrfs_fs_info *fs_info)
|
||||
*/
|
||||
btrfs_set_super_log_root(sb, 0);
|
||||
|
||||
/* We can't trust the free space cache either */
|
||||
btrfs_set_opt(fs_info->mount_opt, CLEAR_CACHE);
|
||||
|
||||
btrfs_warn(fs_info, "try to load backup roots slot %d", i);
|
||||
ret = read_backup_root(fs_info, i);
|
||||
backup_index = ret;
|
||||
@ -2724,7 +2742,8 @@ void btrfs_init_fs_info(struct btrfs_fs_info *fs_info)
|
||||
INIT_LIST_HEAD(&fs_info->allocated_ebs);
|
||||
spin_lock_init(&fs_info->eb_leak_lock);
|
||||
#endif
|
||||
extent_map_tree_init(&fs_info->mapping_tree);
|
||||
fs_info->mapping_tree = RB_ROOT_CACHED;
|
||||
rwlock_init(&fs_info->mapping_tree_lock);
|
||||
btrfs_init_block_rsv(&fs_info->global_block_rsv,
|
||||
BTRFS_BLOCK_RSV_GLOBAL);
|
||||
btrfs_init_block_rsv(&fs_info->trans_block_rsv, BTRFS_BLOCK_RSV_TRANS);
|
||||
@ -2794,6 +2813,9 @@ void btrfs_init_fs_info(struct btrfs_fs_info *fs_info)
|
||||
fs_info->sectorsize_bits = ilog2(4096);
|
||||
fs_info->stripesize = 4096;
|
||||
|
||||
/* Default compress algorithm when user does -o compress */
|
||||
fs_info->compress_type = BTRFS_COMPRESS_ZLIB;
|
||||
|
||||
fs_info->max_extent_size = BTRFS_MAX_EXTENT_SIZE;
|
||||
|
||||
spin_lock_init(&fs_info->swapfile_pins_lock);
|
||||
@ -2930,17 +2952,6 @@ out:
|
||||
return err;
|
||||
}
|
||||
|
||||
/*
|
||||
* Some options only have meaning at mount time and shouldn't persist across
|
||||
* remounts, or be displayed. Clear these at the end of mount and remount
|
||||
* code paths.
|
||||
*/
|
||||
void btrfs_clear_oneshot_options(struct btrfs_fs_info *fs_info)
|
||||
{
|
||||
btrfs_clear_opt(fs_info->mount_opt, USEBACKUPROOT);
|
||||
btrfs_clear_opt(fs_info->mount_opt, CLEAR_CACHE);
|
||||
}
|
||||
|
||||
/*
|
||||
* Mounting logic specific to read-write file systems. Shared by open_ctree
|
||||
* and btrfs_remount when remounting from read-only to read-write.
|
||||
@ -2953,7 +2964,11 @@ int btrfs_start_pre_rw_mount(struct btrfs_fs_info *fs_info)
|
||||
|
||||
if (btrfs_test_opt(fs_info, CLEAR_CACHE) &&
|
||||
btrfs_fs_compat_ro(fs_info, FREE_SPACE_TREE)) {
|
||||
rebuild_free_space_tree = true;
|
||||
if (btrfs_fs_incompat(fs_info, EXTENT_TREE_V2))
|
||||
btrfs_warn(fs_info,
|
||||
"'clear_cache' option is ignored with extent tree v2");
|
||||
else
|
||||
rebuild_free_space_tree = true;
|
||||
} else if (btrfs_fs_compat_ro(fs_info, FREE_SPACE_TREE) &&
|
||||
!btrfs_fs_compat_ro(fs_info, FREE_SPACE_TREE_VALID)) {
|
||||
btrfs_warn(fs_info, "free space tree is invalid");
|
||||
@ -3276,13 +3291,6 @@ int __cold open_ctree(struct super_block *sb, struct btrfs_fs_devices *fs_device
|
||||
if (btrfs_super_flags(disk_super) & BTRFS_SUPER_FLAG_ERROR)
|
||||
WRITE_ONCE(fs_info->fs_error, -EUCLEAN);
|
||||
|
||||
/*
|
||||
* In the long term, we'll store the compression type in the super
|
||||
* block, and it'll be used for per file compression control.
|
||||
*/
|
||||
fs_info->compress_type = BTRFS_COMPRESS_ZLIB;
|
||||
|
||||
|
||||
/* Set up fs_info before parsing mount options */
|
||||
nodesize = btrfs_super_nodesize(disk_super);
|
||||
sectorsize = btrfs_super_sectorsize(disk_super);
|
||||
@ -3296,28 +3304,30 @@ int __cold open_ctree(struct super_block *sb, struct btrfs_fs_devices *fs_device
|
||||
fs_info->csums_per_leaf = BTRFS_MAX_ITEM_SIZE(fs_info) / fs_info->csum_size;
|
||||
fs_info->stripesize = stripesize;
|
||||
|
||||
ret = btrfs_parse_options(fs_info, options, sb->s_flags);
|
||||
if (ret)
|
||||
/*
|
||||
* Handle the space caching options appropriately now that we have the
|
||||
* super block loaded and validated.
|
||||
*/
|
||||
btrfs_set_free_space_cache_settings(fs_info);
|
||||
|
||||
if (!btrfs_check_options(fs_info, &fs_info->mount_opt, sb->s_flags)) {
|
||||
ret = -EINVAL;
|
||||
goto fail_alloc;
|
||||
}
|
||||
|
||||
ret = btrfs_check_features(fs_info, !sb_rdonly(sb));
|
||||
if (ret < 0)
|
||||
goto fail_alloc;
|
||||
|
||||
/*
|
||||
* At this point our mount options are validated, if we set ->max_inline
|
||||
* to something non-standard make sure we truncate it to sectorsize.
|
||||
*/
|
||||
fs_info->max_inline = min_t(u64, fs_info->max_inline, fs_info->sectorsize);
|
||||
|
||||
if (sectorsize < PAGE_SIZE) {
|
||||
struct btrfs_subpage_info *subpage_info;
|
||||
|
||||
/*
|
||||
* V1 space cache has some hardcoded PAGE_SIZE usage, and is
|
||||
* going to be deprecated.
|
||||
*
|
||||
* Force to use v2 cache for subpage case.
|
||||
*/
|
||||
btrfs_clear_opt(fs_info->mount_opt, SPACE_CACHE);
|
||||
btrfs_set_and_info(fs_info, FREE_SPACE_TREE,
|
||||
"forcing free space tree for sector size %u with page size %lu",
|
||||
sectorsize, PAGE_SIZE);
|
||||
|
||||
btrfs_warn(fs_info,
|
||||
"read-write for sector size %u with page size %lu is experimental",
|
||||
sectorsize, PAGE_SIZE);
|
||||
@ -3494,29 +3504,6 @@ int __cold open_ctree(struct super_block *sb, struct btrfs_fs_devices *fs_device
|
||||
goto fail_cleaner;
|
||||
}
|
||||
|
||||
if (!btrfs_test_opt(fs_info, NOSSD) &&
|
||||
!fs_info->fs_devices->rotating) {
|
||||
btrfs_set_and_info(fs_info, SSD, "enabling ssd optimizations");
|
||||
}
|
||||
|
||||
/*
|
||||
* For devices supporting discard turn on discard=async automatically,
|
||||
* unless it's already set or disabled. This could be turned off by
|
||||
* nodiscard for the same mount.
|
||||
*
|
||||
* The zoned mode piggy backs on the discard functionality for
|
||||
* resetting a zone. There is no reason to delay the zone reset as it is
|
||||
* fast enough. So, do not enable async discard for zoned mode.
|
||||
*/
|
||||
if (!(btrfs_test_opt(fs_info, DISCARD_SYNC) ||
|
||||
btrfs_test_opt(fs_info, DISCARD_ASYNC) ||
|
||||
btrfs_test_opt(fs_info, NODISCARD)) &&
|
||||
fs_info->fs_devices->discardable &&
|
||||
!btrfs_is_zoned(fs_info)) {
|
||||
btrfs_set_and_info(fs_info, DISCARD_ASYNC,
|
||||
"auto enabling async discard");
|
||||
}
|
||||
|
||||
ret = btrfs_read_qgroup_config(fs_info);
|
||||
if (ret)
|
||||
goto fail_trans_kthread;
|
||||
@ -3542,7 +3529,7 @@ int __cold open_ctree(struct super_block *sb, struct btrfs_fs_devices *fs_device
|
||||
}
|
||||
|
||||
if (sb_rdonly(sb))
|
||||
goto clear_oneshot;
|
||||
return 0;
|
||||
|
||||
ret = btrfs_start_pre_rw_mount(fs_info);
|
||||
if (ret) {
|
||||
@ -3570,8 +3557,6 @@ int __cold open_ctree(struct super_block *sb, struct btrfs_fs_devices *fs_device
|
||||
if (test_bit(BTRFS_FS_UNFINISHED_DROPS, &fs_info->flags))
|
||||
wake_up_process(fs_info->cleaner_kthread);
|
||||
|
||||
clear_oneshot:
|
||||
btrfs_clear_oneshot_options(fs_info);
|
||||
return 0;
|
||||
|
||||
fail_qgroup:
|
||||
@ -3608,7 +3593,7 @@ fail_sb_buffer:
|
||||
btrfs_stop_all_workers(fs_info);
|
||||
btrfs_free_block_groups(fs_info);
|
||||
fail_alloc:
|
||||
btrfs_mapping_tree_free(&fs_info->mapping_tree);
|
||||
btrfs_mapping_tree_free(fs_info);
|
||||
|
||||
iput(fs_info->btree_inode);
|
||||
fail:
|
||||
@ -4391,7 +4376,7 @@ void __cold close_ctree(struct btrfs_fs_info *fs_info)
|
||||
|
||||
iput(fs_info->btree_inode);
|
||||
|
||||
btrfs_mapping_tree_free(&fs_info->mapping_tree);
|
||||
btrfs_mapping_tree_free(fs_info);
|
||||
btrfs_close_devices(fs_info->fs_devices);
|
||||
}
|
||||
|
||||
|
@ -37,9 +37,6 @@ struct extent_buffer *btrfs_find_create_tree_block(
|
||||
struct btrfs_fs_info *fs_info,
|
||||
u64 bytenr, u64 owner_root,
|
||||
int level);
|
||||
void btrfs_clear_buffer_dirty(struct btrfs_trans_handle *trans,
|
||||
struct extent_buffer *buf);
|
||||
void btrfs_clear_oneshot_options(struct btrfs_fs_info *fs_info);
|
||||
int btrfs_start_pre_rw_mount(struct btrfs_fs_info *fs_info);
|
||||
int btrfs_check_super_csum(struct btrfs_fs_info *fs_info,
|
||||
const struct btrfs_super_block *disk_sb);
|
||||
|
@ -58,12 +58,13 @@ static inline void __btrfs_debug_check_extent_io_range(const char *caller,
|
||||
struct extent_io_tree *tree,
|
||||
u64 start, u64 end)
|
||||
{
|
||||
struct btrfs_inode *inode = tree->inode;
|
||||
const struct btrfs_inode *inode;
|
||||
u64 isize;
|
||||
|
||||
if (!inode)
|
||||
if (tree->owner != IO_TREE_INODE_IO)
|
||||
return;
|
||||
|
||||
inode = extent_io_tree_to_inode_const(tree);
|
||||
isize = i_size_read(&inode->vfs_inode);
|
||||
if (end >= PAGE_SIZE && (end % 2) == 0 && end != isize - 1) {
|
||||
btrfs_debug_rl(inode->root->fs_info,
|
||||
@ -78,31 +79,46 @@ static inline void __btrfs_debug_check_extent_io_range(const char *caller,
|
||||
#define btrfs_debug_check_extent_io_range(c, s, e) do {} while (0)
|
||||
#endif
|
||||
|
||||
/*
|
||||
* For the file_extent_tree, we want to hold the inode lock when we lookup and
|
||||
* update the disk_i_size, but lockdep will complain because our io_tree we hold
|
||||
* the tree lock and get the inode lock when setting delalloc. These two things
|
||||
* are unrelated, so make a class for the file_extent_tree so we don't get the
|
||||
* two locking patterns mixed up.
|
||||
*/
|
||||
static struct lock_class_key file_extent_tree_class;
|
||||
|
||||
struct tree_entry {
|
||||
u64 start;
|
||||
u64 end;
|
||||
struct rb_node rb_node;
|
||||
};
|
||||
/*
|
||||
* The only tree allowed to set the inode is IO_TREE_INODE_IO.
|
||||
*/
|
||||
static bool is_inode_io_tree(const struct extent_io_tree *tree)
|
||||
{
|
||||
return tree->owner == IO_TREE_INODE_IO;
|
||||
}
|
||||
|
||||
/* Return the inode if it's valid for the given tree, otherwise NULL. */
|
||||
struct btrfs_inode *extent_io_tree_to_inode(struct extent_io_tree *tree)
|
||||
{
|
||||
if (tree->owner == IO_TREE_INODE_IO)
|
||||
return tree->inode;
|
||||
return NULL;
|
||||
}
|
||||
|
||||
/* Read-only access to the inode. */
|
||||
const struct btrfs_inode *extent_io_tree_to_inode_const(const struct extent_io_tree *tree)
|
||||
{
|
||||
if (tree->owner == IO_TREE_INODE_IO)
|
||||
return tree->inode;
|
||||
return NULL;
|
||||
}
|
||||
|
||||
/* For read-only access to fs_info. */
|
||||
const struct btrfs_fs_info *extent_io_tree_to_fs_info(const struct extent_io_tree *tree)
|
||||
{
|
||||
if (tree->owner == IO_TREE_INODE_IO)
|
||||
return tree->inode->root->fs_info;
|
||||
return tree->fs_info;
|
||||
}
|
||||
|
||||
void extent_io_tree_init(struct btrfs_fs_info *fs_info,
|
||||
struct extent_io_tree *tree, unsigned int owner)
|
||||
{
|
||||
tree->fs_info = fs_info;
|
||||
tree->state = RB_ROOT;
|
||||
spin_lock_init(&tree->lock);
|
||||
tree->inode = NULL;
|
||||
tree->fs_info = fs_info;
|
||||
tree->owner = owner;
|
||||
if (owner == IO_TREE_INODE_FILE_EXTENT)
|
||||
lockdep_set_class(&tree->lock, &file_extent_tree_class);
|
||||
}
|
||||
|
||||
/*
|
||||
@ -329,10 +345,14 @@ static inline struct extent_state *tree_search(struct extent_io_tree *tree, u64
|
||||
return tree_search_for_insert(tree, offset, NULL, NULL);
|
||||
}
|
||||
|
||||
static void extent_io_tree_panic(struct extent_io_tree *tree, int err)
|
||||
static void extent_io_tree_panic(const struct extent_io_tree *tree,
|
||||
const struct extent_state *state,
|
||||
const char *opname,
|
||||
int err)
|
||||
{
|
||||
btrfs_panic(tree->fs_info, err,
|
||||
"locking error: extent tree was modified by another thread while locked");
|
||||
btrfs_panic(extent_io_tree_to_fs_info(tree), err,
|
||||
"extent io tree error on %s state start %llu end %llu",
|
||||
opname, state->start, state->end);
|
||||
}
|
||||
|
||||
static void merge_prev_state(struct extent_io_tree *tree, struct extent_state *state)
|
||||
@ -341,8 +361,9 @@ static void merge_prev_state(struct extent_io_tree *tree, struct extent_state *s
|
||||
|
||||
prev = prev_state(state);
|
||||
if (prev && prev->end == state->start - 1 && prev->state == state->state) {
|
||||
if (tree->inode)
|
||||
btrfs_merge_delalloc_extent(tree->inode, state, prev);
|
||||
if (is_inode_io_tree(tree))
|
||||
btrfs_merge_delalloc_extent(extent_io_tree_to_inode(tree),
|
||||
state, prev);
|
||||
state->start = prev->start;
|
||||
rb_erase(&prev->rb_node, &tree->state);
|
||||
RB_CLEAR_NODE(&prev->rb_node);
|
||||
@ -356,8 +377,9 @@ static void merge_next_state(struct extent_io_tree *tree, struct extent_state *s
|
||||
|
||||
next = next_state(state);
|
||||
if (next && next->start == state->end + 1 && next->state == state->state) {
|
||||
if (tree->inode)
|
||||
btrfs_merge_delalloc_extent(tree->inode, state, next);
|
||||
if (is_inode_io_tree(tree))
|
||||
btrfs_merge_delalloc_extent(extent_io_tree_to_inode(tree),
|
||||
state, next);
|
||||
state->end = next->end;
|
||||
rb_erase(&next->rb_node, &tree->state);
|
||||
RB_CLEAR_NODE(&next->rb_node);
|
||||
@ -390,8 +412,8 @@ static void set_state_bits(struct extent_io_tree *tree,
|
||||
u32 bits_to_set = bits & ~EXTENT_CTLBITS;
|
||||
int ret;
|
||||
|
||||
if (tree->inode)
|
||||
btrfs_set_delalloc_extent(tree->inode, state, bits);
|
||||
if (is_inode_io_tree(tree))
|
||||
btrfs_set_delalloc_extent(extent_io_tree_to_inode(tree), state, bits);
|
||||
|
||||
ret = add_extent_changeset(state, bits_to_set, changeset, 1);
|
||||
BUG_ON(ret < 0);
|
||||
@ -436,9 +458,10 @@ static struct extent_state *insert_state(struct extent_io_tree *tree,
|
||||
if (state->end < entry->start) {
|
||||
if (try_merge && end == entry->start &&
|
||||
state->state == entry->state) {
|
||||
if (tree->inode)
|
||||
btrfs_merge_delalloc_extent(tree->inode,
|
||||
state, entry);
|
||||
if (is_inode_io_tree(tree))
|
||||
btrfs_merge_delalloc_extent(
|
||||
extent_io_tree_to_inode(tree),
|
||||
state, entry);
|
||||
entry->start = state->start;
|
||||
merge_prev_state(tree, entry);
|
||||
state->state = 0;
|
||||
@ -448,9 +471,10 @@ static struct extent_state *insert_state(struct extent_io_tree *tree,
|
||||
} else if (state->end > entry->end) {
|
||||
if (try_merge && entry->end == start &&
|
||||
state->state == entry->state) {
|
||||
if (tree->inode)
|
||||
btrfs_merge_delalloc_extent(tree->inode,
|
||||
state, entry);
|
||||
if (is_inode_io_tree(tree))
|
||||
btrfs_merge_delalloc_extent(
|
||||
extent_io_tree_to_inode(tree),
|
||||
state, entry);
|
||||
entry->end = state->end;
|
||||
merge_next_state(tree, entry);
|
||||
state->state = 0;
|
||||
@ -458,9 +482,6 @@ static struct extent_state *insert_state(struct extent_io_tree *tree,
|
||||
}
|
||||
node = &(*node)->rb_right;
|
||||
} else {
|
||||
btrfs_err(tree->fs_info,
|
||||
"found node %llu %llu on insert of %llu %llu",
|
||||
entry->start, entry->end, state->start, state->end);
|
||||
return ERR_PTR(-EEXIST);
|
||||
}
|
||||
}
|
||||
@ -505,8 +526,9 @@ static int split_state(struct extent_io_tree *tree, struct extent_state *orig,
|
||||
struct rb_node *parent = NULL;
|
||||
struct rb_node **node;
|
||||
|
||||
if (tree->inode)
|
||||
btrfs_split_delalloc_extent(tree->inode, orig, split);
|
||||
if (is_inode_io_tree(tree))
|
||||
btrfs_split_delalloc_extent(extent_io_tree_to_inode(tree), orig,
|
||||
split);
|
||||
|
||||
prealloc->start = orig->start;
|
||||
prealloc->end = split - 1;
|
||||
@ -553,8 +575,9 @@ static struct extent_state *clear_state_bit(struct extent_io_tree *tree,
|
||||
u32 bits_to_clear = bits & ~EXTENT_CTLBITS;
|
||||
int ret;
|
||||
|
||||
if (tree->inode)
|
||||
btrfs_clear_delalloc_extent(tree->inode, state, bits);
|
||||
if (is_inode_io_tree(tree))
|
||||
btrfs_clear_delalloc_extent(extent_io_tree_to_inode(tree), state,
|
||||
bits);
|
||||
|
||||
ret = add_extent_changeset(state, bits_to_clear, changeset, 0);
|
||||
BUG_ON(ret < 0);
|
||||
@ -695,7 +718,7 @@ hit_next:
|
||||
goto search_again;
|
||||
err = split_state(tree, state, prealloc, start);
|
||||
if (err)
|
||||
extent_io_tree_panic(tree, err);
|
||||
extent_io_tree_panic(tree, state, "split", err);
|
||||
|
||||
prealloc = NULL;
|
||||
if (err)
|
||||
@ -717,7 +740,7 @@ hit_next:
|
||||
goto search_again;
|
||||
err = split_state(tree, state, prealloc, end + 1);
|
||||
if (err)
|
||||
extent_io_tree_panic(tree, err);
|
||||
extent_io_tree_panic(tree, state, "split", err);
|
||||
|
||||
if (wake)
|
||||
wake_up(&state->wq);
|
||||
@ -939,6 +962,8 @@ int find_contiguous_extent_bit(struct extent_io_tree *tree, u64 start,
|
||||
struct extent_state *state;
|
||||
int ret = 1;
|
||||
|
||||
ASSERT(!btrfs_fs_incompat(extent_io_tree_to_fs_info(tree), NO_HOLES));
|
||||
|
||||
spin_lock(&tree->lock);
|
||||
state = find_first_extent_bit_state(tree, start, bits);
|
||||
if (state) {
|
||||
@ -1152,7 +1177,7 @@ hit_next:
|
||||
goto search_again;
|
||||
err = split_state(tree, state, prealloc, start);
|
||||
if (err)
|
||||
extent_io_tree_panic(tree, err);
|
||||
extent_io_tree_panic(tree, state, "split", err);
|
||||
|
||||
prealloc = NULL;
|
||||
if (err)
|
||||
@ -1200,7 +1225,7 @@ hit_next:
|
||||
inserted_state = insert_state(tree, prealloc, bits, changeset);
|
||||
if (IS_ERR(inserted_state)) {
|
||||
err = PTR_ERR(inserted_state);
|
||||
extent_io_tree_panic(tree, err);
|
||||
extent_io_tree_panic(tree, prealloc, "insert", err);
|
||||
}
|
||||
|
||||
cache_state(inserted_state, cached_state);
|
||||
@ -1228,7 +1253,7 @@ hit_next:
|
||||
goto search_again;
|
||||
err = split_state(tree, state, prealloc, end + 1);
|
||||
if (err)
|
||||
extent_io_tree_panic(tree, err);
|
||||
extent_io_tree_panic(tree, state, "split", err);
|
||||
|
||||
set_state_bits(tree, prealloc, bits, changeset);
|
||||
cache_state(prealloc, cached_state);
|
||||
@ -1382,7 +1407,7 @@ hit_next:
|
||||
}
|
||||
err = split_state(tree, state, prealloc, start);
|
||||
if (err)
|
||||
extent_io_tree_panic(tree, err);
|
||||
extent_io_tree_panic(tree, state, "split", err);
|
||||
prealloc = NULL;
|
||||
if (err)
|
||||
goto out;
|
||||
@ -1430,7 +1455,7 @@ hit_next:
|
||||
inserted_state = insert_state(tree, prealloc, bits, NULL);
|
||||
if (IS_ERR(inserted_state)) {
|
||||
err = PTR_ERR(inserted_state);
|
||||
extent_io_tree_panic(tree, err);
|
||||
extent_io_tree_panic(tree, prealloc, "insert", err);
|
||||
}
|
||||
cache_state(inserted_state, cached_state);
|
||||
if (inserted_state == prealloc)
|
||||
@ -1453,7 +1478,7 @@ hit_next:
|
||||
|
||||
err = split_state(tree, state, prealloc, end + 1);
|
||||
if (err)
|
||||
extent_io_tree_panic(tree, err);
|
||||
extent_io_tree_panic(tree, state, "split", err);
|
||||
|
||||
set_state_bits(tree, prealloc, bits, NULL);
|
||||
cache_state(prealloc, cached_state);
|
||||
|
@ -87,9 +87,17 @@ enum {
|
||||
|
||||
struct extent_io_tree {
|
||||
struct rb_root state;
|
||||
struct btrfs_fs_info *fs_info;
|
||||
/* Inode associated with this tree, or NULL. */
|
||||
struct btrfs_inode *inode;
|
||||
/*
|
||||
* The fs_info is needed for trace points, a tree attached to an inode
|
||||
* needs the inode.
|
||||
*
|
||||
* owner == IO_TREE_INODE_IO - then inode is valid and fs_info can be
|
||||
* accessed as inode->root->fs_info
|
||||
*/
|
||||
union {
|
||||
struct btrfs_fs_info *fs_info;
|
||||
struct btrfs_inode *inode;
|
||||
};
|
||||
|
||||
/* Who owns this io tree, should be one of IO_TREE_* */
|
||||
u8 owner;
|
||||
@ -112,6 +120,10 @@ struct extent_state {
|
||||
#endif
|
||||
};
|
||||
|
||||
struct btrfs_inode *extent_io_tree_to_inode(struct extent_io_tree *tree);
|
||||
const struct btrfs_inode *extent_io_tree_to_inode_const(const struct extent_io_tree *tree);
|
||||
const struct btrfs_fs_info *extent_io_tree_to_fs_info(const struct extent_io_tree *tree);
|
||||
|
||||
void extent_io_tree_init(struct btrfs_fs_info *fs_info,
|
||||
struct extent_io_tree *tree, unsigned int owner);
|
||||
void extent_io_tree_release(struct extent_io_tree *tree);
|
||||
|
@ -3447,6 +3447,7 @@ void btrfs_free_tree_block(struct btrfs_trans_handle *trans,
|
||||
{
|
||||
struct btrfs_fs_info *fs_info = trans->fs_info;
|
||||
struct btrfs_ref generic_ref = { 0 };
|
||||
struct btrfs_block_group *bg;
|
||||
int ret;
|
||||
|
||||
btrfs_init_generic_ref(&generic_ref, BTRFS_DROP_DELAYED_REF,
|
||||
@ -3460,67 +3461,64 @@ void btrfs_free_tree_block(struct btrfs_trans_handle *trans,
|
||||
BUG_ON(ret); /* -ENOMEM */
|
||||
}
|
||||
|
||||
if (last_ref && btrfs_header_generation(buf) == trans->transid) {
|
||||
struct btrfs_block_group *cache;
|
||||
bool must_pin = false;
|
||||
if (!last_ref)
|
||||
return;
|
||||
|
||||
if (root_id != BTRFS_TREE_LOG_OBJECTID) {
|
||||
ret = check_ref_cleanup(trans, buf->start);
|
||||
if (!ret) {
|
||||
btrfs_redirty_list_add(trans->transaction, buf);
|
||||
goto out;
|
||||
}
|
||||
}
|
||||
if (btrfs_header_generation(buf) != trans->transid)
|
||||
goto out;
|
||||
|
||||
cache = btrfs_lookup_block_group(fs_info, buf->start);
|
||||
|
||||
if (btrfs_header_flag(buf, BTRFS_HEADER_FLAG_WRITTEN)) {
|
||||
pin_down_extent(trans, cache, buf->start, buf->len, 1);
|
||||
btrfs_put_block_group(cache);
|
||||
if (root_id != BTRFS_TREE_LOG_OBJECTID) {
|
||||
ret = check_ref_cleanup(trans, buf->start);
|
||||
if (!ret)
|
||||
goto out;
|
||||
}
|
||||
|
||||
/*
|
||||
* If there are tree mod log users we may have recorded mod log
|
||||
* operations for this node. If we re-allocate this node we
|
||||
* could replay operations on this node that happened when it
|
||||
* existed in a completely different root. For example if it
|
||||
* was part of root A, then was reallocated to root B, and we
|
||||
* are doing a btrfs_old_search_slot(root b), we could replay
|
||||
* operations that happened when the block was part of root A,
|
||||
* giving us an inconsistent view of the btree.
|
||||
*
|
||||
* We are safe from races here because at this point no other
|
||||
* node or root points to this extent buffer, so if after this
|
||||
* check a new tree mod log user joins we will not have an
|
||||
* existing log of operations on this node that we have to
|
||||
* contend with.
|
||||
*/
|
||||
if (test_bit(BTRFS_FS_TREE_MOD_LOG_USERS, &fs_info->flags))
|
||||
must_pin = true;
|
||||
|
||||
if (must_pin || btrfs_is_zoned(fs_info)) {
|
||||
btrfs_redirty_list_add(trans->transaction, buf);
|
||||
pin_down_extent(trans, cache, buf->start, buf->len, 1);
|
||||
btrfs_put_block_group(cache);
|
||||
goto out;
|
||||
}
|
||||
|
||||
WARN_ON(test_bit(EXTENT_BUFFER_DIRTY, &buf->bflags));
|
||||
|
||||
btrfs_add_free_space(cache, buf->start, buf->len);
|
||||
btrfs_free_reserved_bytes(cache, buf->len, 0);
|
||||
btrfs_put_block_group(cache);
|
||||
trace_btrfs_reserved_extent_free(fs_info, buf->start, buf->len);
|
||||
}
|
||||
|
||||
bg = btrfs_lookup_block_group(fs_info, buf->start);
|
||||
|
||||
if (btrfs_header_flag(buf, BTRFS_HEADER_FLAG_WRITTEN)) {
|
||||
pin_down_extent(trans, bg, buf->start, buf->len, 1);
|
||||
btrfs_put_block_group(bg);
|
||||
goto out;
|
||||
}
|
||||
|
||||
/*
|
||||
* If there are tree mod log users we may have recorded mod log
|
||||
* operations for this node. If we re-allocate this node we
|
||||
* could replay operations on this node that happened when it
|
||||
* existed in a completely different root. For example if it
|
||||
* was part of root A, then was reallocated to root B, and we
|
||||
* are doing a btrfs_old_search_slot(root b), we could replay
|
||||
* operations that happened when the block was part of root A,
|
||||
* giving us an inconsistent view of the btree.
|
||||
*
|
||||
* We are safe from races here because at this point no other
|
||||
* node or root points to this extent buffer, so if after this
|
||||
* check a new tree mod log user joins we will not have an
|
||||
* existing log of operations on this node that we have to
|
||||
* contend with.
|
||||
*/
|
||||
|
||||
if (test_bit(BTRFS_FS_TREE_MOD_LOG_USERS, &fs_info->flags)
|
||||
|| btrfs_is_zoned(fs_info)) {
|
||||
pin_down_extent(trans, bg, buf->start, buf->len, 1);
|
||||
btrfs_put_block_group(bg);
|
||||
goto out;
|
||||
}
|
||||
|
||||
WARN_ON(test_bit(EXTENT_BUFFER_DIRTY, &buf->bflags));
|
||||
|
||||
btrfs_add_free_space(bg, buf->start, buf->len);
|
||||
btrfs_free_reserved_bytes(bg, buf->len, 0);
|
||||
btrfs_put_block_group(bg);
|
||||
trace_btrfs_reserved_extent_free(fs_info, buf->start, buf->len);
|
||||
|
||||
out:
|
||||
if (last_ref) {
|
||||
/*
|
||||
* Deleting the buffer, clear the corrupt flag since it doesn't
|
||||
* matter anymore.
|
||||
*/
|
||||
clear_bit(EXTENT_BUFFER_CORRUPT, &buf->bflags);
|
||||
}
|
||||
|
||||
/*
|
||||
* Deleting the buffer, clear the corrupt flag since it doesn't
|
||||
* matter anymore.
|
||||
*/
|
||||
clear_bit(EXTENT_BUFFER_CORRUPT, &buf->bflags);
|
||||
}
|
||||
|
||||
/* Can return -ENOMEM */
|
||||
@ -5061,7 +5059,7 @@ btrfs_init_new_buffer(struct btrfs_trans_handle *trans, struct btrfs_root *root,
|
||||
__btrfs_tree_lock(buf, nest);
|
||||
btrfs_clear_buffer_dirty(trans, buf);
|
||||
clear_bit(EXTENT_BUFFER_STALE, &buf->bflags);
|
||||
clear_bit(EXTENT_BUFFER_NO_CHECK, &buf->bflags);
|
||||
clear_bit(EXTENT_BUFFER_ZONED_ZEROOUT, &buf->bflags);
|
||||
|
||||
set_extent_buffer_uptodate(buf);
|
||||
|
||||
|
1051
fs/btrfs/extent_io.c
1051
fs/btrfs/extent_io.c
File diff suppressed because it is too large
Load Diff
@ -28,7 +28,8 @@ enum {
|
||||
EXTENT_BUFFER_IN_TREE,
|
||||
/* write IO error */
|
||||
EXTENT_BUFFER_WRITE_ERR,
|
||||
EXTENT_BUFFER_NO_CHECK,
|
||||
/* Indicate the extent buffer is written zeroed out (for zoned) */
|
||||
EXTENT_BUFFER_ZONED_ZEROOUT,
|
||||
/* Indicate that extent buffer pages a being read */
|
||||
EXTENT_BUFFER_READING,
|
||||
};
|
||||
@ -43,10 +44,10 @@ enum {
|
||||
};
|
||||
|
||||
/*
|
||||
* page->private values. Every page that is controlled by the extent
|
||||
* map has page->private set to one.
|
||||
* Folio private values. Every page that is controlled by the extent map has
|
||||
* folio private set to this value.
|
||||
*/
|
||||
#define EXTENT_PAGE_PRIVATE 1
|
||||
#define EXTENT_FOLIO_PRIVATE 1
|
||||
|
||||
/*
|
||||
* The extent buffer bitmap operations are done with byte granularity instead of
|
||||
@ -77,6 +78,13 @@ struct extent_buffer {
|
||||
unsigned long len;
|
||||
unsigned long bflags;
|
||||
struct btrfs_fs_info *fs_info;
|
||||
|
||||
/*
|
||||
* The address where the eb can be accessed without any cross-page handling.
|
||||
* This can be NULL if not possible.
|
||||
*/
|
||||
void *addr;
|
||||
|
||||
spinlock_t refs_lock;
|
||||
atomic_t refs;
|
||||
int read_mirror;
|
||||
@ -86,7 +94,12 @@ struct extent_buffer {
|
||||
|
||||
struct rw_semaphore lock;
|
||||
|
||||
struct page *pages[INLINE_EXTENT_BUFFER_PAGES];
|
||||
/*
|
||||
* Pointers to all the folios of the extent buffer.
|
||||
*
|
||||
* For now the folio is always order 0 (aka, a single page).
|
||||
*/
|
||||
struct folio *folios[INLINE_EXTENT_BUFFER_PAGES];
|
||||
#ifdef CONFIG_BTRFS_DEBUG
|
||||
struct list_head leak_list;
|
||||
pid_t lock_owner;
|
||||
@ -108,29 +121,43 @@ struct btrfs_eb_write_context {
|
||||
*
|
||||
* Will handle both sectorsize == PAGE_SIZE and sectorsize < PAGE_SIZE cases.
|
||||
*/
|
||||
static inline size_t get_eb_offset_in_page(const struct extent_buffer *eb,
|
||||
unsigned long offset)
|
||||
static inline size_t get_eb_offset_in_folio(const struct extent_buffer *eb,
|
||||
unsigned long offset)
|
||||
{
|
||||
/*
|
||||
* For sectorsize == PAGE_SIZE case, eb->start will always be aligned
|
||||
* to PAGE_SIZE, thus adding it won't cause any difference.
|
||||
* 1) sectorsize == PAGE_SIZE and nodesize >= PAGE_SIZE case
|
||||
* 1.1) One large folio covering the whole eb
|
||||
* The eb->start is aligned to folio size, thus adding it
|
||||
* won't cause any difference.
|
||||
* 1.2) Several page sized folios
|
||||
* The eb->start is aligned to folio (page) size, thus
|
||||
* adding it won't cause any difference.
|
||||
*
|
||||
* For sectorsize < PAGE_SIZE, we must only read the data that belongs
|
||||
* to the eb, thus we have to take the eb->start into consideration.
|
||||
* 2) sectorsize < PAGE_SIZE and nodesize < PAGE_SIZE case
|
||||
* In this case there would only be one page sized folio, and there
|
||||
* may be several different extent buffers in the page/folio.
|
||||
* We need to add eb->start to properly access the offset inside
|
||||
* that eb.
|
||||
*/
|
||||
return offset_in_page(offset + eb->start);
|
||||
return offset_in_folio(eb->folios[0], offset + eb->start);
|
||||
}
|
||||
|
||||
static inline unsigned long get_eb_page_index(unsigned long offset)
|
||||
static inline unsigned long get_eb_folio_index(const struct extent_buffer *eb,
|
||||
unsigned long offset)
|
||||
{
|
||||
/*
|
||||
* For sectorsize == PAGE_SIZE case, plain >> PAGE_SHIFT is enough.
|
||||
* 1) sectorsize == PAGE_SIZE and nodesize >= PAGE_SIZE case
|
||||
* 1.1) One large folio covering the whole eb.
|
||||
* the folio_shift would be large enough to always make us
|
||||
* return 0 as index.
|
||||
* 1.2) Several page sized folios
|
||||
* The folio_shift() would be PAGE_SHIFT, giving us the correct
|
||||
* index.
|
||||
*
|
||||
* For sectorsize < PAGE_SIZE case, we only support 64K PAGE_SIZE,
|
||||
* and have ensured that all tree blocks are contained in one page,
|
||||
* thus we always get index == 0.
|
||||
* 2) sectorsize < PAGE_SIZE and nodesize < PAGE_SIZE case
|
||||
* The folio would only be page sized, and always give us 0 as index.
|
||||
*/
|
||||
return offset >> PAGE_SHIFT;
|
||||
return offset >> folio_shift(eb->folios[0]);
|
||||
}
|
||||
|
||||
/*
|
||||
@ -230,6 +257,20 @@ static inline int num_extent_pages(const struct extent_buffer *eb)
|
||||
return (eb->len >> PAGE_SHIFT) ?: 1;
|
||||
}
|
||||
|
||||
/*
|
||||
* This can only be determined at runtime by checking eb::folios[0].
|
||||
*
|
||||
* As we can have either one large folio covering the whole eb
|
||||
* (either nodesize <= PAGE_SIZE, or high order folio), or multiple
|
||||
* single-paged folios.
|
||||
*/
|
||||
static inline int num_extent_folios(const struct extent_buffer *eb)
|
||||
{
|
||||
if (folio_order(eb->folios[0]))
|
||||
return 1;
|
||||
return num_extent_pages(eb);
|
||||
}
|
||||
|
||||
static inline int extent_buffer_uptodate(const struct extent_buffer *eb)
|
||||
{
|
||||
return test_bit(EXTENT_BUFFER_UPTODATE, &eb->bflags);
|
||||
@ -294,7 +335,8 @@ int extent_invalidate_folio(struct extent_io_tree *tree,
|
||||
void btrfs_clear_buffer_dirty(struct btrfs_trans_handle *trans,
|
||||
struct extent_buffer *buf);
|
||||
|
||||
int btrfs_alloc_page_array(unsigned int nr_pages, struct page **page_array);
|
||||
int btrfs_alloc_page_array(unsigned int nr_pages, struct page **page_array,
|
||||
gfp_t extra_gfp);
|
||||
|
||||
#ifdef CONFIG_BTRFS_FS_RUN_SANITY_TESTS
|
||||
bool find_lock_delalloc_range(struct inode *inode,
|
||||
|
@ -50,7 +50,6 @@ struct extent_map *alloc_extent_map(void)
|
||||
if (!em)
|
||||
return NULL;
|
||||
RB_CLEAR_NODE(&em->rb_node);
|
||||
em->compress_type = BTRFS_COMPRESS_NONE;
|
||||
refcount_set(&em->refs, 1);
|
||||
INIT_LIST_HEAD(&em->list);
|
||||
return em;
|
||||
@ -67,8 +66,6 @@ void free_extent_map(struct extent_map *em)
|
||||
if (refcount_dec_and_test(&em->refs)) {
|
||||
WARN_ON(extent_map_in_tree(em));
|
||||
WARN_ON(!list_empty(&em->list));
|
||||
if (test_bit(EXTENT_FLAG_FS_MAPPING, &em->flags))
|
||||
kfree(em->map_lookup);
|
||||
kmem_cache_free(extent_map_cache, em);
|
||||
}
|
||||
}
|
||||
@ -182,50 +179,50 @@ static struct rb_node *__tree_search(struct rb_root *root, u64 offset,
|
||||
return NULL;
|
||||
}
|
||||
|
||||
/* Check to see if two extent_map structs are adjacent and safe to merge. */
|
||||
static int mergable_maps(struct extent_map *prev, struct extent_map *next)
|
||||
static inline u64 extent_map_block_end(const struct extent_map *em)
|
||||
{
|
||||
if (test_bit(EXTENT_FLAG_PINNED, &prev->flags))
|
||||
return 0;
|
||||
if (em->block_start + em->block_len < em->block_start)
|
||||
return (u64)-1;
|
||||
return em->block_start + em->block_len;
|
||||
}
|
||||
|
||||
/*
|
||||
* don't merge compressed extents, we need to know their
|
||||
* actual size
|
||||
*/
|
||||
if (test_bit(EXTENT_FLAG_COMPRESSED, &prev->flags))
|
||||
return 0;
|
||||
static bool can_merge_extent_map(const struct extent_map *em)
|
||||
{
|
||||
if (em->flags & EXTENT_FLAG_PINNED)
|
||||
return false;
|
||||
|
||||
if (test_bit(EXTENT_FLAG_LOGGING, &prev->flags) ||
|
||||
test_bit(EXTENT_FLAG_LOGGING, &next->flags))
|
||||
return 0;
|
||||
/* Don't merge compressed extents, we need to know their actual size. */
|
||||
if (extent_map_is_compressed(em))
|
||||
return false;
|
||||
|
||||
if (em->flags & EXTENT_FLAG_LOGGING)
|
||||
return false;
|
||||
|
||||
/*
|
||||
* We don't want to merge stuff that hasn't been written to the log yet
|
||||
* since it may not reflect exactly what is on disk, and that would be
|
||||
* bad.
|
||||
*/
|
||||
if (!list_empty(&prev->list) || !list_empty(&next->list))
|
||||
return 0;
|
||||
if (!list_empty(&em->list))
|
||||
return false;
|
||||
|
||||
ASSERT(next->block_start != EXTENT_MAP_DELALLOC &&
|
||||
prev->block_start != EXTENT_MAP_DELALLOC);
|
||||
return true;
|
||||
}
|
||||
|
||||
if (prev->map_lookup || next->map_lookup)
|
||||
ASSERT(test_bit(EXTENT_FLAG_FS_MAPPING, &prev->flags) &&
|
||||
test_bit(EXTENT_FLAG_FS_MAPPING, &next->flags));
|
||||
/* Check to see if two extent_map structs are adjacent and safe to merge. */
|
||||
static bool mergeable_maps(const struct extent_map *prev, const struct extent_map *next)
|
||||
{
|
||||
if (extent_map_end(prev) != next->start)
|
||||
return false;
|
||||
|
||||
if (extent_map_end(prev) == next->start &&
|
||||
prev->flags == next->flags &&
|
||||
prev->map_lookup == next->map_lookup &&
|
||||
((next->block_start == EXTENT_MAP_HOLE &&
|
||||
prev->block_start == EXTENT_MAP_HOLE) ||
|
||||
(next->block_start == EXTENT_MAP_INLINE &&
|
||||
prev->block_start == EXTENT_MAP_INLINE) ||
|
||||
(next->block_start < EXTENT_MAP_LAST_BYTE - 1 &&
|
||||
next->block_start == extent_map_block_end(prev)))) {
|
||||
return 1;
|
||||
}
|
||||
return 0;
|
||||
if (prev->flags != next->flags)
|
||||
return false;
|
||||
|
||||
if (next->block_start < EXTENT_MAP_LAST_BYTE - 1)
|
||||
return next->block_start == extent_map_block_end(prev);
|
||||
|
||||
/* HOLES and INLINE extents. */
|
||||
return next->block_start == prev->block_start;
|
||||
}
|
||||
|
||||
static void try_merge_map(struct extent_map_tree *tree, struct extent_map *em)
|
||||
@ -244,11 +241,14 @@ static void try_merge_map(struct extent_map_tree *tree, struct extent_map *em)
|
||||
if (refcount_read(&em->refs) > 2)
|
||||
return;
|
||||
|
||||
if (!can_merge_extent_map(em))
|
||||
return;
|
||||
|
||||
if (em->start != 0) {
|
||||
rb = rb_prev(&em->rb_node);
|
||||
if (rb)
|
||||
merge = rb_entry(rb, struct extent_map, rb_node);
|
||||
if (rb && mergable_maps(merge, em)) {
|
||||
if (rb && can_merge_extent_map(merge) && mergeable_maps(merge, em)) {
|
||||
em->start = merge->start;
|
||||
em->orig_start = merge->orig_start;
|
||||
em->len += merge->len;
|
||||
@ -257,7 +257,7 @@ static void try_merge_map(struct extent_map_tree *tree, struct extent_map *em)
|
||||
em->mod_len = (em->mod_len + em->mod_start) - merge->mod_start;
|
||||
em->mod_start = merge->mod_start;
|
||||
em->generation = max(em->generation, merge->generation);
|
||||
set_bit(EXTENT_FLAG_MERGED, &em->flags);
|
||||
em->flags |= EXTENT_FLAG_MERGED;
|
||||
|
||||
rb_erase_cached(&merge->rb_node, &tree->map);
|
||||
RB_CLEAR_NODE(&merge->rb_node);
|
||||
@ -268,14 +268,14 @@ static void try_merge_map(struct extent_map_tree *tree, struct extent_map *em)
|
||||
rb = rb_next(&em->rb_node);
|
||||
if (rb)
|
||||
merge = rb_entry(rb, struct extent_map, rb_node);
|
||||
if (rb && mergable_maps(em, merge)) {
|
||||
if (rb && can_merge_extent_map(merge) && mergeable_maps(em, merge)) {
|
||||
em->len += merge->len;
|
||||
em->block_len += merge->block_len;
|
||||
rb_erase_cached(&merge->rb_node, &tree->map);
|
||||
RB_CLEAR_NODE(&merge->rb_node);
|
||||
em->mod_len = (merge->mod_start + merge->mod_len) - em->mod_start;
|
||||
em->generation = max(em->generation, merge->generation);
|
||||
set_bit(EXTENT_FLAG_MERGED, &em->flags);
|
||||
em->flags |= EXTENT_FLAG_MERGED;
|
||||
free_extent_map(merge);
|
||||
}
|
||||
}
|
||||
@ -283,7 +283,7 @@ static void try_merge_map(struct extent_map_tree *tree, struct extent_map *em)
|
||||
/*
|
||||
* Unpin an extent from the cache.
|
||||
*
|
||||
* @tree: tree to unpin the extent in
|
||||
* @inode: the inode from which we are unpinning an extent range
|
||||
* @start: logical offset in the file
|
||||
* @len: length of the extent
|
||||
* @gen: generation that this extent has been modified in
|
||||
@ -292,9 +292,10 @@ static void try_merge_map(struct extent_map_tree *tree, struct extent_map *em)
|
||||
* to the generation that actually added the file item to the inode so we know
|
||||
* we need to sync this extent when we call fsync().
|
||||
*/
|
||||
int unpin_extent_cache(struct extent_map_tree *tree, u64 start, u64 len,
|
||||
u64 gen)
|
||||
int unpin_extent_cache(struct btrfs_inode *inode, u64 start, u64 len, u64 gen)
|
||||
{
|
||||
struct btrfs_fs_info *fs_info = inode->root->fs_info;
|
||||
struct extent_map_tree *tree = &inode->extent_tree;
|
||||
int ret = 0;
|
||||
struct extent_map *em;
|
||||
bool prealloc = false;
|
||||
@ -302,19 +303,28 @@ int unpin_extent_cache(struct extent_map_tree *tree, u64 start, u64 len,
|
||||
write_lock(&tree->lock);
|
||||
em = lookup_extent_mapping(tree, start, len);
|
||||
|
||||
WARN_ON(!em || em->start != start);
|
||||
|
||||
if (!em)
|
||||
if (WARN_ON(!em)) {
|
||||
btrfs_warn(fs_info,
|
||||
"no extent map found for inode %llu (root %lld) when unpinning extent range [%llu, %llu), generation %llu",
|
||||
btrfs_ino(inode), btrfs_root_id(inode->root),
|
||||
start, len, gen);
|
||||
goto out;
|
||||
}
|
||||
|
||||
if (WARN_ON(em->start != start))
|
||||
btrfs_warn(fs_info,
|
||||
"found extent map for inode %llu (root %lld) with unexpected start offset %llu when unpinning extent range [%llu, %llu), generation %llu",
|
||||
btrfs_ino(inode), btrfs_root_id(inode->root),
|
||||
em->start, start, len, gen);
|
||||
|
||||
em->generation = gen;
|
||||
clear_bit(EXTENT_FLAG_PINNED, &em->flags);
|
||||
em->flags &= ~EXTENT_FLAG_PINNED;
|
||||
em->mod_start = em->start;
|
||||
em->mod_len = em->len;
|
||||
|
||||
if (test_bit(EXTENT_FLAG_FILLING, &em->flags)) {
|
||||
if (em->flags & EXTENT_FLAG_FILLING) {
|
||||
prealloc = true;
|
||||
clear_bit(EXTENT_FLAG_FILLING, &em->flags);
|
||||
em->flags &= ~EXTENT_FLAG_FILLING;
|
||||
}
|
||||
|
||||
try_merge_map(tree, em);
|
||||
@ -335,7 +345,7 @@ void clear_em_logging(struct extent_map_tree *tree, struct extent_map *em)
|
||||
{
|
||||
lockdep_assert_held_write(&tree->lock);
|
||||
|
||||
clear_bit(EXTENT_FLAG_LOGGING, &em->flags);
|
||||
em->flags &= ~EXTENT_FLAG_LOGGING;
|
||||
if (extent_map_in_tree(em))
|
||||
try_merge_map(tree, em);
|
||||
}
|
||||
@ -348,45 +358,14 @@ static inline void setup_extent_mapping(struct extent_map_tree *tree,
|
||||
em->mod_start = em->start;
|
||||
em->mod_len = em->len;
|
||||
|
||||
ASSERT(list_empty(&em->list));
|
||||
|
||||
if (modified)
|
||||
list_move(&em->list, &tree->modified_extents);
|
||||
list_add(&em->list, &tree->modified_extents);
|
||||
else
|
||||
try_merge_map(tree, em);
|
||||
}
|
||||
|
||||
static void extent_map_device_set_bits(struct extent_map *em, unsigned bits)
|
||||
{
|
||||
struct map_lookup *map = em->map_lookup;
|
||||
u64 stripe_size = em->orig_block_len;
|
||||
int i;
|
||||
|
||||
for (i = 0; i < map->num_stripes; i++) {
|
||||
struct btrfs_io_stripe *stripe = &map->stripes[i];
|
||||
struct btrfs_device *device = stripe->dev;
|
||||
|
||||
set_extent_bit(&device->alloc_state, stripe->physical,
|
||||
stripe->physical + stripe_size - 1,
|
||||
bits | EXTENT_NOWAIT, NULL);
|
||||
}
|
||||
}
|
||||
|
||||
static void extent_map_device_clear_bits(struct extent_map *em, unsigned bits)
|
||||
{
|
||||
struct map_lookup *map = em->map_lookup;
|
||||
u64 stripe_size = em->orig_block_len;
|
||||
int i;
|
||||
|
||||
for (i = 0; i < map->num_stripes; i++) {
|
||||
struct btrfs_io_stripe *stripe = &map->stripes[i];
|
||||
struct btrfs_device *device = stripe->dev;
|
||||
|
||||
__clear_extent_bit(&device->alloc_state, stripe->physical,
|
||||
stripe->physical + stripe_size - 1,
|
||||
bits | EXTENT_NOWAIT,
|
||||
NULL, NULL);
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* Add new extent map to the extent tree
|
||||
*
|
||||
@ -400,8 +379,8 @@ static void extent_map_device_clear_bits(struct extent_map *em, unsigned bits)
|
||||
* into the tree directly, with an additional reference taken, or a
|
||||
* reference dropped if the merge attempt was successful.
|
||||
*/
|
||||
int add_extent_mapping(struct extent_map_tree *tree,
|
||||
struct extent_map *em, int modified)
|
||||
static int add_extent_mapping(struct extent_map_tree *tree,
|
||||
struct extent_map *em, int modified)
|
||||
{
|
||||
int ret = 0;
|
||||
|
||||
@ -412,10 +391,6 @@ int add_extent_mapping(struct extent_map_tree *tree,
|
||||
goto out;
|
||||
|
||||
setup_extent_mapping(tree, em, modified);
|
||||
if (test_bit(EXTENT_FLAG_FS_MAPPING, &em->flags)) {
|
||||
extent_map_device_set_bits(em, CHUNK_ALLOCATED);
|
||||
extent_map_device_clear_bits(em, CHUNK_TRIMMED);
|
||||
}
|
||||
out:
|
||||
return ret;
|
||||
}
|
||||
@ -495,12 +470,10 @@ void remove_extent_mapping(struct extent_map_tree *tree, struct extent_map *em)
|
||||
{
|
||||
lockdep_assert_held_write(&tree->lock);
|
||||
|
||||
WARN_ON(test_bit(EXTENT_FLAG_PINNED, &em->flags));
|
||||
WARN_ON(em->flags & EXTENT_FLAG_PINNED);
|
||||
rb_erase_cached(&em->rb_node, &tree->map);
|
||||
if (!test_bit(EXTENT_FLAG_LOGGING, &em->flags))
|
||||
if (!(em->flags & EXTENT_FLAG_LOGGING))
|
||||
list_del_init(&em->list);
|
||||
if (test_bit(EXTENT_FLAG_FS_MAPPING, &em->flags))
|
||||
extent_map_device_clear_bits(em, CHUNK_ALLOCATED);
|
||||
RB_CLEAR_NODE(&em->rb_node);
|
||||
}
|
||||
|
||||
@ -511,9 +484,9 @@ static void replace_extent_mapping(struct extent_map_tree *tree,
|
||||
{
|
||||
lockdep_assert_held_write(&tree->lock);
|
||||
|
||||
WARN_ON(test_bit(EXTENT_FLAG_PINNED, &cur->flags));
|
||||
WARN_ON(cur->flags & EXTENT_FLAG_PINNED);
|
||||
ASSERT(extent_map_in_tree(cur));
|
||||
if (!test_bit(EXTENT_FLAG_LOGGING, &cur->flags))
|
||||
if (!(cur->flags & EXTENT_FLAG_LOGGING))
|
||||
list_del_init(&cur->list);
|
||||
rb_replace_node_cached(&cur->rb_node, &new->rb_node, &tree->map);
|
||||
RB_CLEAR_NODE(&cur->rb_node);
|
||||
@ -576,7 +549,7 @@ static noinline int merge_extent_mapping(struct extent_map_tree *em_tree,
|
||||
em->start = start;
|
||||
em->len = end - start;
|
||||
if (em->block_start < EXTENT_MAP_LAST_BYTE &&
|
||||
!test_bit(EXTENT_FLAG_COMPRESSED, &em->flags)) {
|
||||
!extent_map_is_compressed(em)) {
|
||||
em->block_start += start_diff;
|
||||
em->block_len = em->len;
|
||||
}
|
||||
@ -626,8 +599,6 @@ int btrfs_add_extent_mapping(struct btrfs_fs_info *fs_info,
|
||||
if (ret == -EEXIST) {
|
||||
struct extent_map *existing;
|
||||
|
||||
ret = 0;
|
||||
|
||||
existing = search_extent_mapping(em_tree, start, len);
|
||||
|
||||
trace_btrfs_handle_em_exist(fs_info, existing, em, start, len);
|
||||
@ -681,8 +652,7 @@ static void drop_all_extent_maps_fast(struct extent_map_tree *tree)
|
||||
|
||||
node = rb_first_cached(&tree->map);
|
||||
em = rb_entry(node, struct extent_map, rb_node);
|
||||
clear_bit(EXTENT_FLAG_PINNED, &em->flags);
|
||||
clear_bit(EXTENT_FLAG_LOGGING, &em->flags);
|
||||
em->flags &= ~(EXTENT_FLAG_PINNED | EXTENT_FLAG_LOGGING);
|
||||
remove_extent_mapping(tree, em);
|
||||
free_extent_map(em);
|
||||
cond_resched_rwlock_write(&tree->lock);
|
||||
@ -758,19 +728,18 @@ void btrfs_drop_extent_map_range(struct btrfs_inode *inode, u64 start, u64 end,
|
||||
}
|
||||
}
|
||||
|
||||
if (skip_pinned && test_bit(EXTENT_FLAG_PINNED, &em->flags)) {
|
||||
if (skip_pinned && (em->flags & EXTENT_FLAG_PINNED)) {
|
||||
start = em_end;
|
||||
goto next;
|
||||
}
|
||||
|
||||
flags = em->flags;
|
||||
clear_bit(EXTENT_FLAG_PINNED, &em->flags);
|
||||
/*
|
||||
* In case we split the extent map, we want to preserve the
|
||||
* EXTENT_FLAG_LOGGING flag on our extent map, but we don't want
|
||||
* it on the new extent maps.
|
||||
*/
|
||||
clear_bit(EXTENT_FLAG_LOGGING, &flags);
|
||||
em->flags &= ~(EXTENT_FLAG_PINNED | EXTENT_FLAG_LOGGING);
|
||||
modified = !list_empty(&em->list);
|
||||
|
||||
/*
|
||||
@ -781,7 +750,7 @@ void btrfs_drop_extent_map_range(struct btrfs_inode *inode, u64 start, u64 end,
|
||||
goto remove_em;
|
||||
|
||||
gen = em->generation;
|
||||
compressed = test_bit(EXTENT_FLAG_COMPRESSED, &em->flags);
|
||||
compressed = extent_map_is_compressed(em);
|
||||
|
||||
if (em->start < start) {
|
||||
if (!split) {
|
||||
@ -814,7 +783,6 @@ void btrfs_drop_extent_map_range(struct btrfs_inode *inode, u64 start, u64 end,
|
||||
|
||||
split->generation = gen;
|
||||
split->flags = flags;
|
||||
split->compress_type = em->compress_type;
|
||||
replace_extent_mapping(em_tree, em, split, modified);
|
||||
free_extent_map(split);
|
||||
split = split2;
|
||||
@ -831,7 +799,6 @@ void btrfs_drop_extent_map_range(struct btrfs_inode *inode, u64 start, u64 end,
|
||||
split->len = em_end - end;
|
||||
split->block_start = em->block_start;
|
||||
split->flags = flags;
|
||||
split->compress_type = em->compress_type;
|
||||
split->generation = gen;
|
||||
|
||||
if (em->block_start < EXTENT_MAP_LAST_BYTE) {
|
||||
@ -997,14 +964,14 @@ int split_extent_map(struct btrfs_inode *inode, u64 start, u64 len, u64 pre,
|
||||
}
|
||||
|
||||
ASSERT(em->len == len);
|
||||
ASSERT(!test_bit(EXTENT_FLAG_COMPRESSED, &em->flags));
|
||||
ASSERT(!extent_map_is_compressed(em));
|
||||
ASSERT(em->block_start < EXTENT_MAP_LAST_BYTE);
|
||||
ASSERT(test_bit(EXTENT_FLAG_PINNED, &em->flags));
|
||||
ASSERT(!test_bit(EXTENT_FLAG_LOGGING, &em->flags));
|
||||
ASSERT(em->flags & EXTENT_FLAG_PINNED);
|
||||
ASSERT(!(em->flags & EXTENT_FLAG_LOGGING));
|
||||
ASSERT(!list_empty(&em->list));
|
||||
|
||||
flags = em->flags;
|
||||
clear_bit(EXTENT_FLAG_PINNED, &em->flags);
|
||||
em->flags &= ~EXTENT_FLAG_PINNED;
|
||||
|
||||
/* First, replace the em with a new extent_map starting from * em->start */
|
||||
split_pre->start = em->start;
|
||||
@ -1015,7 +982,6 @@ int split_extent_map(struct btrfs_inode *inode, u64 start, u64 len, u64 pre,
|
||||
split_pre->orig_block_len = split_pre->block_len;
|
||||
split_pre->ram_bytes = split_pre->len;
|
||||
split_pre->flags = flags;
|
||||
split_pre->compress_type = em->compress_type;
|
||||
split_pre->generation = em->generation;
|
||||
|
||||
replace_extent_mapping(em_tree, em, split_pre, 1);
|
||||
@ -1034,7 +1000,6 @@ int split_extent_map(struct btrfs_inode *inode, u64 start, u64 len, u64 pre,
|
||||
split_mid->orig_block_len = split_mid->block_len;
|
||||
split_mid->ram_bytes = split_mid->len;
|
||||
split_mid->flags = flags;
|
||||
split_mid->compress_type = em->compress_type;
|
||||
split_mid->generation = em->generation;
|
||||
add_extent_mapping(em_tree, split_mid, 1);
|
||||
|
||||
|
@ -5,30 +5,33 @@
|
||||
|
||||
#include <linux/rbtree.h>
|
||||
#include <linux/refcount.h>
|
||||
#include "compression.h"
|
||||
|
||||
#define EXTENT_MAP_LAST_BYTE ((u64)-4)
|
||||
#define EXTENT_MAP_HOLE ((u64)-3)
|
||||
#define EXTENT_MAP_INLINE ((u64)-2)
|
||||
/* used only during fiemap calls */
|
||||
#define EXTENT_MAP_DELALLOC ((u64)-1)
|
||||
|
||||
/* bits for the extent_map::flags field */
|
||||
enum {
|
||||
/* this entry not yet on disk, don't free it */
|
||||
EXTENT_FLAG_PINNED,
|
||||
EXTENT_FLAG_COMPRESSED,
|
||||
ENUM_BIT(EXTENT_FLAG_PINNED),
|
||||
ENUM_BIT(EXTENT_FLAG_COMPRESS_ZLIB),
|
||||
ENUM_BIT(EXTENT_FLAG_COMPRESS_LZO),
|
||||
ENUM_BIT(EXTENT_FLAG_COMPRESS_ZSTD),
|
||||
/* pre-allocated extent */
|
||||
EXTENT_FLAG_PREALLOC,
|
||||
ENUM_BIT(EXTENT_FLAG_PREALLOC),
|
||||
/* Logging this extent */
|
||||
EXTENT_FLAG_LOGGING,
|
||||
ENUM_BIT(EXTENT_FLAG_LOGGING),
|
||||
/* Filling in a preallocated extent */
|
||||
EXTENT_FLAG_FILLING,
|
||||
/* filesystem extent mapping type */
|
||||
EXTENT_FLAG_FS_MAPPING,
|
||||
ENUM_BIT(EXTENT_FLAG_FILLING),
|
||||
/* This em is merged from two or more physically adjacent ems */
|
||||
EXTENT_FLAG_MERGED,
|
||||
ENUM_BIT(EXTENT_FLAG_MERGED),
|
||||
};
|
||||
|
||||
/*
|
||||
* Keep this structure as compact as possible, as we can have really large
|
||||
* amounts of allocated extent maps at any time.
|
||||
*/
|
||||
struct extent_map {
|
||||
struct rb_node rb_node;
|
||||
|
||||
@ -49,11 +52,8 @@ struct extent_map {
|
||||
* For non-merged extents, it's from btrfs_file_extent_item::generation.
|
||||
*/
|
||||
u64 generation;
|
||||
unsigned long flags;
|
||||
/* Used for chunk mappings, flag EXTENT_FLAG_FS_MAPPING must be set */
|
||||
struct map_lookup *map_lookup;
|
||||
u32 flags;
|
||||
refcount_t refs;
|
||||
unsigned int compress_type;
|
||||
struct list_head list;
|
||||
};
|
||||
|
||||
@ -65,30 +65,57 @@ struct extent_map_tree {
|
||||
|
||||
struct btrfs_inode;
|
||||
|
||||
static inline void extent_map_set_compression(struct extent_map *em,
|
||||
enum btrfs_compression_type type)
|
||||
{
|
||||
if (type == BTRFS_COMPRESS_ZLIB)
|
||||
em->flags |= EXTENT_FLAG_COMPRESS_ZLIB;
|
||||
else if (type == BTRFS_COMPRESS_LZO)
|
||||
em->flags |= EXTENT_FLAG_COMPRESS_LZO;
|
||||
else if (type == BTRFS_COMPRESS_ZSTD)
|
||||
em->flags |= EXTENT_FLAG_COMPRESS_ZSTD;
|
||||
}
|
||||
|
||||
static inline enum btrfs_compression_type extent_map_compression(const struct extent_map *em)
|
||||
{
|
||||
if (em->flags & EXTENT_FLAG_COMPRESS_ZLIB)
|
||||
return BTRFS_COMPRESS_ZLIB;
|
||||
|
||||
if (em->flags & EXTENT_FLAG_COMPRESS_LZO)
|
||||
return BTRFS_COMPRESS_LZO;
|
||||
|
||||
if (em->flags & EXTENT_FLAG_COMPRESS_ZSTD)
|
||||
return BTRFS_COMPRESS_ZSTD;
|
||||
|
||||
return BTRFS_COMPRESS_NONE;
|
||||
}
|
||||
|
||||
/*
|
||||
* More efficient way to determine if extent is compressed, instead of using
|
||||
* 'extent_map_compression() != BTRFS_COMPRESS_NONE'.
|
||||
*/
|
||||
static inline bool extent_map_is_compressed(const struct extent_map *em)
|
||||
{
|
||||
return (em->flags & (EXTENT_FLAG_COMPRESS_ZLIB |
|
||||
EXTENT_FLAG_COMPRESS_LZO |
|
||||
EXTENT_FLAG_COMPRESS_ZSTD)) != 0;
|
||||
}
|
||||
|
||||
static inline int extent_map_in_tree(const struct extent_map *em)
|
||||
{
|
||||
return !RB_EMPTY_NODE(&em->rb_node);
|
||||
}
|
||||
|
||||
static inline u64 extent_map_end(struct extent_map *em)
|
||||
static inline u64 extent_map_end(const struct extent_map *em)
|
||||
{
|
||||
if (em->start + em->len < em->start)
|
||||
return (u64)-1;
|
||||
return em->start + em->len;
|
||||
}
|
||||
|
||||
static inline u64 extent_map_block_end(struct extent_map *em)
|
||||
{
|
||||
if (em->block_start + em->block_len < em->block_start)
|
||||
return (u64)-1;
|
||||
return em->block_start + em->block_len;
|
||||
}
|
||||
|
||||
void extent_map_tree_init(struct extent_map_tree *tree);
|
||||
struct extent_map *lookup_extent_mapping(struct extent_map_tree *tree,
|
||||
u64 start, u64 len);
|
||||
int add_extent_mapping(struct extent_map_tree *tree,
|
||||
struct extent_map *em, int modified);
|
||||
void remove_extent_mapping(struct extent_map_tree *tree, struct extent_map *em);
|
||||
int split_extent_map(struct btrfs_inode *inode, u64 start, u64 len, u64 pre,
|
||||
u64 new_logical);
|
||||
@ -97,7 +124,7 @@ struct extent_map *alloc_extent_map(void);
|
||||
void free_extent_map(struct extent_map *em);
|
||||
int __init extent_map_init(void);
|
||||
void __cold extent_map_exit(void);
|
||||
int unpin_extent_cache(struct extent_map_tree *tree, u64 start, u64 len, u64 gen);
|
||||
int unpin_extent_cache(struct btrfs_inode *inode, u64 start, u64 len, u64 gen);
|
||||
void clear_em_logging(struct extent_map_tree *tree, struct extent_map *em);
|
||||
struct extent_map *search_extent_mapping(struct extent_map_tree *tree,
|
||||
u64 start, u64 len);
|
||||
|
@ -59,7 +59,7 @@ void btrfs_inode_safe_disk_i_size_write(struct btrfs_inode *inode, u64 new_i_siz
|
||||
goto out_unlock;
|
||||
}
|
||||
|
||||
ret = find_contiguous_extent_bit(&inode->file_extent_tree, 0, &start,
|
||||
ret = find_contiguous_extent_bit(inode->file_extent_tree, 0, &start,
|
||||
&end, EXTENT_DIRTY);
|
||||
if (!ret && start == 0)
|
||||
i_size = min(i_size, end + 1);
|
||||
@ -94,7 +94,7 @@ int btrfs_inode_set_file_extent_range(struct btrfs_inode *inode, u64 start,
|
||||
|
||||
if (btrfs_fs_incompat(inode->root->fs_info, NO_HOLES))
|
||||
return 0;
|
||||
return set_extent_bit(&inode->file_extent_tree, start, start + len - 1,
|
||||
return set_extent_bit(inode->file_extent_tree, start, start + len - 1,
|
||||
EXTENT_DIRTY, NULL);
|
||||
}
|
||||
|
||||
@ -123,7 +123,7 @@ int btrfs_inode_clear_file_extent_range(struct btrfs_inode *inode, u64 start,
|
||||
|
||||
if (btrfs_fs_incompat(inode->root->fs_info, NO_HOLES))
|
||||
return 0;
|
||||
return clear_extent_bit(&inode->file_extent_tree, start,
|
||||
return clear_extent_bit(inode->file_extent_tree, start,
|
||||
start + len - 1, EXTENT_DIRTY, NULL);
|
||||
}
|
||||
|
||||
@ -1294,8 +1294,7 @@ void btrfs_extent_item_to_extent_map(struct btrfs_inode *inode,
|
||||
return;
|
||||
}
|
||||
if (compress_type != BTRFS_COMPRESS_NONE) {
|
||||
set_bit(EXTENT_FLAG_COMPRESSED, &em->flags);
|
||||
em->compress_type = compress_type;
|
||||
extent_map_set_compression(em, compress_type);
|
||||
em->block_start = bytenr;
|
||||
em->block_len = em->orig_block_len;
|
||||
} else {
|
||||
@ -1303,7 +1302,7 @@ void btrfs_extent_item_to_extent_map(struct btrfs_inode *inode,
|
||||
em->block_start = bytenr;
|
||||
em->block_len = em->len;
|
||||
if (type == BTRFS_FILE_EXTENT_PREALLOC)
|
||||
set_bit(EXTENT_FLAG_PREALLOC, &em->flags);
|
||||
em->flags |= EXTENT_FLAG_PREALLOC;
|
||||
}
|
||||
} else if (type == BTRFS_FILE_EXTENT_INLINE) {
|
||||
em->block_start = EXTENT_MAP_INLINE;
|
||||
@ -1315,9 +1314,7 @@ void btrfs_extent_item_to_extent_map(struct btrfs_inode *inode,
|
||||
*/
|
||||
em->orig_start = EXTENT_MAP_HOLE;
|
||||
em->block_len = (u64)-1;
|
||||
em->compress_type = compress_type;
|
||||
if (compress_type != BTRFS_COMPRESS_NONE)
|
||||
set_bit(EXTENT_FLAG_COMPRESSED, &em->flags);
|
||||
extent_map_set_compression(em, compress_type);
|
||||
} else {
|
||||
btrfs_err(fs_info,
|
||||
"unknown file extent item type %d, inode %llu, offset %llu, "
|
||||
|
@ -111,8 +111,8 @@ static void btrfs_drop_pages(struct btrfs_fs_info *fs_info,
|
||||
* accessed as prepare_pages should have marked them accessed
|
||||
* in prepare_pages via find_or_create_page()
|
||||
*/
|
||||
btrfs_page_clamp_clear_checked(fs_info, pages[i], block_start,
|
||||
block_len);
|
||||
btrfs_folio_clamp_clear_checked(fs_info, page_folio(pages[i]),
|
||||
block_start, block_len);
|
||||
unlock_page(pages[i]);
|
||||
put_page(pages[i]);
|
||||
}
|
||||
@ -168,9 +168,12 @@ int btrfs_dirty_pages(struct btrfs_inode *inode, struct page **pages,
|
||||
for (i = 0; i < num_pages; i++) {
|
||||
struct page *p = pages[i];
|
||||
|
||||
btrfs_page_clamp_set_uptodate(fs_info, p, start_pos, num_bytes);
|
||||
btrfs_page_clamp_clear_checked(fs_info, p, start_pos, num_bytes);
|
||||
btrfs_page_clamp_set_dirty(fs_info, p, start_pos, num_bytes);
|
||||
btrfs_folio_clamp_set_uptodate(fs_info, page_folio(p),
|
||||
start_pos, num_bytes);
|
||||
btrfs_folio_clamp_clear_checked(fs_info, page_folio(p),
|
||||
start_pos, num_bytes);
|
||||
btrfs_folio_clamp_set_dirty(fs_info, page_folio(p),
|
||||
start_pos, num_bytes);
|
||||
}
|
||||
|
||||
/*
|
||||
@ -869,9 +872,9 @@ static int prepare_uptodate_page(struct inode *inode,
|
||||
* released.
|
||||
*
|
||||
* The private flag check is essential for subpage as we need
|
||||
* to store extra bitmap using page->private.
|
||||
* to store extra bitmap using folio private.
|
||||
*/
|
||||
if (page->mapping != inode->i_mapping || !PagePrivate(page)) {
|
||||
if (page->mapping != inode->i_mapping || !folio_test_private(folio)) {
|
||||
unlock_page(page);
|
||||
return -EAGAIN;
|
||||
}
|
||||
@ -2150,7 +2153,6 @@ out:
|
||||
hole_em->block_start = EXTENT_MAP_HOLE;
|
||||
hole_em->block_len = 0;
|
||||
hole_em->orig_block_len = 0;
|
||||
hole_em->compress_type = BTRFS_COMPRESS_NONE;
|
||||
hole_em->generation = trans->transid;
|
||||
|
||||
ret = btrfs_replace_extent_map_range(inode, hole_em, true);
|
||||
@ -2839,7 +2841,7 @@ static int btrfs_zero_range_check_range_boundary(struct btrfs_inode *inode,
|
||||
|
||||
if (em->block_start == EXTENT_MAP_HOLE)
|
||||
ret = RANGE_BOUNDARY_HOLE;
|
||||
else if (test_bit(EXTENT_FLAG_PREALLOC, &em->flags))
|
||||
else if (em->flags & EXTENT_FLAG_PREALLOC)
|
||||
ret = RANGE_BOUNDARY_PREALLOC_EXTENT;
|
||||
else
|
||||
ret = RANGE_BOUNDARY_WRITTEN_EXTENT;
|
||||
@ -2879,8 +2881,7 @@ static int btrfs_zero_range(struct inode *inode,
|
||||
* extents and holes, we drop all the existing extents and allocate a
|
||||
* new prealloc extent, so that we get a larger contiguous disk extent.
|
||||
*/
|
||||
if (em->start <= alloc_start &&
|
||||
test_bit(EXTENT_FLAG_PREALLOC, &em->flags)) {
|
||||
if (em->start <= alloc_start && (em->flags & EXTENT_FLAG_PREALLOC)) {
|
||||
const u64 em_end = em->start + em->len;
|
||||
|
||||
if (em_end >= offset + len) {
|
||||
@ -2915,7 +2916,7 @@ static int btrfs_zero_range(struct inode *inode,
|
||||
goto out;
|
||||
}
|
||||
|
||||
if (test_bit(EXTENT_FLAG_PREALLOC, &em->flags)) {
|
||||
if (em->flags & EXTENT_FLAG_PREALLOC) {
|
||||
free_extent_map(em);
|
||||
ret = btrfs_fallocate_update_isize(inode, offset + len,
|
||||
mode);
|
||||
@ -3136,7 +3137,7 @@ static long btrfs_fallocate(struct file *file, int mode,
|
||||
last_byte = ALIGN(last_byte, blocksize);
|
||||
if (em->block_start == EXTENT_MAP_HOLE ||
|
||||
(cur_offset >= inode->i_size &&
|
||||
!test_bit(EXTENT_FLAG_PREALLOC, &em->flags))) {
|
||||
!(em->flags & EXTENT_FLAG_PREALLOC))) {
|
||||
const u64 range_len = last_byte - cur_offset;
|
||||
|
||||
ret = add_falloc_range(&reserve_list, cur_offset, range_len);
|
||||
|
@ -439,8 +439,8 @@ static void io_ctl_drop_pages(struct btrfs_io_ctl *io_ctl)
|
||||
|
||||
for (i = 0; i < io_ctl->num_pages; i++) {
|
||||
if (io_ctl->pages[i]) {
|
||||
btrfs_page_clear_checked(io_ctl->fs_info,
|
||||
io_ctl->pages[i],
|
||||
btrfs_folio_clear_checked(io_ctl->fs_info,
|
||||
page_folio(io_ctl->pages[i]),
|
||||
page_offset(io_ctl->pages[i]),
|
||||
PAGE_SIZE);
|
||||
unlock_page(io_ctl->pages[i]);
|
||||
|
@ -188,6 +188,7 @@ enum {
|
||||
BTRFS_MOUNT_IGNOREBADROOTS = (1UL << 27),
|
||||
BTRFS_MOUNT_IGNOREDATACSUMS = (1UL << 28),
|
||||
BTRFS_MOUNT_NODISCARD = (1UL << 29),
|
||||
BTRFS_MOUNT_NOSPACECACHE = (1UL << 30),
|
||||
};
|
||||
|
||||
/*
|
||||
@ -398,7 +399,8 @@ struct btrfs_fs_info {
|
||||
struct extent_io_tree excluded_extents;
|
||||
|
||||
/* logical->physical extent mapping */
|
||||
struct extent_map_tree mapping_tree;
|
||||
struct rb_root_cached mapping_tree;
|
||||
rwlock_t mapping_tree_lock;
|
||||
|
||||
/*
|
||||
* Block reservation for extent, checksum, root tree and delayed dir
|
||||
@ -960,20 +962,6 @@ void __btrfs_clear_fs_compat_ro(struct btrfs_fs_info *fs_info, u64 flag,
|
||||
#define btrfs_test_opt(fs_info, opt) ((fs_info)->mount_opt & \
|
||||
BTRFS_MOUNT_##opt)
|
||||
|
||||
#define btrfs_set_and_info(fs_info, opt, fmt, args...) \
|
||||
do { \
|
||||
if (!btrfs_test_opt(fs_info, opt)) \
|
||||
btrfs_info(fs_info, fmt, ##args); \
|
||||
btrfs_set_opt(fs_info->mount_opt, opt); \
|
||||
} while (0)
|
||||
|
||||
#define btrfs_clear_and_info(fs_info, opt, fmt, args...) \
|
||||
do { \
|
||||
if (btrfs_test_opt(fs_info, opt)) \
|
||||
btrfs_info(fs_info, fmt, ##args); \
|
||||
btrfs_clear_opt(fs_info->mount_opt, opt); \
|
||||
} while (0)
|
||||
|
||||
static inline int btrfs_fs_closing(struct btrfs_fs_info *fs_info)
|
||||
{
|
||||
/* Do it this way so we only ever do one test_bit in the normal case. */
|
||||
|
153
fs/btrfs/inode.c
153
fs/btrfs/inode.c
@ -114,6 +114,15 @@ struct data_reloc_warn {
|
||||
int mirror_num;
|
||||
};
|
||||
|
||||
/*
|
||||
* For the file_extent_tree, we want to hold the inode lock when we lookup and
|
||||
* update the disk_i_size, but lockdep will complain because our io_tree we hold
|
||||
* the tree lock and get the inode lock when setting delalloc. These two things
|
||||
* are unrelated, so make a class for the file_extent_tree so we don't get the
|
||||
* two locking patterns mixed up.
|
||||
*/
|
||||
static struct lock_class_key file_extent_tree_class;
|
||||
|
||||
static const struct inode_operations btrfs_dir_inode_operations;
|
||||
static const struct inode_operations btrfs_symlink_inode_operations;
|
||||
static const struct inode_operations btrfs_special_inode_operations;
|
||||
@ -447,8 +456,8 @@ static inline void btrfs_cleanup_ordered_extents(struct btrfs_inode *inode,
|
||||
* range, then btrfs_mark_ordered_io_finished() will handle
|
||||
* the ordered extent accounting for the range.
|
||||
*/
|
||||
btrfs_page_clamp_clear_ordered(inode->root->fs_info, page,
|
||||
offset, bytes);
|
||||
btrfs_folio_clamp_clear_ordered(inode->root->fs_info,
|
||||
page_folio(page), offset, bytes);
|
||||
put_page(page);
|
||||
}
|
||||
|
||||
@ -1037,7 +1046,7 @@ free_pages:
|
||||
if (pages) {
|
||||
for (i = 0; i < nr_pages; i++) {
|
||||
WARN_ON(pages[i]->mapping);
|
||||
put_page(pages[i]);
|
||||
btrfs_free_compr_page(pages[i]);
|
||||
}
|
||||
kfree(pages);
|
||||
}
|
||||
@ -1052,7 +1061,7 @@ static void free_async_extent_pages(struct async_extent *async_extent)
|
||||
|
||||
for (i = 0; i < async_extent->nr_pages; i++) {
|
||||
WARN_ON(async_extent->pages[i]->mapping);
|
||||
put_page(async_extent->pages[i]);
|
||||
btrfs_free_compr_page(async_extent->pages[i]);
|
||||
}
|
||||
kfree(async_extent->pages);
|
||||
async_extent->nr_pages = 0;
|
||||
@ -2793,7 +2802,7 @@ out_page:
|
||||
PAGE_SIZE, !ret);
|
||||
clear_page_dirty_for_io(page);
|
||||
}
|
||||
btrfs_page_clear_checked(fs_info, page, page_start, PAGE_SIZE);
|
||||
btrfs_folio_clear_checked(fs_info, page_folio(page), page_start, PAGE_SIZE);
|
||||
unlock_page(page);
|
||||
put_page(page);
|
||||
kfree(fixup);
|
||||
@ -2848,7 +2857,7 @@ int btrfs_writepage_cow_fixup(struct page *page)
|
||||
* page->mapping outside of the page lock.
|
||||
*/
|
||||
ihold(inode);
|
||||
btrfs_page_set_checked(fs_info, page, page_offset(page), PAGE_SIZE);
|
||||
btrfs_folio_set_checked(fs_info, page_folio(page), page_offset(page), PAGE_SIZE);
|
||||
get_page(page);
|
||||
btrfs_init_work(&fixup->work, btrfs_writepage_fixup_worker, NULL);
|
||||
fixup->page = page;
|
||||
@ -3118,7 +3127,7 @@ int btrfs_finish_one_ordered(struct btrfs_ordered_extent *ordered_extent)
|
||||
ordered_extent->disk_num_bytes);
|
||||
}
|
||||
}
|
||||
unpin_extent_cache(&inode->extent_tree, ordered_extent->file_offset,
|
||||
unpin_extent_cache(inode, ordered_extent->file_offset,
|
||||
ordered_extent->num_bytes, trans->transid);
|
||||
if (ret < 0) {
|
||||
btrfs_abort_transaction(trans, ret);
|
||||
@ -3796,7 +3805,7 @@ cache_index:
|
||||
* cache.
|
||||
*
|
||||
* This is required for both inode re-read from disk and delayed inode
|
||||
* in delayed_nodes_tree.
|
||||
* in the delayed_nodes xarray.
|
||||
*/
|
||||
if (BTRFS_I(inode)->last_trans == btrfs_get_fs_generation(fs_info))
|
||||
set_bit(BTRFS_INODE_NEEDS_FULL_SYNC,
|
||||
@ -4725,7 +4734,7 @@ again:
|
||||
/*
|
||||
* We unlock the page after the io is completed and then re-lock it
|
||||
* above. release_folio() could have come in between that and cleared
|
||||
* PagePrivate(), but left the page in the mapping. Set the page mapped
|
||||
* folio private, but left the page in the mapping. Set the page mapped
|
||||
* here to make sure it's properly set for the subpage stuff.
|
||||
*/
|
||||
ret = set_page_extent_mapped(page);
|
||||
@ -4767,9 +4776,10 @@ again:
|
||||
memzero_page(page, (block_start - page_offset(page)) + offset,
|
||||
len);
|
||||
}
|
||||
btrfs_page_clear_checked(fs_info, page, block_start,
|
||||
block_end + 1 - block_start);
|
||||
btrfs_page_set_dirty(fs_info, page, block_start, block_end + 1 - block_start);
|
||||
btrfs_folio_clear_checked(fs_info, page_folio(page), block_start,
|
||||
block_end + 1 - block_start);
|
||||
btrfs_folio_set_dirty(fs_info, page_folio(page), block_start,
|
||||
block_end + 1 - block_start);
|
||||
unlock_extent(io_tree, block_start, block_end, &cached_state);
|
||||
|
||||
if (only_release_metadata)
|
||||
@ -4889,7 +4899,7 @@ int btrfs_cont_expand(struct btrfs_inode *inode, loff_t oldsize, loff_t size)
|
||||
last_byte = ALIGN(last_byte, fs_info->sectorsize);
|
||||
hole_size = last_byte - cur_offset;
|
||||
|
||||
if (!test_bit(EXTENT_FLAG_PREALLOC, &em->flags)) {
|
||||
if (!(em->flags & EXTENT_FLAG_PREALLOC)) {
|
||||
struct extent_map *hole_em;
|
||||
|
||||
err = maybe_insert_hole(inode, cur_offset, hole_size);
|
||||
@ -4917,7 +4927,6 @@ int btrfs_cont_expand(struct btrfs_inode *inode, loff_t oldsize, loff_t size)
|
||||
hole_em->block_len = 0;
|
||||
hole_em->orig_block_len = 0;
|
||||
hole_em->ram_bytes = hole_size;
|
||||
hole_em->compress_type = BTRFS_COMPRESS_NONE;
|
||||
hole_em->generation = btrfs_get_fs_generation(fs_info);
|
||||
|
||||
err = btrfs_replace_extent_map_range(inode, hole_em, true);
|
||||
@ -6216,6 +6225,13 @@ int btrfs_create_new_inode(struct btrfs_trans_handle *trans,
|
||||
BTRFS_I(inode)->generation = trans->transid;
|
||||
inode->i_generation = BTRFS_I(inode)->generation;
|
||||
|
||||
/*
|
||||
* We don't have any capability xattrs set here yet, shortcut any
|
||||
* queries for the xattrs here. If we add them later via the inode
|
||||
* security init path or any other path this flag will be cleared.
|
||||
*/
|
||||
set_bit(BTRFS_INODE_NO_CAP_XATTR, &BTRFS_I(inode)->runtime_flags);
|
||||
|
||||
/*
|
||||
* Subvolumes don't inherit flags from their parent directory.
|
||||
* Originally this was probably by accident, but we probably can't
|
||||
@ -7258,13 +7274,11 @@ static struct extent_map *create_io_em(struct btrfs_inode *inode, u64 start,
|
||||
em->orig_block_len = orig_block_len;
|
||||
em->ram_bytes = ram_bytes;
|
||||
em->generation = -1;
|
||||
set_bit(EXTENT_FLAG_PINNED, &em->flags);
|
||||
if (type == BTRFS_ORDERED_PREALLOC) {
|
||||
set_bit(EXTENT_FLAG_FILLING, &em->flags);
|
||||
} else if (type == BTRFS_ORDERED_COMPRESSED) {
|
||||
set_bit(EXTENT_FLAG_COMPRESSED, &em->flags);
|
||||
em->compress_type = compress_type;
|
||||
}
|
||||
em->flags |= EXTENT_FLAG_PINNED;
|
||||
if (type == BTRFS_ORDERED_PREALLOC)
|
||||
em->flags |= EXTENT_FLAG_FILLING;
|
||||
else if (type == BTRFS_ORDERED_COMPRESSED)
|
||||
extent_map_set_compression(em, compress_type);
|
||||
|
||||
ret = btrfs_replace_extent_map_range(inode, em, true);
|
||||
if (ret) {
|
||||
@ -7304,10 +7318,10 @@ static int btrfs_get_blocks_direct_write(struct extent_map **map,
|
||||
* just use the extent.
|
||||
*
|
||||
*/
|
||||
if (test_bit(EXTENT_FLAG_PREALLOC, &em->flags) ||
|
||||
if ((em->flags & EXTENT_FLAG_PREALLOC) ||
|
||||
((BTRFS_I(inode)->flags & BTRFS_INODE_NODATACOW) &&
|
||||
em->block_start != EXTENT_MAP_HOLE)) {
|
||||
if (test_bit(EXTENT_FLAG_PREALLOC, &em->flags))
|
||||
if (em->flags & EXTENT_FLAG_PREALLOC)
|
||||
type = BTRFS_ORDERED_PREALLOC;
|
||||
else
|
||||
type = BTRFS_ORDERED_NOCOW;
|
||||
@ -7542,7 +7556,7 @@ static int btrfs_dio_iomap_begin(struct inode *inode, loff_t start,
|
||||
* to buffered IO. Don't blame me, this is the price we pay for using
|
||||
* the generic code.
|
||||
*/
|
||||
if (test_bit(EXTENT_FLAG_COMPRESSED, &em->flags) ||
|
||||
if (extent_map_is_compressed(em) ||
|
||||
em->block_start == EXTENT_MAP_INLINE) {
|
||||
free_extent_map(em);
|
||||
/*
|
||||
@ -7638,7 +7652,7 @@ static int btrfs_dio_iomap_begin(struct inode *inode, loff_t start,
|
||||
* that, since we have locked only the parts we are performing I/O in.
|
||||
*/
|
||||
if ((em->block_start == EXTENT_MAP_HOLE) ||
|
||||
(test_bit(EXTENT_FLAG_PREALLOC, &em->flags) && !write)) {
|
||||
((em->flags & EXTENT_FLAG_PREALLOC) && !write)) {
|
||||
iomap->addr = IOMAP_NULL_ADDR;
|
||||
iomap->type = IOMAP_HOLE;
|
||||
} else {
|
||||
@ -7851,13 +7865,14 @@ static void btrfs_readahead(struct readahead_control *rac)
|
||||
static void wait_subpage_spinlock(struct page *page)
|
||||
{
|
||||
struct btrfs_fs_info *fs_info = btrfs_sb(page->mapping->host->i_sb);
|
||||
struct folio *folio = page_folio(page);
|
||||
struct btrfs_subpage *subpage;
|
||||
|
||||
if (!btrfs_is_subpage(fs_info, page))
|
||||
if (!btrfs_is_subpage(fs_info, page->mapping))
|
||||
return;
|
||||
|
||||
ASSERT(PagePrivate(page) && page->private);
|
||||
subpage = (struct btrfs_subpage *)page->private;
|
||||
ASSERT(folio_test_private(folio) && folio_get_private(folio));
|
||||
subpage = folio_get_private(folio);
|
||||
|
||||
/*
|
||||
* This may look insane as we just acquire the spinlock and release it,
|
||||
@ -7995,7 +8010,7 @@ static void btrfs_invalidate_folio(struct folio *folio, size_t offset,
|
||||
page_end);
|
||||
ASSERT(range_end + 1 - cur < U32_MAX);
|
||||
range_len = range_end + 1 - cur;
|
||||
if (!btrfs_page_test_ordered(fs_info, &folio->page, cur, range_len)) {
|
||||
if (!btrfs_folio_test_ordered(fs_info, folio, cur, range_len)) {
|
||||
/*
|
||||
* If Ordered (Private2) is cleared, it means endio has
|
||||
* already been executed for the range.
|
||||
@ -8004,7 +8019,7 @@ static void btrfs_invalidate_folio(struct folio *folio, size_t offset,
|
||||
*/
|
||||
goto next;
|
||||
}
|
||||
btrfs_page_clear_ordered(fs_info, &folio->page, cur, range_len);
|
||||
btrfs_folio_clear_ordered(fs_info, folio, cur, range_len);
|
||||
|
||||
/*
|
||||
* IO on this page will never be started, so we need to account
|
||||
@ -8074,7 +8089,7 @@ next:
|
||||
* did something wrong.
|
||||
*/
|
||||
ASSERT(!folio_test_ordered(folio));
|
||||
btrfs_page_clear_checked(fs_info, &folio->page, folio_pos(folio), folio_size(folio));
|
||||
btrfs_folio_clear_checked(fs_info, folio, folio_pos(folio), folio_size(folio));
|
||||
if (!inode_evicting)
|
||||
__btrfs_release_folio(folio, GFP_NOFS);
|
||||
clear_page_extent_mapped(&folio->page);
|
||||
@ -8098,6 +8113,7 @@ next:
|
||||
vm_fault_t btrfs_page_mkwrite(struct vm_fault *vmf)
|
||||
{
|
||||
struct page *page = vmf->page;
|
||||
struct folio *folio = page_folio(page);
|
||||
struct inode *inode = file_inode(vmf->vma->vm_file);
|
||||
struct btrfs_fs_info *fs_info = btrfs_sb(inode->i_sb);
|
||||
struct extent_io_tree *io_tree = &BTRFS_I(inode)->io_tree;
|
||||
@ -8114,6 +8130,8 @@ vm_fault_t btrfs_page_mkwrite(struct vm_fault *vmf)
|
||||
u64 page_end;
|
||||
u64 end;
|
||||
|
||||
ASSERT(folio_order(folio) == 0);
|
||||
|
||||
reserved_space = PAGE_SIZE;
|
||||
|
||||
sb_start_pagefault(inode->i_sb);
|
||||
@ -8217,9 +8235,9 @@ again:
|
||||
if (zero_start != PAGE_SIZE)
|
||||
memzero_page(page, zero_start, PAGE_SIZE - zero_start);
|
||||
|
||||
btrfs_page_clear_checked(fs_info, page, page_start, PAGE_SIZE);
|
||||
btrfs_page_set_dirty(fs_info, page, page_start, end + 1 - page_start);
|
||||
btrfs_page_set_uptodate(fs_info, page, page_start, end + 1 - page_start);
|
||||
btrfs_folio_clear_checked(fs_info, folio, page_start, PAGE_SIZE);
|
||||
btrfs_folio_set_dirty(fs_info, folio, page_start, end + 1 - page_start);
|
||||
btrfs_folio_set_uptodate(fs_info, folio, page_start, end + 1 - page_start);
|
||||
|
||||
btrfs_set_inode_last_sub_trans(BTRFS_I(inode));
|
||||
|
||||
@ -8462,10 +8480,20 @@ struct inode *btrfs_alloc_inode(struct super_block *sb)
|
||||
struct btrfs_fs_info *fs_info = btrfs_sb(sb);
|
||||
struct btrfs_inode *ei;
|
||||
struct inode *inode;
|
||||
struct extent_io_tree *file_extent_tree = NULL;
|
||||
|
||||
/* Self tests may pass a NULL fs_info. */
|
||||
if (fs_info && !btrfs_fs_incompat(fs_info, NO_HOLES)) {
|
||||
file_extent_tree = kmalloc(sizeof(struct extent_io_tree), GFP_KERNEL);
|
||||
if (!file_extent_tree)
|
||||
return NULL;
|
||||
}
|
||||
|
||||
ei = alloc_inode_sb(sb, btrfs_inode_cachep, GFP_KERNEL);
|
||||
if (!ei)
|
||||
if (!ei) {
|
||||
kfree(file_extent_tree);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
ei->root = NULL;
|
||||
ei->generation = 0;
|
||||
@ -8501,10 +8529,18 @@ struct inode *btrfs_alloc_inode(struct super_block *sb)
|
||||
|
||||
inode = &ei->vfs_inode;
|
||||
extent_map_tree_init(&ei->extent_tree);
|
||||
|
||||
/* This io tree sets the valid inode. */
|
||||
extent_io_tree_init(fs_info, &ei->io_tree, IO_TREE_INODE_IO);
|
||||
ei->io_tree.inode = ei;
|
||||
extent_io_tree_init(fs_info, &ei->file_extent_tree,
|
||||
IO_TREE_INODE_FILE_EXTENT);
|
||||
|
||||
ei->file_extent_tree = file_extent_tree;
|
||||
if (file_extent_tree) {
|
||||
extent_io_tree_init(fs_info, ei->file_extent_tree,
|
||||
IO_TREE_INODE_FILE_EXTENT);
|
||||
/* Lockdep class is set only for the file extent tree. */
|
||||
lockdep_set_class(&ei->file_extent_tree->lock, &file_extent_tree_class);
|
||||
}
|
||||
mutex_init(&ei->log_mutex);
|
||||
spin_lock_init(&ei->ordered_tree_lock);
|
||||
ei->ordered_tree = RB_ROOT;
|
||||
@ -8521,12 +8557,14 @@ struct inode *btrfs_alloc_inode(struct super_block *sb)
|
||||
void btrfs_test_destroy_inode(struct inode *inode)
|
||||
{
|
||||
btrfs_drop_extent_map_range(BTRFS_I(inode), 0, (u64)-1, false);
|
||||
kfree(BTRFS_I(inode)->file_extent_tree);
|
||||
kmem_cache_free(btrfs_inode_cachep, BTRFS_I(inode));
|
||||
}
|
||||
#endif
|
||||
|
||||
void btrfs_free_inode(struct inode *inode)
|
||||
{
|
||||
kfree(BTRFS_I(inode)->file_extent_tree);
|
||||
kmem_cache_free(btrfs_inode_cachep, BTRFS_I(inode));
|
||||
}
|
||||
|
||||
@ -9632,7 +9670,7 @@ static int __btrfs_prealloc_file_range(struct inode *inode, int mode,
|
||||
em->block_len = ins.offset;
|
||||
em->orig_block_len = ins.offset;
|
||||
em->ram_bytes = ins.offset;
|
||||
set_bit(EXTENT_FLAG_PREALLOC, &em->flags);
|
||||
em->flags |= EXTENT_FLAG_PREALLOC;
|
||||
em->generation = trans->transid;
|
||||
|
||||
ret = btrfs_replace_extent_map_range(BTRFS_I(inode), em, true);
|
||||
@ -9785,7 +9823,9 @@ void btrfs_set_range_writeback(struct btrfs_inode *inode, u64 start, u64 end)
|
||||
page = find_get_page(inode->vfs_inode.i_mapping, index);
|
||||
ASSERT(page); /* Pages should be in the extent_io_tree */
|
||||
|
||||
btrfs_page_set_writeback(fs_info, page, start, len);
|
||||
/* This is for data, which doesn't yet support larger folio. */
|
||||
ASSERT(folio_order(page_folio(page)) == 0);
|
||||
btrfs_folio_set_writeback(fs_info, page_folio(page), start, len);
|
||||
put_page(page);
|
||||
index++;
|
||||
}
|
||||
@ -9994,7 +10034,7 @@ static ssize_t btrfs_encoded_read_regular(struct kiocb *iocb,
|
||||
pages = kcalloc(nr_pages, sizeof(struct page *), GFP_NOFS);
|
||||
if (!pages)
|
||||
return -ENOMEM;
|
||||
ret = btrfs_alloc_page_array(nr_pages, pages);
|
||||
ret = btrfs_alloc_page_array(nr_pages, pages, 0);
|
||||
if (ret) {
|
||||
ret = -ENOMEM;
|
||||
goto out;
|
||||
@ -10113,12 +10153,12 @@ ssize_t btrfs_encoded_read(struct kiocb *iocb, struct iov_iter *iter,
|
||||
encoded->len = min_t(u64, extent_map_end(em),
|
||||
inode->vfs_inode.i_size) - iocb->ki_pos;
|
||||
if (em->block_start == EXTENT_MAP_HOLE ||
|
||||
test_bit(EXTENT_FLAG_PREALLOC, &em->flags)) {
|
||||
(em->flags & EXTENT_FLAG_PREALLOC)) {
|
||||
disk_bytenr = EXTENT_MAP_HOLE;
|
||||
count = min_t(u64, count, encoded->len);
|
||||
encoded->len = count;
|
||||
encoded->unencoded_len = count;
|
||||
} else if (test_bit(EXTENT_FLAG_COMPRESSED, &em->flags)) {
|
||||
} else if (extent_map_is_compressed(em)) {
|
||||
disk_bytenr = em->block_start;
|
||||
/*
|
||||
* Bail if the buffer isn't large enough to return the whole
|
||||
@ -10133,7 +10173,7 @@ ssize_t btrfs_encoded_read(struct kiocb *iocb, struct iov_iter *iter,
|
||||
encoded->unencoded_len = em->ram_bytes;
|
||||
encoded->unencoded_offset = iocb->ki_pos - em->orig_start;
|
||||
ret = btrfs_encoded_io_compression_from_extent(fs_info,
|
||||
em->compress_type);
|
||||
extent_map_compression(em));
|
||||
if (ret < 0)
|
||||
goto out_em;
|
||||
encoded->compression = ret;
|
||||
@ -10564,6 +10604,7 @@ static int btrfs_swap_activate(struct swap_info_struct *sis, struct file *file,
|
||||
struct extent_io_tree *io_tree = &BTRFS_I(inode)->io_tree;
|
||||
struct extent_state *cached_state = NULL;
|
||||
struct extent_map *em = NULL;
|
||||
struct btrfs_chunk_map *map = NULL;
|
||||
struct btrfs_device *device = NULL;
|
||||
struct btrfs_swap_info bsi = {
|
||||
.lowest_ppage = (sector_t)-1ULL,
|
||||
@ -10680,7 +10721,7 @@ static int btrfs_swap_activate(struct swap_info_struct *sis, struct file *file,
|
||||
ret = -EINVAL;
|
||||
goto out;
|
||||
}
|
||||
if (test_bit(EXTENT_FLAG_COMPRESSED, &em->flags)) {
|
||||
if (extent_map_is_compressed(em)) {
|
||||
btrfs_warn(fs_info, "swapfile must not be compressed");
|
||||
ret = -EINVAL;
|
||||
goto out;
|
||||
@ -10703,13 +10744,13 @@ static int btrfs_swap_activate(struct swap_info_struct *sis, struct file *file,
|
||||
goto out;
|
||||
}
|
||||
|
||||
em = btrfs_get_chunk_map(fs_info, logical_block_start, len);
|
||||
if (IS_ERR(em)) {
|
||||
ret = PTR_ERR(em);
|
||||
map = btrfs_get_chunk_map(fs_info, logical_block_start, len);
|
||||
if (IS_ERR(map)) {
|
||||
ret = PTR_ERR(map);
|
||||
goto out;
|
||||
}
|
||||
|
||||
if (em->map_lookup->type & BTRFS_BLOCK_GROUP_PROFILE_MASK) {
|
||||
if (map->type & BTRFS_BLOCK_GROUP_PROFILE_MASK) {
|
||||
btrfs_warn(fs_info,
|
||||
"swapfile must have single data profile");
|
||||
ret = -EINVAL;
|
||||
@ -10717,23 +10758,23 @@ static int btrfs_swap_activate(struct swap_info_struct *sis, struct file *file,
|
||||
}
|
||||
|
||||
if (device == NULL) {
|
||||
device = em->map_lookup->stripes[0].dev;
|
||||
device = map->stripes[0].dev;
|
||||
ret = btrfs_add_swapfile_pin(inode, device, false);
|
||||
if (ret == 1)
|
||||
ret = 0;
|
||||
else if (ret)
|
||||
goto out;
|
||||
} else if (device != em->map_lookup->stripes[0].dev) {
|
||||
} else if (device != map->stripes[0].dev) {
|
||||
btrfs_warn(fs_info, "swapfile must be on one device");
|
||||
ret = -EINVAL;
|
||||
goto out;
|
||||
}
|
||||
|
||||
physical_block_start = (em->map_lookup->stripes[0].physical +
|
||||
(logical_block_start - em->start));
|
||||
len = min(len, em->len - (logical_block_start - em->start));
|
||||
free_extent_map(em);
|
||||
em = NULL;
|
||||
physical_block_start = (map->stripes[0].physical +
|
||||
(logical_block_start - map->start));
|
||||
len = min(len, map->chunk_len - (logical_block_start - map->start));
|
||||
btrfs_free_chunk_map(map);
|
||||
map = NULL;
|
||||
|
||||
bg = btrfs_lookup_block_group(fs_info, logical_block_start);
|
||||
if (!bg) {
|
||||
@ -10786,6 +10827,8 @@ static int btrfs_swap_activate(struct swap_info_struct *sis, struct file *file,
|
||||
out:
|
||||
if (!IS_ERR_OR_NULL(em))
|
||||
free_extent_map(em);
|
||||
if (!IS_ERR_OR_NULL(map))
|
||||
btrfs_free_chunk_map(map);
|
||||
|
||||
unlock_extent(io_tree, 0, isize - 1, &cached_state);
|
||||
|
||||
|
@ -9,7 +9,7 @@
|
||||
*
|
||||
* @cache: The cache.
|
||||
* @max_size: Maximum size (number of entries) for the cache.
|
||||
* Use 0 for unlimited size, it's the user's responsability to
|
||||
* Use 0 for unlimited size, it's the user's responsibility to
|
||||
* trim the cache in that case.
|
||||
*/
|
||||
void btrfs_lru_cache_init(struct btrfs_lru_cache *cache, unsigned int max_size)
|
||||
|
@ -152,7 +152,7 @@ static int copy_compressed_data_to_page(char *compressed_data,
|
||||
cur_page = out_pages[*cur_out / PAGE_SIZE];
|
||||
/* Allocate a new page */
|
||||
if (!cur_page) {
|
||||
cur_page = alloc_page(GFP_NOFS);
|
||||
cur_page = btrfs_alloc_compr_page();
|
||||
if (!cur_page)
|
||||
return -ENOMEM;
|
||||
out_pages[*cur_out / PAGE_SIZE] = cur_page;
|
||||
@ -178,7 +178,7 @@ static int copy_compressed_data_to_page(char *compressed_data,
|
||||
cur_page = out_pages[*cur_out / PAGE_SIZE];
|
||||
/* Allocate a new page */
|
||||
if (!cur_page) {
|
||||
cur_page = alloc_page(GFP_NOFS);
|
||||
cur_page = btrfs_alloc_compr_page();
|
||||
if (!cur_page)
|
||||
return -ENOMEM;
|
||||
out_pages[*cur_out / PAGE_SIZE] = cur_page;
|
||||
|
@ -287,7 +287,7 @@ void __cold btrfs_err_32bit_limit(struct btrfs_fs_info *fs_info)
|
||||
* panic or BUGs, depending on mount options.
|
||||
*/
|
||||
__cold
|
||||
void __btrfs_panic(struct btrfs_fs_info *fs_info, const char *function,
|
||||
void __btrfs_panic(const struct btrfs_fs_info *fs_info, const char *function,
|
||||
unsigned int line, int error, const char *fmt, ...)
|
||||
{
|
||||
char *s_id = "<unknown>";
|
||||
|
@ -194,7 +194,7 @@ const char * __attribute_const__ btrfs_decode_error(int error);
|
||||
|
||||
__printf(5, 6)
|
||||
__cold
|
||||
void __btrfs_panic(struct btrfs_fs_info *fs_info, const char *function,
|
||||
void __btrfs_panic(const struct btrfs_fs_info *fs_info, const char *function,
|
||||
unsigned int line, int error, const char *fmt, ...);
|
||||
/*
|
||||
* If BTRFS_MOUNT_PANIC_ON_FATAL_ERROR is in mount_opt, __btrfs_panic
|
||||
|
@ -323,9 +323,10 @@ static bool can_finish_ordered_extent(struct btrfs_ordered_extent *ordered,
|
||||
*
|
||||
* If there's no such bit, we need to skip to next range.
|
||||
*/
|
||||
if (!btrfs_page_test_ordered(fs_info, page, file_offset, len))
|
||||
if (!btrfs_folio_test_ordered(fs_info, page_folio(page),
|
||||
file_offset, len))
|
||||
return false;
|
||||
btrfs_page_clear_ordered(fs_info, page, file_offset, len);
|
||||
btrfs_folio_clear_ordered(fs_info, page_folio(page), file_offset, len);
|
||||
}
|
||||
|
||||
/* Now we're fine to update the accounting. */
|
||||
|
@ -96,13 +96,6 @@ struct btrfs_ordered_extent {
|
||||
/* number of bytes that still need writing */
|
||||
u64 bytes_left;
|
||||
|
||||
/*
|
||||
* the end of the ordered extent which is behind it but
|
||||
* didn't update disk_i_size. Please see the comment of
|
||||
* btrfs_ordered_update_i_size();
|
||||
*/
|
||||
u64 outstanding_isize;
|
||||
|
||||
/*
|
||||
* If we get truncated we need to adjust the file extent we enter for
|
||||
* this ordered extent so that we do not expose stale data.
|
||||
|
@ -194,7 +194,7 @@ static struct btrfs_qgroup *find_qgroup_rb(struct btrfs_fs_info *fs_info,
|
||||
*
|
||||
* Must be called with qgroup_lock held and @prealloc preallocated.
|
||||
*
|
||||
* The control on the lifespan of @prealloc would be transfered to this
|
||||
* The control on the lifespan of @prealloc would be transferred to this
|
||||
* function, thus caller should no longer touch @prealloc.
|
||||
*/
|
||||
static struct btrfs_qgroup *add_qgroup_rb(struct btrfs_fs_info *fs_info,
|
||||
|
@ -964,7 +964,7 @@ static int alloc_rbio_pages(struct btrfs_raid_bio *rbio)
|
||||
{
|
||||
int ret;
|
||||
|
||||
ret = btrfs_alloc_page_array(rbio->nr_pages, rbio->stripe_pages);
|
||||
ret = btrfs_alloc_page_array(rbio->nr_pages, rbio->stripe_pages, 0);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
/* Mapping all sectors */
|
||||
@ -979,7 +979,7 @@ static int alloc_rbio_parity_pages(struct btrfs_raid_bio *rbio)
|
||||
int ret;
|
||||
|
||||
ret = btrfs_alloc_page_array(rbio->nr_pages - data_pages,
|
||||
rbio->stripe_pages + data_pages);
|
||||
rbio->stripe_pages + data_pages, 0);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
@ -1530,7 +1530,7 @@ static int alloc_rbio_data_pages(struct btrfs_raid_bio *rbio)
|
||||
const int data_pages = rbio->nr_data * rbio->stripe_npages;
|
||||
int ret;
|
||||
|
||||
ret = btrfs_alloc_page_array(data_pages, rbio->stripe_pages);
|
||||
ret = btrfs_alloc_page_array(data_pages, rbio->stripe_pages, 0);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
@ -1549,7 +1549,6 @@ struct btrfs_plug_cb {
|
||||
struct blk_plug_cb cb;
|
||||
struct btrfs_fs_info *info;
|
||||
struct list_head rbio_list;
|
||||
struct work_struct work;
|
||||
};
|
||||
|
||||
/*
|
||||
|
@ -164,7 +164,7 @@ struct raid56_bio_trace_info {
|
||||
u8 stripe_nr;
|
||||
};
|
||||
|
||||
static inline int nr_data_stripes(const struct map_lookup *map)
|
||||
static inline int nr_data_stripes(const struct btrfs_chunk_map *map)
|
||||
{
|
||||
return map->num_stripes - btrfs_nr_parity_stripes(map->type);
|
||||
}
|
||||
|
@ -141,9 +141,9 @@ static int copy_inline_to_page(struct btrfs_inode *inode,
|
||||
if (datal < block_size)
|
||||
memzero_page(page, datal, block_size - datal);
|
||||
|
||||
btrfs_page_set_uptodate(fs_info, page, file_offset, block_size);
|
||||
btrfs_page_clear_checked(fs_info, page, file_offset, block_size);
|
||||
btrfs_page_set_dirty(fs_info, page, file_offset, block_size);
|
||||
btrfs_folio_set_uptodate(fs_info, page_folio(page), file_offset, block_size);
|
||||
btrfs_folio_clear_checked(fs_info, page_folio(page), file_offset, block_size);
|
||||
btrfs_folio_set_dirty(fs_info, page_folio(page), file_offset, block_size);
|
||||
out_unlock:
|
||||
if (page) {
|
||||
unlock_page(page);
|
||||
|
@ -2895,7 +2895,7 @@ static noinline_for_stack int prealloc_file_extent_cluster(
|
||||
* will re-read the whole page anyway.
|
||||
*/
|
||||
if (page) {
|
||||
btrfs_subpage_clear_uptodate(fs_info, page, i_size,
|
||||
btrfs_subpage_clear_uptodate(fs_info, page_folio(page), i_size,
|
||||
round_up(i_size, PAGE_SIZE) - i_size);
|
||||
unlock_page(page);
|
||||
put_page(page);
|
||||
@ -2951,7 +2951,7 @@ static noinline_for_stack int setup_relocation_extent_mapping(struct inode *inod
|
||||
em->len = end + 1 - start;
|
||||
em->block_len = em->len;
|
||||
em->block_start = block_start;
|
||||
set_bit(EXTENT_FLAG_PINNED, &em->flags);
|
||||
em->flags |= EXTENT_FLAG_PINNED;
|
||||
|
||||
lock_extent(&BTRFS_I(inode)->io_tree, start, end, &cached_state);
|
||||
ret = btrfs_replace_extent_map_range(BTRFS_I(inode), em, false);
|
||||
@ -3070,7 +3070,8 @@ static int relocate_one_page(struct inode *inode, struct file_ra_state *ra,
|
||||
clamped_len);
|
||||
goto release_page;
|
||||
}
|
||||
btrfs_page_set_dirty(fs_info, page, clamped_start, clamped_len);
|
||||
btrfs_folio_set_dirty(fs_info, page_folio(page),
|
||||
clamped_start, clamped_len);
|
||||
|
||||
/*
|
||||
* Set the boundary if it's inside the page.
|
||||
|
@ -43,7 +43,7 @@ struct scrub_ctx;
|
||||
/*
|
||||
* The following value only influences the performance.
|
||||
*
|
||||
* This detemines how many stripes would be submitted in one go,
|
||||
* This determines how many stripes would be submitted in one go,
|
||||
* which is 512KiB (BTRFS_STRIPE_LEN * SCRUB_STRIPES_PER_GROUP).
|
||||
*/
|
||||
#define SCRUB_STRIPES_PER_GROUP 8
|
||||
@ -192,7 +192,6 @@ struct scrub_ctx {
|
||||
int cur_stripe;
|
||||
atomic_t cancel_req;
|
||||
int readonly;
|
||||
int sectors_per_bio;
|
||||
|
||||
/* State of IO submission throttling affecting the associated device */
|
||||
ktime_t throttle_deadline;
|
||||
@ -262,7 +261,7 @@ static int init_scrub_stripe(struct btrfs_fs_info *fs_info,
|
||||
atomic_set(&stripe->pending_io, 0);
|
||||
spin_lock_init(&stripe->write_error_lock);
|
||||
|
||||
ret = btrfs_alloc_page_array(SCRUB_STRIPE_PAGES, stripe->pages);
|
||||
ret = btrfs_alloc_page_array(SCRUB_STRIPE_PAGES, stripe->pages, 0);
|
||||
if (ret < 0)
|
||||
goto error;
|
||||
|
||||
@ -710,7 +709,7 @@ static void scrub_verify_one_sector(struct scrub_stripe *stripe, int sector_nr)
|
||||
/* Metadata, verify the full tree block. */
|
||||
if (sector->is_metadata) {
|
||||
/*
|
||||
* Check if the tree block crosses the stripe boudary. If
|
||||
* Check if the tree block crosses the stripe boundary. If
|
||||
* crossed the boundary, we cannot verify it but only give a
|
||||
* warning.
|
||||
*
|
||||
@ -884,7 +883,7 @@ static void scrub_stripe_report_errors(struct scrub_ctx *sctx,
|
||||
/*
|
||||
* Init needed infos for error reporting.
|
||||
*
|
||||
* Although our scrub_stripe infrastucture is mostly based on btrfs_submit_bio()
|
||||
* Although our scrub_stripe infrastructure is mostly based on btrfs_submit_bio()
|
||||
* thus no need for dev/physical, error reporting still needs dev and physical.
|
||||
*/
|
||||
if (!bitmap_empty(&stripe->init_error_bitmap, stripe->nr_sectors)) {
|
||||
@ -1280,7 +1279,7 @@ static void scrub_throttle_dev_io(struct scrub_ctx *sctx, struct btrfs_device *d
|
||||
* return 0 if it is a data stripe, 1 means parity stripe.
|
||||
*/
|
||||
static int get_raid56_logic_offset(u64 physical, int num,
|
||||
struct map_lookup *map, u64 *offset,
|
||||
struct btrfs_chunk_map *map, u64 *offset,
|
||||
u64 *stripe_start)
|
||||
{
|
||||
int i;
|
||||
@ -1409,14 +1408,11 @@ search_forward:
|
||||
if (ret > 0)
|
||||
break;
|
||||
next:
|
||||
path->slots[0]++;
|
||||
if (path->slots[0] >= btrfs_header_nritems(path->nodes[0])) {
|
||||
ret = btrfs_next_leaf(extent_root, path);
|
||||
if (ret) {
|
||||
/* Either no more item or fatal error */
|
||||
btrfs_release_path(path);
|
||||
return ret;
|
||||
}
|
||||
ret = btrfs_next_item(extent_root, path);
|
||||
if (ret) {
|
||||
/* Either no more items or a fatal error. */
|
||||
btrfs_release_path(path);
|
||||
return ret;
|
||||
}
|
||||
}
|
||||
btrfs_release_path(path);
|
||||
@ -1816,7 +1812,7 @@ static int flush_scrub_stripes(struct scrub_ctx *sctx)
|
||||
if (sctx->is_dev_replace) {
|
||||
/*
|
||||
* For dev-replace, if we know there is something wrong with
|
||||
* metadata, we should immedately abort.
|
||||
* metadata, we should immediately abort.
|
||||
*/
|
||||
for (int i = 0; i < nr_stripes; i++) {
|
||||
if (stripe_has_metadata_error(&sctx->stripes[i])) {
|
||||
@ -1898,7 +1894,7 @@ static int queue_scrub_stripe(struct scrub_ctx *sctx, struct btrfs_block_group *
|
||||
static int scrub_raid56_parity_stripe(struct scrub_ctx *sctx,
|
||||
struct btrfs_device *scrub_dev,
|
||||
struct btrfs_block_group *bg,
|
||||
struct map_lookup *map,
|
||||
struct btrfs_chunk_map *map,
|
||||
u64 full_stripe_start)
|
||||
{
|
||||
DECLARE_COMPLETION_ONSTACK(io_done);
|
||||
@ -2067,7 +2063,7 @@ out:
|
||||
*/
|
||||
static int scrub_simple_mirror(struct scrub_ctx *sctx,
|
||||
struct btrfs_block_group *bg,
|
||||
struct map_lookup *map,
|
||||
struct btrfs_chunk_map *map,
|
||||
u64 logical_start, u64 logical_length,
|
||||
struct btrfs_device *device,
|
||||
u64 physical, int mirror_num)
|
||||
@ -2128,7 +2124,7 @@ static int scrub_simple_mirror(struct scrub_ctx *sctx,
|
||||
}
|
||||
|
||||
/* Calculate the full stripe length for simple stripe based profiles */
|
||||
static u64 simple_stripe_full_stripe_len(const struct map_lookup *map)
|
||||
static u64 simple_stripe_full_stripe_len(const struct btrfs_chunk_map *map)
|
||||
{
|
||||
ASSERT(map->type & (BTRFS_BLOCK_GROUP_RAID0 |
|
||||
BTRFS_BLOCK_GROUP_RAID10));
|
||||
@ -2137,7 +2133,7 @@ static u64 simple_stripe_full_stripe_len(const struct map_lookup *map)
|
||||
}
|
||||
|
||||
/* Get the logical bytenr for the stripe */
|
||||
static u64 simple_stripe_get_logical(struct map_lookup *map,
|
||||
static u64 simple_stripe_get_logical(struct btrfs_chunk_map *map,
|
||||
struct btrfs_block_group *bg,
|
||||
int stripe_index)
|
||||
{
|
||||
@ -2154,7 +2150,7 @@ static u64 simple_stripe_get_logical(struct map_lookup *map,
|
||||
}
|
||||
|
||||
/* Get the mirror number for the stripe */
|
||||
static int simple_stripe_mirror_num(struct map_lookup *map, int stripe_index)
|
||||
static int simple_stripe_mirror_num(struct btrfs_chunk_map *map, int stripe_index)
|
||||
{
|
||||
ASSERT(map->type & (BTRFS_BLOCK_GROUP_RAID0 |
|
||||
BTRFS_BLOCK_GROUP_RAID10));
|
||||
@ -2166,7 +2162,7 @@ static int simple_stripe_mirror_num(struct map_lookup *map, int stripe_index)
|
||||
|
||||
static int scrub_simple_stripe(struct scrub_ctx *sctx,
|
||||
struct btrfs_block_group *bg,
|
||||
struct map_lookup *map,
|
||||
struct btrfs_chunk_map *map,
|
||||
struct btrfs_device *device,
|
||||
int stripe_index)
|
||||
{
|
||||
@ -2199,18 +2195,17 @@ static int scrub_simple_stripe(struct scrub_ctx *sctx,
|
||||
|
||||
static noinline_for_stack int scrub_stripe(struct scrub_ctx *sctx,
|
||||
struct btrfs_block_group *bg,
|
||||
struct extent_map *em,
|
||||
struct btrfs_chunk_map *map,
|
||||
struct btrfs_device *scrub_dev,
|
||||
int stripe_index)
|
||||
{
|
||||
struct btrfs_fs_info *fs_info = sctx->fs_info;
|
||||
struct map_lookup *map = em->map_lookup;
|
||||
const u64 profile = map->type & BTRFS_BLOCK_GROUP_PROFILE_MASK;
|
||||
const u64 chunk_logical = bg->start;
|
||||
int ret;
|
||||
int ret2;
|
||||
u64 physical = map->stripes[stripe_index].physical;
|
||||
const u64 dev_stripe_len = btrfs_calc_stripe_length(em);
|
||||
const u64 dev_stripe_len = btrfs_calc_stripe_length(map);
|
||||
const u64 physical_end = physical + dev_stripe_len;
|
||||
u64 logical;
|
||||
u64 logic_end;
|
||||
@ -2373,17 +2368,12 @@ static noinline_for_stack int scrub_chunk(struct scrub_ctx *sctx,
|
||||
u64 dev_extent_len)
|
||||
{
|
||||
struct btrfs_fs_info *fs_info = sctx->fs_info;
|
||||
struct extent_map_tree *map_tree = &fs_info->mapping_tree;
|
||||
struct map_lookup *map;
|
||||
struct extent_map *em;
|
||||
struct btrfs_chunk_map *map;
|
||||
int i;
|
||||
int ret = 0;
|
||||
|
||||
read_lock(&map_tree->lock);
|
||||
em = lookup_extent_mapping(map_tree, bg->start, bg->length);
|
||||
read_unlock(&map_tree->lock);
|
||||
|
||||
if (!em) {
|
||||
map = btrfs_find_chunk_map(fs_info, bg->start, bg->length);
|
||||
if (!map) {
|
||||
/*
|
||||
* Might have been an unused block group deleted by the cleaner
|
||||
* kthread or relocation.
|
||||
@ -2395,22 +2385,21 @@ static noinline_for_stack int scrub_chunk(struct scrub_ctx *sctx,
|
||||
|
||||
return ret;
|
||||
}
|
||||
if (em->start != bg->start)
|
||||
if (map->start != bg->start)
|
||||
goto out;
|
||||
if (em->len < dev_extent_len)
|
||||
if (map->chunk_len < dev_extent_len)
|
||||
goto out;
|
||||
|
||||
map = em->map_lookup;
|
||||
for (i = 0; i < map->num_stripes; ++i) {
|
||||
if (map->stripes[i].dev->bdev == scrub_dev->bdev &&
|
||||
map->stripes[i].physical == dev_offset) {
|
||||
ret = scrub_stripe(sctx, bg, em, scrub_dev, i);
|
||||
ret = scrub_stripe(sctx, bg, map, scrub_dev, i);
|
||||
if (ret)
|
||||
goto out;
|
||||
}
|
||||
}
|
||||
out:
|
||||
free_extent_map(em);
|
||||
btrfs_free_chunk_map(map);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
@ -64,7 +64,7 @@
|
||||
* This means a slightly higher tree locking latency.
|
||||
*/
|
||||
|
||||
bool btrfs_is_subpage(const struct btrfs_fs_info *fs_info, struct page *page)
|
||||
bool btrfs_is_subpage(const struct btrfs_fs_info *fs_info, struct address_space *mapping)
|
||||
{
|
||||
if (fs_info->sectorsize >= PAGE_SIZE)
|
||||
return false;
|
||||
@ -74,8 +74,7 @@ bool btrfs_is_subpage(const struct btrfs_fs_info *fs_info, struct page *page)
|
||||
* mapping. And if page->mapping->host is data inode, it's subpage.
|
||||
* As we have ruled our sectorsize >= PAGE_SIZE case already.
|
||||
*/
|
||||
if (!page->mapping || !page->mapping->host ||
|
||||
is_data_inode(page->mapping->host))
|
||||
if (!mapping || !mapping->host || is_data_inode(mapping->host))
|
||||
return true;
|
||||
|
||||
/*
|
||||
@ -116,7 +115,7 @@ void btrfs_init_subpage_info(struct btrfs_subpage_info *subpage_info, u32 sector
|
||||
}
|
||||
|
||||
int btrfs_attach_subpage(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page, enum btrfs_subpage_type type)
|
||||
struct folio *folio, enum btrfs_subpage_type type)
|
||||
{
|
||||
struct btrfs_subpage *subpage;
|
||||
|
||||
@ -124,31 +123,30 @@ int btrfs_attach_subpage(const struct btrfs_fs_info *fs_info,
|
||||
* We have cases like a dummy extent buffer page, which is not mapped
|
||||
* and doesn't need to be locked.
|
||||
*/
|
||||
if (page->mapping)
|
||||
ASSERT(PageLocked(page));
|
||||
if (folio->mapping)
|
||||
ASSERT(folio_test_locked(folio));
|
||||
|
||||
/* Either not subpage, or the page already has private attached */
|
||||
if (!btrfs_is_subpage(fs_info, page) || PagePrivate(page))
|
||||
/* Either not subpage, or the folio already has private attached. */
|
||||
if (!btrfs_is_subpage(fs_info, folio->mapping) || folio_test_private(folio))
|
||||
return 0;
|
||||
|
||||
subpage = btrfs_alloc_subpage(fs_info, type);
|
||||
if (IS_ERR(subpage))
|
||||
return PTR_ERR(subpage);
|
||||
|
||||
attach_page_private(page, subpage);
|
||||
folio_attach_private(folio, subpage);
|
||||
return 0;
|
||||
}
|
||||
|
||||
void btrfs_detach_subpage(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page)
|
||||
void btrfs_detach_subpage(const struct btrfs_fs_info *fs_info, struct folio *folio)
|
||||
{
|
||||
struct btrfs_subpage *subpage;
|
||||
|
||||
/* Either not subpage, or already detached */
|
||||
if (!btrfs_is_subpage(fs_info, page) || !PagePrivate(page))
|
||||
/* Either not subpage, or the folio already has private attached. */
|
||||
if (!btrfs_is_subpage(fs_info, folio->mapping) || !folio_test_private(folio))
|
||||
return;
|
||||
|
||||
subpage = detach_page_private(page);
|
||||
subpage = folio_detach_private(folio);
|
||||
ASSERT(subpage);
|
||||
btrfs_free_subpage(subpage);
|
||||
}
|
||||
@ -188,77 +186,78 @@ void btrfs_free_subpage(struct btrfs_subpage *subpage)
|
||||
* This is important for eb allocation, to prevent race with last eb freeing
|
||||
* of the same page.
|
||||
* With the eb_refs increased before the eb inserted into radix tree,
|
||||
* detach_extent_buffer_page() won't detach the page private while we're still
|
||||
* detach_extent_buffer_page() won't detach the folio private while we're still
|
||||
* allocating the extent buffer.
|
||||
*/
|
||||
void btrfs_page_inc_eb_refs(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page)
|
||||
void btrfs_folio_inc_eb_refs(const struct btrfs_fs_info *fs_info, struct folio *folio)
|
||||
{
|
||||
struct btrfs_subpage *subpage;
|
||||
|
||||
if (!btrfs_is_subpage(fs_info, page))
|
||||
if (!btrfs_is_subpage(fs_info, folio->mapping))
|
||||
return;
|
||||
|
||||
ASSERT(PagePrivate(page) && page->mapping);
|
||||
lockdep_assert_held(&page->mapping->i_private_lock);
|
||||
ASSERT(folio_test_private(folio) && folio->mapping);
|
||||
lockdep_assert_held(&folio->mapping->i_private_lock);
|
||||
|
||||
subpage = (struct btrfs_subpage *)page->private;
|
||||
subpage = folio_get_private(folio);
|
||||
atomic_inc(&subpage->eb_refs);
|
||||
}
|
||||
|
||||
void btrfs_page_dec_eb_refs(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page)
|
||||
void btrfs_folio_dec_eb_refs(const struct btrfs_fs_info *fs_info, struct folio *folio)
|
||||
{
|
||||
struct btrfs_subpage *subpage;
|
||||
|
||||
if (!btrfs_is_subpage(fs_info, page))
|
||||
if (!btrfs_is_subpage(fs_info, folio->mapping))
|
||||
return;
|
||||
|
||||
ASSERT(PagePrivate(page) && page->mapping);
|
||||
lockdep_assert_held(&page->mapping->i_private_lock);
|
||||
ASSERT(folio_test_private(folio) && folio->mapping);
|
||||
lockdep_assert_held(&folio->mapping->i_private_lock);
|
||||
|
||||
subpage = (struct btrfs_subpage *)page->private;
|
||||
subpage = folio_get_private(folio);
|
||||
ASSERT(atomic_read(&subpage->eb_refs));
|
||||
atomic_dec(&subpage->eb_refs);
|
||||
}
|
||||
|
||||
static void btrfs_subpage_assert(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page, u64 start, u32 len)
|
||||
struct folio *folio, u64 start, u32 len)
|
||||
{
|
||||
/* For subpage support, the folio must be single page. */
|
||||
ASSERT(folio_order(folio) == 0);
|
||||
|
||||
/* Basic checks */
|
||||
ASSERT(PagePrivate(page) && page->private);
|
||||
ASSERT(folio_test_private(folio) && folio_get_private(folio));
|
||||
ASSERT(IS_ALIGNED(start, fs_info->sectorsize) &&
|
||||
IS_ALIGNED(len, fs_info->sectorsize));
|
||||
/*
|
||||
* The range check only works for mapped page, we can still have
|
||||
* unmapped page like dummy extent buffer pages.
|
||||
*/
|
||||
if (page->mapping)
|
||||
ASSERT(page_offset(page) <= start &&
|
||||
start + len <= page_offset(page) + PAGE_SIZE);
|
||||
if (folio->mapping)
|
||||
ASSERT(folio_pos(folio) <= start &&
|
||||
start + len <= folio_pos(folio) + PAGE_SIZE);
|
||||
}
|
||||
|
||||
void btrfs_subpage_start_reader(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page, u64 start, u32 len)
|
||||
struct folio *folio, u64 start, u32 len)
|
||||
{
|
||||
struct btrfs_subpage *subpage = (struct btrfs_subpage *)page->private;
|
||||
struct btrfs_subpage *subpage = folio_get_private(folio);
|
||||
const int nbits = len >> fs_info->sectorsize_bits;
|
||||
|
||||
btrfs_subpage_assert(fs_info, page, start, len);
|
||||
btrfs_subpage_assert(fs_info, folio, start, len);
|
||||
|
||||
atomic_add(nbits, &subpage->readers);
|
||||
}
|
||||
|
||||
void btrfs_subpage_end_reader(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page, u64 start, u32 len)
|
||||
struct folio *folio, u64 start, u32 len)
|
||||
{
|
||||
struct btrfs_subpage *subpage = (struct btrfs_subpage *)page->private;
|
||||
struct btrfs_subpage *subpage = folio_get_private(folio);
|
||||
const int nbits = len >> fs_info->sectorsize_bits;
|
||||
bool is_data;
|
||||
bool last;
|
||||
|
||||
btrfs_subpage_assert(fs_info, page, start, len);
|
||||
is_data = is_data_inode(page->mapping->host);
|
||||
btrfs_subpage_assert(fs_info, folio, start, len);
|
||||
is_data = is_data_inode(folio->mapping->host);
|
||||
ASSERT(atomic_read(&subpage->readers) >= nbits);
|
||||
last = atomic_sub_and_test(nbits, &subpage->readers);
|
||||
|
||||
@ -270,35 +269,35 @@ void btrfs_subpage_end_reader(const struct btrfs_fs_info *fs_info,
|
||||
* As we want the atomic_sub_and_test() to be always executed.
|
||||
*/
|
||||
if (is_data && last)
|
||||
unlock_page(page);
|
||||
folio_unlock(folio);
|
||||
}
|
||||
|
||||
static void btrfs_subpage_clamp_range(struct page *page, u64 *start, u32 *len)
|
||||
static void btrfs_subpage_clamp_range(struct folio *folio, u64 *start, u32 *len)
|
||||
{
|
||||
u64 orig_start = *start;
|
||||
u32 orig_len = *len;
|
||||
|
||||
*start = max_t(u64, page_offset(page), orig_start);
|
||||
*start = max_t(u64, folio_pos(folio), orig_start);
|
||||
/*
|
||||
* For certain call sites like btrfs_drop_pages(), we may have pages
|
||||
* beyond the target range. In that case, just set @len to 0, subpage
|
||||
* helpers can handle @len == 0 without any problem.
|
||||
*/
|
||||
if (page_offset(page) >= orig_start + orig_len)
|
||||
if (folio_pos(folio) >= orig_start + orig_len)
|
||||
*len = 0;
|
||||
else
|
||||
*len = min_t(u64, page_offset(page) + PAGE_SIZE,
|
||||
*len = min_t(u64, folio_pos(folio) + PAGE_SIZE,
|
||||
orig_start + orig_len) - *start;
|
||||
}
|
||||
|
||||
void btrfs_subpage_start_writer(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page, u64 start, u32 len)
|
||||
struct folio *folio, u64 start, u32 len)
|
||||
{
|
||||
struct btrfs_subpage *subpage = (struct btrfs_subpage *)page->private;
|
||||
struct btrfs_subpage *subpage = folio_get_private(folio);
|
||||
const int nbits = (len >> fs_info->sectorsize_bits);
|
||||
int ret;
|
||||
|
||||
btrfs_subpage_assert(fs_info, page, start, len);
|
||||
btrfs_subpage_assert(fs_info, folio, start, len);
|
||||
|
||||
ASSERT(atomic_read(&subpage->readers) == 0);
|
||||
ret = atomic_add_return(nbits, &subpage->writers);
|
||||
@ -306,12 +305,12 @@ void btrfs_subpage_start_writer(const struct btrfs_fs_info *fs_info,
|
||||
}
|
||||
|
||||
bool btrfs_subpage_end_and_test_writer(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page, u64 start, u32 len)
|
||||
struct folio *folio, u64 start, u32 len)
|
||||
{
|
||||
struct btrfs_subpage *subpage = (struct btrfs_subpage *)page->private;
|
||||
struct btrfs_subpage *subpage = folio_get_private(folio);
|
||||
const int nbits = (len >> fs_info->sectorsize_bits);
|
||||
|
||||
btrfs_subpage_assert(fs_info, page, start, len);
|
||||
btrfs_subpage_assert(fs_info, folio, start, len);
|
||||
|
||||
/*
|
||||
* We have call sites passing @lock_page into
|
||||
@ -328,7 +327,7 @@ bool btrfs_subpage_end_and_test_writer(const struct btrfs_fs_info *fs_info,
|
||||
}
|
||||
|
||||
/*
|
||||
* Lock a page for delalloc page writeback.
|
||||
* Lock a folio for delalloc page writeback.
|
||||
*
|
||||
* Return -EAGAIN if the page is not properly initialized.
|
||||
* Return 0 with the page locked, and writer counter updated.
|
||||
@ -337,38 +336,40 @@ bool btrfs_subpage_end_and_test_writer(const struct btrfs_fs_info *fs_info,
|
||||
* it's really the correct page, as the caller is using
|
||||
* filemap_get_folios_contig(), which can race with page invalidating.
|
||||
*/
|
||||
int btrfs_page_start_writer_lock(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page, u64 start, u32 len)
|
||||
int btrfs_folio_start_writer_lock(const struct btrfs_fs_info *fs_info,
|
||||
struct folio *folio, u64 start, u32 len)
|
||||
{
|
||||
if (unlikely(!fs_info) || !btrfs_is_subpage(fs_info, page)) {
|
||||
lock_page(page);
|
||||
if (unlikely(!fs_info) || !btrfs_is_subpage(fs_info, folio->mapping)) {
|
||||
folio_lock(folio);
|
||||
return 0;
|
||||
}
|
||||
lock_page(page);
|
||||
if (!PagePrivate(page) || !page->private) {
|
||||
unlock_page(page);
|
||||
folio_lock(folio);
|
||||
if (!folio_test_private(folio) || !folio_get_private(folio)) {
|
||||
folio_unlock(folio);
|
||||
return -EAGAIN;
|
||||
}
|
||||
btrfs_subpage_clamp_range(page, &start, &len);
|
||||
btrfs_subpage_start_writer(fs_info, page, start, len);
|
||||
btrfs_subpage_clamp_range(folio, &start, &len);
|
||||
btrfs_subpage_start_writer(fs_info, folio, start, len);
|
||||
return 0;
|
||||
}
|
||||
|
||||
void btrfs_page_end_writer_lock(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page, u64 start, u32 len)
|
||||
void btrfs_folio_end_writer_lock(const struct btrfs_fs_info *fs_info,
|
||||
struct folio *folio, u64 start, u32 len)
|
||||
{
|
||||
if (unlikely(!fs_info) || !btrfs_is_subpage(fs_info, page))
|
||||
return unlock_page(page);
|
||||
btrfs_subpage_clamp_range(page, &start, &len);
|
||||
if (btrfs_subpage_end_and_test_writer(fs_info, page, start, len))
|
||||
unlock_page(page);
|
||||
if (unlikely(!fs_info) || !btrfs_is_subpage(fs_info, folio->mapping)) {
|
||||
folio_unlock(folio);
|
||||
return;
|
||||
}
|
||||
btrfs_subpage_clamp_range(folio, &start, &len);
|
||||
if (btrfs_subpage_end_and_test_writer(fs_info, folio, start, len))
|
||||
folio_unlock(folio);
|
||||
}
|
||||
|
||||
#define subpage_calc_start_bit(fs_info, page, name, start, len) \
|
||||
#define subpage_calc_start_bit(fs_info, folio, name, start, len) \
|
||||
({ \
|
||||
unsigned int start_bit; \
|
||||
\
|
||||
btrfs_subpage_assert(fs_info, page, start, len); \
|
||||
btrfs_subpage_assert(fs_info, folio, start, len); \
|
||||
start_bit = offset_in_page(start) >> fs_info->sectorsize_bits; \
|
||||
start_bit += fs_info->subpage_info->name##_offset; \
|
||||
start_bit; \
|
||||
@ -385,46 +386,46 @@ void btrfs_page_end_writer_lock(const struct btrfs_fs_info *fs_info,
|
||||
fs_info->subpage_info->bitmap_nr_bits)
|
||||
|
||||
void btrfs_subpage_set_uptodate(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page, u64 start, u32 len)
|
||||
struct folio *folio, u64 start, u32 len)
|
||||
{
|
||||
struct btrfs_subpage *subpage = (struct btrfs_subpage *)page->private;
|
||||
unsigned int start_bit = subpage_calc_start_bit(fs_info, page,
|
||||
struct btrfs_subpage *subpage = folio_get_private(folio);
|
||||
unsigned int start_bit = subpage_calc_start_bit(fs_info, folio,
|
||||
uptodate, start, len);
|
||||
unsigned long flags;
|
||||
|
||||
spin_lock_irqsave(&subpage->lock, flags);
|
||||
bitmap_set(subpage->bitmaps, start_bit, len >> fs_info->sectorsize_bits);
|
||||
if (subpage_test_bitmap_all_set(fs_info, subpage, uptodate))
|
||||
SetPageUptodate(page);
|
||||
folio_mark_uptodate(folio);
|
||||
spin_unlock_irqrestore(&subpage->lock, flags);
|
||||
}
|
||||
|
||||
void btrfs_subpage_clear_uptodate(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page, u64 start, u32 len)
|
||||
struct folio *folio, u64 start, u32 len)
|
||||
{
|
||||
struct btrfs_subpage *subpage = (struct btrfs_subpage *)page->private;
|
||||
unsigned int start_bit = subpage_calc_start_bit(fs_info, page,
|
||||
struct btrfs_subpage *subpage = folio_get_private(folio);
|
||||
unsigned int start_bit = subpage_calc_start_bit(fs_info, folio,
|
||||
uptodate, start, len);
|
||||
unsigned long flags;
|
||||
|
||||
spin_lock_irqsave(&subpage->lock, flags);
|
||||
bitmap_clear(subpage->bitmaps, start_bit, len >> fs_info->sectorsize_bits);
|
||||
ClearPageUptodate(page);
|
||||
folio_clear_uptodate(folio);
|
||||
spin_unlock_irqrestore(&subpage->lock, flags);
|
||||
}
|
||||
|
||||
void btrfs_subpage_set_dirty(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page, u64 start, u32 len)
|
||||
struct folio *folio, u64 start, u32 len)
|
||||
{
|
||||
struct btrfs_subpage *subpage = (struct btrfs_subpage *)page->private;
|
||||
unsigned int start_bit = subpage_calc_start_bit(fs_info, page,
|
||||
struct btrfs_subpage *subpage = folio_get_private(folio);
|
||||
unsigned int start_bit = subpage_calc_start_bit(fs_info, folio,
|
||||
dirty, start, len);
|
||||
unsigned long flags;
|
||||
|
||||
spin_lock_irqsave(&subpage->lock, flags);
|
||||
bitmap_set(subpage->bitmaps, start_bit, len >> fs_info->sectorsize_bits);
|
||||
spin_unlock_irqrestore(&subpage->lock, flags);
|
||||
set_page_dirty(page);
|
||||
folio_mark_dirty(folio);
|
||||
}
|
||||
|
||||
/*
|
||||
@ -438,10 +439,10 @@ void btrfs_subpage_set_dirty(const struct btrfs_fs_info *fs_info,
|
||||
* extra handling for tree blocks.
|
||||
*/
|
||||
bool btrfs_subpage_clear_and_test_dirty(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page, u64 start, u32 len)
|
||||
struct folio *folio, u64 start, u32 len)
|
||||
{
|
||||
struct btrfs_subpage *subpage = (struct btrfs_subpage *)page->private;
|
||||
unsigned int start_bit = subpage_calc_start_bit(fs_info, page,
|
||||
struct btrfs_subpage *subpage = folio_get_private(folio);
|
||||
unsigned int start_bit = subpage_calc_start_bit(fs_info, folio,
|
||||
dirty, start, len);
|
||||
unsigned long flags;
|
||||
bool last = false;
|
||||
@ -455,101 +456,101 @@ bool btrfs_subpage_clear_and_test_dirty(const struct btrfs_fs_info *fs_info,
|
||||
}
|
||||
|
||||
void btrfs_subpage_clear_dirty(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page, u64 start, u32 len)
|
||||
struct folio *folio, u64 start, u32 len)
|
||||
{
|
||||
bool last;
|
||||
|
||||
last = btrfs_subpage_clear_and_test_dirty(fs_info, page, start, len);
|
||||
last = btrfs_subpage_clear_and_test_dirty(fs_info, folio, start, len);
|
||||
if (last)
|
||||
clear_page_dirty_for_io(page);
|
||||
folio_clear_dirty_for_io(folio);
|
||||
}
|
||||
|
||||
void btrfs_subpage_set_writeback(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page, u64 start, u32 len)
|
||||
struct folio *folio, u64 start, u32 len)
|
||||
{
|
||||
struct btrfs_subpage *subpage = (struct btrfs_subpage *)page->private;
|
||||
unsigned int start_bit = subpage_calc_start_bit(fs_info, page,
|
||||
struct btrfs_subpage *subpage = folio_get_private(folio);
|
||||
unsigned int start_bit = subpage_calc_start_bit(fs_info, folio,
|
||||
writeback, start, len);
|
||||
unsigned long flags;
|
||||
|
||||
spin_lock_irqsave(&subpage->lock, flags);
|
||||
bitmap_set(subpage->bitmaps, start_bit, len >> fs_info->sectorsize_bits);
|
||||
set_page_writeback(page);
|
||||
folio_start_writeback(folio);
|
||||
spin_unlock_irqrestore(&subpage->lock, flags);
|
||||
}
|
||||
|
||||
void btrfs_subpage_clear_writeback(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page, u64 start, u32 len)
|
||||
struct folio *folio, u64 start, u32 len)
|
||||
{
|
||||
struct btrfs_subpage *subpage = (struct btrfs_subpage *)page->private;
|
||||
unsigned int start_bit = subpage_calc_start_bit(fs_info, page,
|
||||
struct btrfs_subpage *subpage = folio_get_private(folio);
|
||||
unsigned int start_bit = subpage_calc_start_bit(fs_info, folio,
|
||||
writeback, start, len);
|
||||
unsigned long flags;
|
||||
|
||||
spin_lock_irqsave(&subpage->lock, flags);
|
||||
bitmap_clear(subpage->bitmaps, start_bit, len >> fs_info->sectorsize_bits);
|
||||
if (subpage_test_bitmap_all_zero(fs_info, subpage, writeback)) {
|
||||
ASSERT(PageWriteback(page));
|
||||
end_page_writeback(page);
|
||||
ASSERT(folio_test_writeback(folio));
|
||||
folio_end_writeback(folio);
|
||||
}
|
||||
spin_unlock_irqrestore(&subpage->lock, flags);
|
||||
}
|
||||
|
||||
void btrfs_subpage_set_ordered(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page, u64 start, u32 len)
|
||||
struct folio *folio, u64 start, u32 len)
|
||||
{
|
||||
struct btrfs_subpage *subpage = (struct btrfs_subpage *)page->private;
|
||||
unsigned int start_bit = subpage_calc_start_bit(fs_info, page,
|
||||
struct btrfs_subpage *subpage = folio_get_private(folio);
|
||||
unsigned int start_bit = subpage_calc_start_bit(fs_info, folio,
|
||||
ordered, start, len);
|
||||
unsigned long flags;
|
||||
|
||||
spin_lock_irqsave(&subpage->lock, flags);
|
||||
bitmap_set(subpage->bitmaps, start_bit, len >> fs_info->sectorsize_bits);
|
||||
SetPageOrdered(page);
|
||||
folio_set_ordered(folio);
|
||||
spin_unlock_irqrestore(&subpage->lock, flags);
|
||||
}
|
||||
|
||||
void btrfs_subpage_clear_ordered(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page, u64 start, u32 len)
|
||||
struct folio *folio, u64 start, u32 len)
|
||||
{
|
||||
struct btrfs_subpage *subpage = (struct btrfs_subpage *)page->private;
|
||||
unsigned int start_bit = subpage_calc_start_bit(fs_info, page,
|
||||
struct btrfs_subpage *subpage = folio_get_private(folio);
|
||||
unsigned int start_bit = subpage_calc_start_bit(fs_info, folio,
|
||||
ordered, start, len);
|
||||
unsigned long flags;
|
||||
|
||||
spin_lock_irqsave(&subpage->lock, flags);
|
||||
bitmap_clear(subpage->bitmaps, start_bit, len >> fs_info->sectorsize_bits);
|
||||
if (subpage_test_bitmap_all_zero(fs_info, subpage, ordered))
|
||||
ClearPageOrdered(page);
|
||||
folio_clear_ordered(folio);
|
||||
spin_unlock_irqrestore(&subpage->lock, flags);
|
||||
}
|
||||
|
||||
void btrfs_subpage_set_checked(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page, u64 start, u32 len)
|
||||
struct folio *folio, u64 start, u32 len)
|
||||
{
|
||||
struct btrfs_subpage *subpage = (struct btrfs_subpage *)page->private;
|
||||
unsigned int start_bit = subpage_calc_start_bit(fs_info, page,
|
||||
struct btrfs_subpage *subpage = folio_get_private(folio);
|
||||
unsigned int start_bit = subpage_calc_start_bit(fs_info, folio,
|
||||
checked, start, len);
|
||||
unsigned long flags;
|
||||
|
||||
spin_lock_irqsave(&subpage->lock, flags);
|
||||
bitmap_set(subpage->bitmaps, start_bit, len >> fs_info->sectorsize_bits);
|
||||
if (subpage_test_bitmap_all_set(fs_info, subpage, checked))
|
||||
SetPageChecked(page);
|
||||
folio_set_checked(folio);
|
||||
spin_unlock_irqrestore(&subpage->lock, flags);
|
||||
}
|
||||
|
||||
void btrfs_subpage_clear_checked(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page, u64 start, u32 len)
|
||||
struct folio *folio, u64 start, u32 len)
|
||||
{
|
||||
struct btrfs_subpage *subpage = (struct btrfs_subpage *)page->private;
|
||||
unsigned int start_bit = subpage_calc_start_bit(fs_info, page,
|
||||
struct btrfs_subpage *subpage = folio_get_private(folio);
|
||||
unsigned int start_bit = subpage_calc_start_bit(fs_info, folio,
|
||||
checked, start, len);
|
||||
unsigned long flags;
|
||||
|
||||
spin_lock_irqsave(&subpage->lock, flags);
|
||||
bitmap_clear(subpage->bitmaps, start_bit, len >> fs_info->sectorsize_bits);
|
||||
ClearPageChecked(page);
|
||||
folio_clear_checked(folio);
|
||||
spin_unlock_irqrestore(&subpage->lock, flags);
|
||||
}
|
||||
|
||||
@ -559,10 +560,10 @@ void btrfs_subpage_clear_checked(const struct btrfs_fs_info *fs_info,
|
||||
*/
|
||||
#define IMPLEMENT_BTRFS_SUBPAGE_TEST_OP(name) \
|
||||
bool btrfs_subpage_test_##name(const struct btrfs_fs_info *fs_info, \
|
||||
struct page *page, u64 start, u32 len) \
|
||||
struct folio *folio, u64 start, u32 len) \
|
||||
{ \
|
||||
struct btrfs_subpage *subpage = (struct btrfs_subpage *)page->private; \
|
||||
unsigned int start_bit = subpage_calc_start_bit(fs_info, page, \
|
||||
struct btrfs_subpage *subpage = folio_get_private(folio); \
|
||||
unsigned int start_bit = subpage_calc_start_bit(fs_info, folio, \
|
||||
name, start, len); \
|
||||
unsigned long flags; \
|
||||
bool ret; \
|
||||
@ -584,88 +585,94 @@ IMPLEMENT_BTRFS_SUBPAGE_TEST_OP(checked);
|
||||
* in. We only test sectorsize == PAGE_SIZE cases so far, thus we can fall
|
||||
* back to regular sectorsize branch.
|
||||
*/
|
||||
#define IMPLEMENT_BTRFS_PAGE_OPS(name, set_page_func, clear_page_func, \
|
||||
test_page_func) \
|
||||
void btrfs_page_set_##name(const struct btrfs_fs_info *fs_info, \
|
||||
struct page *page, u64 start, u32 len) \
|
||||
#define IMPLEMENT_BTRFS_PAGE_OPS(name, folio_set_func, \
|
||||
folio_clear_func, folio_test_func) \
|
||||
void btrfs_folio_set_##name(const struct btrfs_fs_info *fs_info, \
|
||||
struct folio *folio, u64 start, u32 len) \
|
||||
{ \
|
||||
if (unlikely(!fs_info) || !btrfs_is_subpage(fs_info, page)) { \
|
||||
set_page_func(page); \
|
||||
if (unlikely(!fs_info) || \
|
||||
!btrfs_is_subpage(fs_info, folio->mapping)) { \
|
||||
folio_set_func(folio); \
|
||||
return; \
|
||||
} \
|
||||
btrfs_subpage_set_##name(fs_info, page, start, len); \
|
||||
btrfs_subpage_set_##name(fs_info, folio, start, len); \
|
||||
} \
|
||||
void btrfs_page_clear_##name(const struct btrfs_fs_info *fs_info, \
|
||||
struct page *page, u64 start, u32 len) \
|
||||
void btrfs_folio_clear_##name(const struct btrfs_fs_info *fs_info, \
|
||||
struct folio *folio, u64 start, u32 len) \
|
||||
{ \
|
||||
if (unlikely(!fs_info) || !btrfs_is_subpage(fs_info, page)) { \
|
||||
clear_page_func(page); \
|
||||
if (unlikely(!fs_info) || \
|
||||
!btrfs_is_subpage(fs_info, folio->mapping)) { \
|
||||
folio_clear_func(folio); \
|
||||
return; \
|
||||
} \
|
||||
btrfs_subpage_clear_##name(fs_info, page, start, len); \
|
||||
btrfs_subpage_clear_##name(fs_info, folio, start, len); \
|
||||
} \
|
||||
bool btrfs_page_test_##name(const struct btrfs_fs_info *fs_info, \
|
||||
struct page *page, u64 start, u32 len) \
|
||||
bool btrfs_folio_test_##name(const struct btrfs_fs_info *fs_info, \
|
||||
struct folio *folio, u64 start, u32 len) \
|
||||
{ \
|
||||
if (unlikely(!fs_info) || !btrfs_is_subpage(fs_info, page)) \
|
||||
return test_page_func(page); \
|
||||
return btrfs_subpage_test_##name(fs_info, page, start, len); \
|
||||
if (unlikely(!fs_info) || \
|
||||
!btrfs_is_subpage(fs_info, folio->mapping)) \
|
||||
return folio_test_func(folio); \
|
||||
return btrfs_subpage_test_##name(fs_info, folio, start, len); \
|
||||
} \
|
||||
void btrfs_page_clamp_set_##name(const struct btrfs_fs_info *fs_info, \
|
||||
struct page *page, u64 start, u32 len) \
|
||||
void btrfs_folio_clamp_set_##name(const struct btrfs_fs_info *fs_info, \
|
||||
struct folio *folio, u64 start, u32 len) \
|
||||
{ \
|
||||
if (unlikely(!fs_info) || !btrfs_is_subpage(fs_info, page)) { \
|
||||
set_page_func(page); \
|
||||
if (unlikely(!fs_info) || \
|
||||
!btrfs_is_subpage(fs_info, folio->mapping)) { \
|
||||
folio_set_func(folio); \
|
||||
return; \
|
||||
} \
|
||||
btrfs_subpage_clamp_range(page, &start, &len); \
|
||||
btrfs_subpage_set_##name(fs_info, page, start, len); \
|
||||
btrfs_subpage_clamp_range(folio, &start, &len); \
|
||||
btrfs_subpage_set_##name(fs_info, folio, start, len); \
|
||||
} \
|
||||
void btrfs_page_clamp_clear_##name(const struct btrfs_fs_info *fs_info, \
|
||||
struct page *page, u64 start, u32 len) \
|
||||
void btrfs_folio_clamp_clear_##name(const struct btrfs_fs_info *fs_info, \
|
||||
struct folio *folio, u64 start, u32 len) \
|
||||
{ \
|
||||
if (unlikely(!fs_info) || !btrfs_is_subpage(fs_info, page)) { \
|
||||
clear_page_func(page); \
|
||||
if (unlikely(!fs_info) || \
|
||||
!btrfs_is_subpage(fs_info, folio->mapping)) { \
|
||||
folio_clear_func(folio); \
|
||||
return; \
|
||||
} \
|
||||
btrfs_subpage_clamp_range(page, &start, &len); \
|
||||
btrfs_subpage_clear_##name(fs_info, page, start, len); \
|
||||
btrfs_subpage_clamp_range(folio, &start, &len); \
|
||||
btrfs_subpage_clear_##name(fs_info, folio, start, len); \
|
||||
} \
|
||||
bool btrfs_page_clamp_test_##name(const struct btrfs_fs_info *fs_info, \
|
||||
struct page *page, u64 start, u32 len) \
|
||||
bool btrfs_folio_clamp_test_##name(const struct btrfs_fs_info *fs_info, \
|
||||
struct folio *folio, u64 start, u32 len) \
|
||||
{ \
|
||||
if (unlikely(!fs_info) || !btrfs_is_subpage(fs_info, page)) \
|
||||
return test_page_func(page); \
|
||||
btrfs_subpage_clamp_range(page, &start, &len); \
|
||||
return btrfs_subpage_test_##name(fs_info, page, start, len); \
|
||||
if (unlikely(!fs_info) || \
|
||||
!btrfs_is_subpage(fs_info, folio->mapping)) \
|
||||
return folio_test_func(folio); \
|
||||
btrfs_subpage_clamp_range(folio, &start, &len); \
|
||||
return btrfs_subpage_test_##name(fs_info, folio, start, len); \
|
||||
}
|
||||
IMPLEMENT_BTRFS_PAGE_OPS(uptodate, SetPageUptodate, ClearPageUptodate,
|
||||
PageUptodate);
|
||||
IMPLEMENT_BTRFS_PAGE_OPS(dirty, set_page_dirty, clear_page_dirty_for_io,
|
||||
PageDirty);
|
||||
IMPLEMENT_BTRFS_PAGE_OPS(writeback, set_page_writeback, end_page_writeback,
|
||||
PageWriteback);
|
||||
IMPLEMENT_BTRFS_PAGE_OPS(ordered, SetPageOrdered, ClearPageOrdered,
|
||||
PageOrdered);
|
||||
IMPLEMENT_BTRFS_PAGE_OPS(checked, SetPageChecked, ClearPageChecked, PageChecked);
|
||||
IMPLEMENT_BTRFS_PAGE_OPS(uptodate, folio_mark_uptodate, folio_clear_uptodate,
|
||||
folio_test_uptodate);
|
||||
IMPLEMENT_BTRFS_PAGE_OPS(dirty, folio_mark_dirty, folio_clear_dirty_for_io,
|
||||
folio_test_dirty);
|
||||
IMPLEMENT_BTRFS_PAGE_OPS(writeback, folio_start_writeback, folio_end_writeback,
|
||||
folio_test_writeback);
|
||||
IMPLEMENT_BTRFS_PAGE_OPS(ordered, folio_set_ordered, folio_clear_ordered,
|
||||
folio_test_ordered);
|
||||
IMPLEMENT_BTRFS_PAGE_OPS(checked, folio_set_checked, folio_clear_checked,
|
||||
folio_test_checked);
|
||||
|
||||
/*
|
||||
* Make sure not only the page dirty bit is cleared, but also subpage dirty bit
|
||||
* is cleared.
|
||||
*/
|
||||
void btrfs_page_assert_not_dirty(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page)
|
||||
void btrfs_folio_assert_not_dirty(const struct btrfs_fs_info *fs_info, struct folio *folio)
|
||||
{
|
||||
struct btrfs_subpage *subpage = (struct btrfs_subpage *)page->private;
|
||||
struct btrfs_subpage *subpage = folio_get_private(folio);
|
||||
|
||||
if (!IS_ENABLED(CONFIG_BTRFS_ASSERT))
|
||||
return;
|
||||
|
||||
ASSERT(!PageDirty(page));
|
||||
if (!btrfs_is_subpage(fs_info, page))
|
||||
ASSERT(!folio_test_dirty(folio));
|
||||
if (!btrfs_is_subpage(fs_info, folio->mapping))
|
||||
return;
|
||||
|
||||
ASSERT(PagePrivate(page) && page->private);
|
||||
ASSERT(folio_test_private(folio) && folio_get_private(folio));
|
||||
ASSERT(subpage_test_bitmap_all_zero(fs_info, subpage, dirty));
|
||||
}
|
||||
|
||||
@ -684,18 +691,20 @@ void btrfs_page_assert_not_dirty(const struct btrfs_fs_info *fs_info,
|
||||
* extent_write_locked_range().
|
||||
* In this case, we have to call subpage helper to handle the case.
|
||||
*/
|
||||
void btrfs_page_unlock_writer(struct btrfs_fs_info *fs_info, struct page *page,
|
||||
u64 start, u32 len)
|
||||
void btrfs_folio_unlock_writer(struct btrfs_fs_info *fs_info,
|
||||
struct folio *folio, u64 start, u32 len)
|
||||
{
|
||||
struct btrfs_subpage *subpage;
|
||||
|
||||
ASSERT(PageLocked(page));
|
||||
ASSERT(folio_test_locked(folio));
|
||||
/* For non-subpage case, we just unlock the page */
|
||||
if (!btrfs_is_subpage(fs_info, page))
|
||||
return unlock_page(page);
|
||||
if (!btrfs_is_subpage(fs_info, folio->mapping)) {
|
||||
folio_unlock(folio);
|
||||
return;
|
||||
}
|
||||
|
||||
ASSERT(PagePrivate(page) && page->private);
|
||||
subpage = (struct btrfs_subpage *)page->private;
|
||||
ASSERT(folio_test_private(folio) && folio_get_private(folio));
|
||||
subpage = folio_get_private(folio);
|
||||
|
||||
/*
|
||||
* For subpage case, there are two types of locked page. With or
|
||||
@ -704,12 +713,14 @@ void btrfs_page_unlock_writer(struct btrfs_fs_info *fs_info, struct page *page,
|
||||
* Since we own the page lock, no one else could touch subpage::writers
|
||||
* and we are safe to do several atomic operations without spinlock.
|
||||
*/
|
||||
if (atomic_read(&subpage->writers) == 0)
|
||||
if (atomic_read(&subpage->writers) == 0) {
|
||||
/* No writers, locked by plain lock_page() */
|
||||
return unlock_page(page);
|
||||
folio_unlock(folio);
|
||||
return;
|
||||
}
|
||||
|
||||
/* Have writers, use proper subpage helper to end it */
|
||||
btrfs_page_end_writer_lock(fs_info, page, start, len);
|
||||
btrfs_folio_end_writer_lock(fs_info, folio, start, len);
|
||||
}
|
||||
|
||||
#define GET_SUBPAGE_BITMAP(subpage, subpage_info, name, dst) \
|
||||
@ -717,7 +728,7 @@ void btrfs_page_unlock_writer(struct btrfs_fs_info *fs_info, struct page *page,
|
||||
subpage_info->name##_offset, subpage_info->bitmap_nr_bits)
|
||||
|
||||
void __cold btrfs_subpage_dump_bitmap(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page, u64 start, u32 len)
|
||||
struct folio *folio, u64 start, u32 len)
|
||||
{
|
||||
struct btrfs_subpage_info *subpage_info = fs_info->subpage_info;
|
||||
struct btrfs_subpage *subpage;
|
||||
@ -729,9 +740,9 @@ void __cold btrfs_subpage_dump_bitmap(const struct btrfs_fs_info *fs_info,
|
||||
unsigned long checked_bitmap;
|
||||
unsigned long flags;
|
||||
|
||||
ASSERT(PagePrivate(page) && page->private);
|
||||
ASSERT(folio_test_private(folio) && folio_get_private(folio));
|
||||
ASSERT(subpage_info);
|
||||
subpage = (struct btrfs_subpage *)page->private;
|
||||
subpage = folio_get_private(folio);
|
||||
|
||||
spin_lock_irqsave(&subpage->lock, flags);
|
||||
GET_SUBPAGE_BITMAP(subpage, subpage_info, uptodate, &uptodate_bitmap);
|
||||
@ -741,10 +752,10 @@ void __cold btrfs_subpage_dump_bitmap(const struct btrfs_fs_info *fs_info,
|
||||
GET_SUBPAGE_BITMAP(subpage, subpage_info, checked, &checked_bitmap);
|
||||
spin_unlock_irqrestore(&subpage->lock, flags);
|
||||
|
||||
dump_page(page, "btrfs subpage dump");
|
||||
dump_page(folio_page(folio, 0), "btrfs subpage dump");
|
||||
btrfs_warn(fs_info,
|
||||
"start=%llu len=%u page=%llu, bitmaps uptodate=%*pbl error=%*pbl dirty=%*pbl writeback=%*pbl ordered=%*pbl checked=%*pbl",
|
||||
start, len, page_offset(page),
|
||||
start, len, folio_pos(folio),
|
||||
subpage_info->bitmap_nr_bits, &uptodate_bitmap,
|
||||
subpage_info->bitmap_nr_bits, &error_bitmap,
|
||||
subpage_info->bitmap_nr_bits, &dirty_bitmap,
|
||||
|
@ -73,71 +73,68 @@ enum btrfs_subpage_type {
|
||||
BTRFS_SUBPAGE_DATA,
|
||||
};
|
||||
|
||||
bool btrfs_is_subpage(const struct btrfs_fs_info *fs_info, struct page *page);
|
||||
bool btrfs_is_subpage(const struct btrfs_fs_info *fs_info, struct address_space *mapping);
|
||||
|
||||
void btrfs_init_subpage_info(struct btrfs_subpage_info *subpage_info, u32 sectorsize);
|
||||
int btrfs_attach_subpage(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page, enum btrfs_subpage_type type);
|
||||
void btrfs_detach_subpage(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page);
|
||||
struct folio *folio, enum btrfs_subpage_type type);
|
||||
void btrfs_detach_subpage(const struct btrfs_fs_info *fs_info, struct folio *folio);
|
||||
|
||||
/* Allocate additional data where page represents more than one sector */
|
||||
struct btrfs_subpage *btrfs_alloc_subpage(const struct btrfs_fs_info *fs_info,
|
||||
enum btrfs_subpage_type type);
|
||||
void btrfs_free_subpage(struct btrfs_subpage *subpage);
|
||||
|
||||
void btrfs_page_inc_eb_refs(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page);
|
||||
void btrfs_page_dec_eb_refs(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page);
|
||||
void btrfs_folio_inc_eb_refs(const struct btrfs_fs_info *fs_info, struct folio *folio);
|
||||
void btrfs_folio_dec_eb_refs(const struct btrfs_fs_info *fs_info, struct folio *folio);
|
||||
|
||||
void btrfs_subpage_start_reader(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page, u64 start, u32 len);
|
||||
struct folio *folio, u64 start, u32 len);
|
||||
void btrfs_subpage_end_reader(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page, u64 start, u32 len);
|
||||
struct folio *folio, u64 start, u32 len);
|
||||
|
||||
void btrfs_subpage_start_writer(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page, u64 start, u32 len);
|
||||
struct folio *folio, u64 start, u32 len);
|
||||
bool btrfs_subpage_end_and_test_writer(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page, u64 start, u32 len);
|
||||
int btrfs_page_start_writer_lock(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page, u64 start, u32 len);
|
||||
void btrfs_page_end_writer_lock(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page, u64 start, u32 len);
|
||||
struct folio *folio, u64 start, u32 len);
|
||||
int btrfs_folio_start_writer_lock(const struct btrfs_fs_info *fs_info,
|
||||
struct folio *folio, u64 start, u32 len);
|
||||
void btrfs_folio_end_writer_lock(const struct btrfs_fs_info *fs_info,
|
||||
struct folio *folio, u64 start, u32 len);
|
||||
|
||||
/*
|
||||
* Template for subpage related operations.
|
||||
*
|
||||
* btrfs_subpage_*() are for call sites where the page has subpage attached and
|
||||
* the range is ensured to be inside the page.
|
||||
* btrfs_subpage_*() are for call sites where the folio has subpage attached and
|
||||
* the range is ensured to be inside the folio's single page.
|
||||
*
|
||||
* btrfs_page_*() are for call sites where the page can either be subpage
|
||||
* specific or regular page. The function will handle both cases.
|
||||
* But the range still needs to be inside the page.
|
||||
* btrfs_folio_*() are for call sites where the page can either be subpage
|
||||
* specific or regular folios. The function will handle both cases.
|
||||
* But the range still needs to be inside one single page.
|
||||
*
|
||||
* btrfs_page_clamp_*() are similar to btrfs_page_*(), except the range doesn't
|
||||
* btrfs_folio_clamp_*() are similar to btrfs_folio_*(), except the range doesn't
|
||||
* need to be inside the page. Those functions will truncate the range
|
||||
* automatically.
|
||||
*/
|
||||
#define DECLARE_BTRFS_SUBPAGE_OPS(name) \
|
||||
void btrfs_subpage_set_##name(const struct btrfs_fs_info *fs_info, \
|
||||
struct page *page, u64 start, u32 len); \
|
||||
struct folio *folio, u64 start, u32 len); \
|
||||
void btrfs_subpage_clear_##name(const struct btrfs_fs_info *fs_info, \
|
||||
struct page *page, u64 start, u32 len); \
|
||||
struct folio *folio, u64 start, u32 len); \
|
||||
bool btrfs_subpage_test_##name(const struct btrfs_fs_info *fs_info, \
|
||||
struct page *page, u64 start, u32 len); \
|
||||
void btrfs_page_set_##name(const struct btrfs_fs_info *fs_info, \
|
||||
struct page *page, u64 start, u32 len); \
|
||||
void btrfs_page_clear_##name(const struct btrfs_fs_info *fs_info, \
|
||||
struct page *page, u64 start, u32 len); \
|
||||
bool btrfs_page_test_##name(const struct btrfs_fs_info *fs_info, \
|
||||
struct page *page, u64 start, u32 len); \
|
||||
void btrfs_page_clamp_set_##name(const struct btrfs_fs_info *fs_info, \
|
||||
struct page *page, u64 start, u32 len); \
|
||||
void btrfs_page_clamp_clear_##name(const struct btrfs_fs_info *fs_info, \
|
||||
struct page *page, u64 start, u32 len); \
|
||||
bool btrfs_page_clamp_test_##name(const struct btrfs_fs_info *fs_info, \
|
||||
struct page *page, u64 start, u32 len);
|
||||
struct folio *folio, u64 start, u32 len); \
|
||||
void btrfs_folio_set_##name(const struct btrfs_fs_info *fs_info, \
|
||||
struct folio *folio, u64 start, u32 len); \
|
||||
void btrfs_folio_clear_##name(const struct btrfs_fs_info *fs_info, \
|
||||
struct folio *folio, u64 start, u32 len); \
|
||||
bool btrfs_folio_test_##name(const struct btrfs_fs_info *fs_info, \
|
||||
struct folio *folio, u64 start, u32 len); \
|
||||
void btrfs_folio_clamp_set_##name(const struct btrfs_fs_info *fs_info, \
|
||||
struct folio *folio, u64 start, u32 len); \
|
||||
void btrfs_folio_clamp_clear_##name(const struct btrfs_fs_info *fs_info, \
|
||||
struct folio *folio, u64 start, u32 len); \
|
||||
bool btrfs_folio_clamp_test_##name(const struct btrfs_fs_info *fs_info, \
|
||||
struct folio *folio, u64 start, u32 len);
|
||||
|
||||
DECLARE_BTRFS_SUBPAGE_OPS(uptodate);
|
||||
DECLARE_BTRFS_SUBPAGE_OPS(dirty);
|
||||
@ -146,13 +143,12 @@ DECLARE_BTRFS_SUBPAGE_OPS(ordered);
|
||||
DECLARE_BTRFS_SUBPAGE_OPS(checked);
|
||||
|
||||
bool btrfs_subpage_clear_and_test_dirty(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page, u64 start, u32 len);
|
||||
struct folio *folio, u64 start, u32 len);
|
||||
|
||||
void btrfs_page_assert_not_dirty(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page);
|
||||
void btrfs_page_unlock_writer(struct btrfs_fs_info *fs_info, struct page *page,
|
||||
u64 start, u32 len);
|
||||
void btrfs_folio_assert_not_dirty(const struct btrfs_fs_info *fs_info, struct folio *folio);
|
||||
void btrfs_folio_unlock_writer(struct btrfs_fs_info *fs_info,
|
||||
struct folio *folio, u64 start, u32 len);
|
||||
void __cold btrfs_subpage_dump_bitmap(const struct btrfs_fs_info *fs_info,
|
||||
struct page *page, u64 start, u32 len);
|
||||
struct folio *folio, u64 start, u32 len);
|
||||
|
||||
#endif
|
||||
|
2359
fs/btrfs/super.c
2359
fs/btrfs/super.c
File diff suppressed because it is too large
Load Diff
@ -3,11 +3,12 @@
|
||||
#ifndef BTRFS_SUPER_H
|
||||
#define BTRFS_SUPER_H
|
||||
|
||||
int btrfs_parse_options(struct btrfs_fs_info *info, char *options,
|
||||
unsigned long new_flags);
|
||||
bool btrfs_check_options(struct btrfs_fs_info *info, unsigned long *mount_opt,
|
||||
unsigned long flags);
|
||||
int btrfs_sync_fs(struct super_block *sb, int wait);
|
||||
char *btrfs_get_subvol_name_from_objectid(struct btrfs_fs_info *fs_info,
|
||||
u64 subvol_objectid);
|
||||
void btrfs_set_free_space_cache_settings(struct btrfs_fs_info *fs_info);
|
||||
|
||||
static inline struct btrfs_fs_info *btrfs_sb(struct super_block *sb)
|
||||
{
|
||||
|
@ -1783,6 +1783,10 @@ static ssize_t btrfs_devinfo_scrub_speed_max_store(struct kobject *kobj,
|
||||
unsigned long long limit;
|
||||
|
||||
limit = memparse(buf, &endptr);
|
||||
/* There could be trailing '\n', also catch any typos after the value. */
|
||||
endptr = skip_spaces(endptr);
|
||||
if (*endptr != 0)
|
||||
return -EINVAL;
|
||||
WRITE_ONCE(device->scrub_speed_max, limit);
|
||||
return len;
|
||||
}
|
||||
|
@ -28,6 +28,7 @@ const char *test_error[] = {
|
||||
[TEST_ALLOC_INODE] = "cannot allocate inode",
|
||||
[TEST_ALLOC_BLOCK_GROUP] = "cannot allocate block group",
|
||||
[TEST_ALLOC_EXTENT_MAP] = "cannot allocate extent map",
|
||||
[TEST_ALLOC_CHUNK_MAP] = "cannot allocate chunk map",
|
||||
};
|
||||
|
||||
static const struct super_operations btrfs_test_super_ops = {
|
||||
@ -102,7 +103,7 @@ struct btrfs_device *btrfs_alloc_dummy_device(struct btrfs_fs_info *fs_info)
|
||||
if (!dev)
|
||||
return ERR_PTR(-ENOMEM);
|
||||
|
||||
extent_io_tree_init(NULL, &dev->alloc_state, 0);
|
||||
extent_io_tree_init(fs_info, &dev->alloc_state, 0);
|
||||
INIT_LIST_HEAD(&dev->dev_list);
|
||||
list_add(&dev->dev_list, &fs_info->fs_devices->devices);
|
||||
|
||||
@ -185,7 +186,7 @@ void btrfs_free_dummy_fs_info(struct btrfs_fs_info *fs_info)
|
||||
}
|
||||
spin_unlock(&fs_info->buffer_lock);
|
||||
|
||||
btrfs_mapping_tree_free(&fs_info->mapping_tree);
|
||||
btrfs_mapping_tree_free(fs_info);
|
||||
list_for_each_entry_safe(dev, tmp, &fs_info->fs_devices->devices,
|
||||
dev_list) {
|
||||
btrfs_free_dummy_device(dev);
|
||||
|
@ -23,6 +23,7 @@ enum {
|
||||
TEST_ALLOC_INODE,
|
||||
TEST_ALLOC_BLOCK_GROUP,
|
||||
TEST_ALLOC_EXTENT_MAP,
|
||||
TEST_ALLOC_CHUNK_MAP,
|
||||
};
|
||||
|
||||
extern const char *test_error[];
|
||||
|
@ -652,7 +652,7 @@ static void dump_eb_and_memory_contents(struct extent_buffer *eb, void *memory,
|
||||
const char *test_name)
|
||||
{
|
||||
for (int i = 0; i < eb->len; i++) {
|
||||
struct page *page = eb->pages[i >> PAGE_SHIFT];
|
||||
struct page *page = folio_page(eb->folios[i >> PAGE_SHIFT], 0);
|
||||
void *addr = page_address(page) + offset_in_page(i);
|
||||
|
||||
if (memcmp(addr, memory + i, 1) != 0) {
|
||||
@ -668,7 +668,7 @@ static int verify_eb_and_memory(struct extent_buffer *eb, void *memory,
|
||||
const char *test_name)
|
||||
{
|
||||
for (int i = 0; i < (eb->len >> PAGE_SHIFT); i++) {
|
||||
void *eb_addr = page_address(eb->pages[i]);
|
||||
void *eb_addr = folio_address(eb->folios[i]);
|
||||
|
||||
if (memcmp(memory + (i << PAGE_SHIFT), eb_addr, PAGE_SIZE) != 0) {
|
||||
dump_eb_and_memory_contents(eb, memory, test_name);
|
||||
|
@ -25,7 +25,7 @@ static void free_extent_map_tree(struct extent_map_tree *em_tree)
|
||||
#ifdef CONFIG_BTRFS_DEBUG
|
||||
if (refcount_read(&em->refs) != 1) {
|
||||
test_err(
|
||||
"em leak: em (start 0x%llx len 0x%llx block_start 0x%llx block_len 0x%llx) refs %d",
|
||||
"em leak: em (start %llu len %llu block_start %llu block_len %llu) refs %d",
|
||||
em->start, em->len, em->block_start,
|
||||
em->block_len, refcount_read(&em->refs));
|
||||
|
||||
@ -73,7 +73,7 @@ static int test_case_1(struct btrfs_fs_info *fs_info,
|
||||
em->block_start = 0;
|
||||
em->block_len = SZ_16K;
|
||||
write_lock(&em_tree->lock);
|
||||
ret = add_extent_mapping(em_tree, em, 0);
|
||||
ret = btrfs_add_extent_mapping(fs_info, em_tree, &em, em->start, em->len);
|
||||
write_unlock(&em_tree->lock);
|
||||
if (ret < 0) {
|
||||
test_err("cannot add extent range [0, 16K)");
|
||||
@ -94,7 +94,7 @@ static int test_case_1(struct btrfs_fs_info *fs_info,
|
||||
em->block_start = SZ_32K; /* avoid merging */
|
||||
em->block_len = SZ_4K;
|
||||
write_lock(&em_tree->lock);
|
||||
ret = add_extent_mapping(em_tree, em, 0);
|
||||
ret = btrfs_add_extent_mapping(fs_info, em_tree, &em, em->start, em->len);
|
||||
write_unlock(&em_tree->lock);
|
||||
if (ret < 0) {
|
||||
test_err("cannot add extent range [16K, 20K)");
|
||||
@ -121,9 +121,14 @@ static int test_case_1(struct btrfs_fs_info *fs_info,
|
||||
test_err("case1 [%llu %llu]: ret %d", start, start + len, ret);
|
||||
goto out;
|
||||
}
|
||||
if (em &&
|
||||
(em->start != 0 || extent_map_end(em) != SZ_16K ||
|
||||
em->block_start != 0 || em->block_len != SZ_16K)) {
|
||||
if (!em) {
|
||||
test_err("case1 [%llu %llu]: no extent map returned",
|
||||
start, start + len);
|
||||
ret = -ENOENT;
|
||||
goto out;
|
||||
}
|
||||
if (em->start != 0 || extent_map_end(em) != SZ_16K ||
|
||||
em->block_start != 0 || em->block_len != SZ_16K) {
|
||||
test_err(
|
||||
"case1 [%llu %llu]: ret %d return a wrong em (start %llu len %llu block_start %llu block_len %llu",
|
||||
start, start + len, ret, em->start, em->len,
|
||||
@ -161,7 +166,7 @@ static int test_case_2(struct btrfs_fs_info *fs_info,
|
||||
em->block_start = EXTENT_MAP_INLINE;
|
||||
em->block_len = (u64)-1;
|
||||
write_lock(&em_tree->lock);
|
||||
ret = add_extent_mapping(em_tree, em, 0);
|
||||
ret = btrfs_add_extent_mapping(fs_info, em_tree, &em, em->start, em->len);
|
||||
write_unlock(&em_tree->lock);
|
||||
if (ret < 0) {
|
||||
test_err("cannot add extent range [0, 1K)");
|
||||
@ -182,7 +187,7 @@ static int test_case_2(struct btrfs_fs_info *fs_info,
|
||||
em->block_start = SZ_4K;
|
||||
em->block_len = SZ_4K;
|
||||
write_lock(&em_tree->lock);
|
||||
ret = add_extent_mapping(em_tree, em, 0);
|
||||
ret = btrfs_add_extent_mapping(fs_info, em_tree, &em, em->start, em->len);
|
||||
write_unlock(&em_tree->lock);
|
||||
if (ret < 0) {
|
||||
test_err("cannot add extent range [4K, 8K)");
|
||||
@ -209,9 +214,13 @@ static int test_case_2(struct btrfs_fs_info *fs_info,
|
||||
test_err("case2 [0 1K]: ret %d", ret);
|
||||
goto out;
|
||||
}
|
||||
if (em &&
|
||||
(em->start != 0 || extent_map_end(em) != SZ_1K ||
|
||||
em->block_start != EXTENT_MAP_INLINE || em->block_len != (u64)-1)) {
|
||||
if (!em) {
|
||||
test_err("case2 [0 1K]: no extent map returned");
|
||||
ret = -ENOENT;
|
||||
goto out;
|
||||
}
|
||||
if (em->start != 0 || extent_map_end(em) != SZ_1K ||
|
||||
em->block_start != EXTENT_MAP_INLINE || em->block_len != (u64)-1) {
|
||||
test_err(
|
||||
"case2 [0 1K]: ret %d return a wrong em (start %llu len %llu block_start %llu block_len %llu",
|
||||
ret, em->start, em->len, em->block_start,
|
||||
@ -244,7 +253,7 @@ static int __test_case_3(struct btrfs_fs_info *fs_info,
|
||||
em->block_start = SZ_4K;
|
||||
em->block_len = SZ_4K;
|
||||
write_lock(&em_tree->lock);
|
||||
ret = add_extent_mapping(em_tree, em, 0);
|
||||
ret = btrfs_add_extent_mapping(fs_info, em_tree, &em, em->start, em->len);
|
||||
write_unlock(&em_tree->lock);
|
||||
if (ret < 0) {
|
||||
test_err("cannot add extent range [4K, 8K)");
|
||||
@ -268,19 +277,24 @@ static int __test_case_3(struct btrfs_fs_info *fs_info,
|
||||
ret = btrfs_add_extent_mapping(fs_info, em_tree, &em, start, len);
|
||||
write_unlock(&em_tree->lock);
|
||||
if (ret) {
|
||||
test_err("case3 [0x%llx 0x%llx): ret %d",
|
||||
test_err("case3 [%llu %llu): ret %d",
|
||||
start, start + len, ret);
|
||||
goto out;
|
||||
}
|
||||
if (!em) {
|
||||
test_err("case3 [%llu %llu): no extent map returned",
|
||||
start, start + len);
|
||||
ret = -ENOENT;
|
||||
goto out;
|
||||
}
|
||||
/*
|
||||
* Since bytes within em are contiguous, em->block_start is identical to
|
||||
* em->start.
|
||||
*/
|
||||
if (em &&
|
||||
(start < em->start || start + len > extent_map_end(em) ||
|
||||
em->start != em->block_start || em->len != em->block_len)) {
|
||||
if (start < em->start || start + len > extent_map_end(em) ||
|
||||
em->start != em->block_start || em->len != em->block_len) {
|
||||
test_err(
|
||||
"case3 [0x%llx 0x%llx): ret %d em (start 0x%llx len 0x%llx block_start 0x%llx block_len 0x%llx)",
|
||||
"case3 [%llu %llu): ret %d em (start %llu len %llu block_start %llu block_len %llu)",
|
||||
start, start + len, ret, em->start, em->len,
|
||||
em->block_start, em->block_len);
|
||||
ret = -EINVAL;
|
||||
@ -343,7 +357,7 @@ static int __test_case_4(struct btrfs_fs_info *fs_info,
|
||||
em->block_start = 0;
|
||||
em->block_len = SZ_8K;
|
||||
write_lock(&em_tree->lock);
|
||||
ret = add_extent_mapping(em_tree, em, 0);
|
||||
ret = btrfs_add_extent_mapping(fs_info, em_tree, &em, em->start, em->len);
|
||||
write_unlock(&em_tree->lock);
|
||||
if (ret < 0) {
|
||||
test_err("cannot add extent range [0, 8K)");
|
||||
@ -364,7 +378,7 @@ static int __test_case_4(struct btrfs_fs_info *fs_info,
|
||||
em->block_start = SZ_16K; /* avoid merging */
|
||||
em->block_len = 24 * SZ_1K;
|
||||
write_lock(&em_tree->lock);
|
||||
ret = add_extent_mapping(em_tree, em, 0);
|
||||
ret = btrfs_add_extent_mapping(fs_info, em_tree, &em, em->start, em->len);
|
||||
write_unlock(&em_tree->lock);
|
||||
if (ret < 0) {
|
||||
test_err("cannot add extent range [8K, 32K)");
|
||||
@ -387,14 +401,20 @@ static int __test_case_4(struct btrfs_fs_info *fs_info,
|
||||
ret = btrfs_add_extent_mapping(fs_info, em_tree, &em, start, len);
|
||||
write_unlock(&em_tree->lock);
|
||||
if (ret) {
|
||||
test_err("case4 [0x%llx 0x%llx): ret %d",
|
||||
start, len, ret);
|
||||
test_err("case4 [%llu %llu): ret %d",
|
||||
start, start + len, ret);
|
||||
goto out;
|
||||
}
|
||||
if (em && (start < em->start || start + len > extent_map_end(em))) {
|
||||
if (!em) {
|
||||
test_err("case4 [%llu %llu): no extent map returned",
|
||||
start, start + len);
|
||||
ret = -ENOENT;
|
||||
goto out;
|
||||
}
|
||||
if (start < em->start || start + len > extent_map_end(em)) {
|
||||
test_err(
|
||||
"case4 [0x%llx 0x%llx): ret %d, added wrong em (start 0x%llx len 0x%llx block_start 0x%llx block_len 0x%llx)",
|
||||
start, len, ret, em->start, em->len, em->block_start,
|
||||
"case4 [%llu %llu): ret %d, added wrong em (start %llu len %llu block_start %llu block_len %llu)",
|
||||
start, start + len, ret, em->start, em->len, em->block_start,
|
||||
em->block_len);
|
||||
ret = -EINVAL;
|
||||
}
|
||||
@ -443,7 +463,8 @@ static int test_case_4(struct btrfs_fs_info *fs_info,
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int add_compressed_extent(struct extent_map_tree *em_tree,
|
||||
static int add_compressed_extent(struct btrfs_fs_info *fs_info,
|
||||
struct extent_map_tree *em_tree,
|
||||
u64 start, u64 len, u64 block_start)
|
||||
{
|
||||
struct extent_map *em;
|
||||
@ -459,9 +480,9 @@ static int add_compressed_extent(struct extent_map_tree *em_tree,
|
||||
em->len = len;
|
||||
em->block_start = block_start;
|
||||
em->block_len = SZ_4K;
|
||||
set_bit(EXTENT_FLAG_COMPRESSED, &em->flags);
|
||||
em->flags |= EXTENT_FLAG_COMPRESS_ZLIB;
|
||||
write_lock(&em_tree->lock);
|
||||
ret = add_extent_mapping(em_tree, em, 0);
|
||||
ret = btrfs_add_extent_mapping(fs_info, em_tree, &em, em->start, em->len);
|
||||
write_unlock(&em_tree->lock);
|
||||
free_extent_map(em);
|
||||
if (ret < 0) {
|
||||
@ -567,7 +588,7 @@ static int validate_range(struct extent_map_tree *em_tree, int index)
|
||||
* They'll have the EXTENT_FLAG_COMPRESSED flag set to keep the em tree from
|
||||
* merging the em's.
|
||||
*/
|
||||
static int test_case_5(void)
|
||||
static int test_case_5(struct btrfs_fs_info *fs_info)
|
||||
{
|
||||
struct extent_map_tree *em_tree;
|
||||
struct inode *inode;
|
||||
@ -585,35 +606,35 @@ static int test_case_5(void)
|
||||
em_tree = &BTRFS_I(inode)->extent_tree;
|
||||
|
||||
/* [0, 12k) */
|
||||
ret = add_compressed_extent(em_tree, 0, SZ_4K * 3, 0);
|
||||
ret = add_compressed_extent(fs_info, em_tree, 0, SZ_4K * 3, 0);
|
||||
if (ret) {
|
||||
test_err("cannot add extent range [0, 12K)");
|
||||
goto out;
|
||||
}
|
||||
|
||||
/* [12k, 24k) */
|
||||
ret = add_compressed_extent(em_tree, SZ_4K * 3, SZ_4K * 3, SZ_4K);
|
||||
ret = add_compressed_extent(fs_info, em_tree, SZ_4K * 3, SZ_4K * 3, SZ_4K);
|
||||
if (ret) {
|
||||
test_err("cannot add extent range [12k, 24k)");
|
||||
goto out;
|
||||
}
|
||||
|
||||
/* [24k, 36k) */
|
||||
ret = add_compressed_extent(em_tree, SZ_4K * 6, SZ_4K * 3, SZ_8K);
|
||||
ret = add_compressed_extent(fs_info, em_tree, SZ_4K * 6, SZ_4K * 3, SZ_8K);
|
||||
if (ret) {
|
||||
test_err("cannot add extent range [12k, 24k)");
|
||||
goto out;
|
||||
}
|
||||
|
||||
/* [36k, 40k) */
|
||||
ret = add_compressed_extent(em_tree, SZ_32K + SZ_4K, SZ_4K, SZ_4K * 3);
|
||||
ret = add_compressed_extent(fs_info, em_tree, SZ_32K + SZ_4K, SZ_4K, SZ_4K * 3);
|
||||
if (ret) {
|
||||
test_err("cannot add extent range [12k, 24k)");
|
||||
goto out;
|
||||
}
|
||||
|
||||
/* [40k, 64k) */
|
||||
ret = add_compressed_extent(em_tree, SZ_4K * 10, SZ_4K * 6, SZ_16K);
|
||||
ret = add_compressed_extent(fs_info, em_tree, SZ_4K * 10, SZ_4K * 6, SZ_16K);
|
||||
if (ret) {
|
||||
test_err("cannot add extent range [12k, 24k)");
|
||||
goto out;
|
||||
@ -665,11 +686,11 @@ static int test_case_6(struct btrfs_fs_info *fs_info, struct extent_map_tree *em
|
||||
struct extent_map *em = NULL;
|
||||
int ret;
|
||||
|
||||
ret = add_compressed_extent(em_tree, 0, SZ_4K, 0);
|
||||
ret = add_compressed_extent(fs_info, em_tree, 0, SZ_4K, 0);
|
||||
if (ret)
|
||||
goto out;
|
||||
|
||||
ret = add_compressed_extent(em_tree, SZ_4K, SZ_4K, 0);
|
||||
ret = add_compressed_extent(fs_info, em_tree, SZ_4K, SZ_4K, 0);
|
||||
if (ret)
|
||||
goto out;
|
||||
|
||||
@ -713,7 +734,7 @@ out:
|
||||
* true would mess up the start/end calculations and subsequent splits would be
|
||||
* incorrect.
|
||||
*/
|
||||
static int test_case_7(void)
|
||||
static int test_case_7(struct btrfs_fs_info *fs_info)
|
||||
{
|
||||
struct extent_map_tree *em_tree;
|
||||
struct extent_map *em;
|
||||
@ -742,9 +763,9 @@ static int test_case_7(void)
|
||||
em->len = SZ_16K;
|
||||
em->block_start = 0;
|
||||
em->block_len = SZ_4K;
|
||||
set_bit(EXTENT_FLAG_PINNED, &em->flags);
|
||||
em->flags |= EXTENT_FLAG_PINNED;
|
||||
write_lock(&em_tree->lock);
|
||||
ret = add_extent_mapping(em_tree, em, 0);
|
||||
ret = btrfs_add_extent_mapping(fs_info, em_tree, &em, em->start, em->len);
|
||||
write_unlock(&em_tree->lock);
|
||||
if (ret < 0) {
|
||||
test_err("couldn't add extent map");
|
||||
@ -765,7 +786,7 @@ static int test_case_7(void)
|
||||
em->block_start = SZ_32K;
|
||||
em->block_len = SZ_16K;
|
||||
write_lock(&em_tree->lock);
|
||||
ret = add_extent_mapping(em_tree, em, 0);
|
||||
ret = btrfs_add_extent_mapping(fs_info, em_tree, &em, em->start, em->len);
|
||||
write_unlock(&em_tree->lock);
|
||||
if (ret < 0) {
|
||||
test_err("couldn't add extent map");
|
||||
@ -859,33 +880,21 @@ struct rmap_test_vector {
|
||||
static int test_rmap_block(struct btrfs_fs_info *fs_info,
|
||||
struct rmap_test_vector *test)
|
||||
{
|
||||
struct extent_map *em;
|
||||
struct map_lookup *map = NULL;
|
||||
struct btrfs_chunk_map *map;
|
||||
u64 *logical = NULL;
|
||||
int i, out_ndaddrs, out_stripe_len;
|
||||
int ret;
|
||||
|
||||
em = alloc_extent_map();
|
||||
if (!em) {
|
||||
test_std_err(TEST_ALLOC_EXTENT_MAP);
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
map = kmalloc(map_lookup_size(test->num_stripes), GFP_KERNEL);
|
||||
map = btrfs_alloc_chunk_map(test->num_stripes, GFP_KERNEL);
|
||||
if (!map) {
|
||||
kfree(em);
|
||||
test_std_err(TEST_ALLOC_EXTENT_MAP);
|
||||
test_std_err(TEST_ALLOC_CHUNK_MAP);
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
set_bit(EXTENT_FLAG_FS_MAPPING, &em->flags);
|
||||
/* Start at 4GiB logical address */
|
||||
em->start = SZ_4G;
|
||||
em->len = test->data_stripe_size * test->num_data_stripes;
|
||||
em->block_len = em->len;
|
||||
em->orig_block_len = test->data_stripe_size;
|
||||
em->map_lookup = map;
|
||||
|
||||
map->start = SZ_4G;
|
||||
map->chunk_len = test->data_stripe_size * test->num_data_stripes;
|
||||
map->stripe_size = test->data_stripe_size;
|
||||
map->num_stripes = test->num_stripes;
|
||||
map->type = test->raid_type;
|
||||
|
||||
@ -901,15 +910,13 @@ static int test_rmap_block(struct btrfs_fs_info *fs_info,
|
||||
map->stripes[i].physical = test->data_stripe_phys_start[i];
|
||||
}
|
||||
|
||||
write_lock(&fs_info->mapping_tree.lock);
|
||||
ret = add_extent_mapping(&fs_info->mapping_tree, em, 0);
|
||||
write_unlock(&fs_info->mapping_tree.lock);
|
||||
ret = btrfs_add_chunk_map(fs_info, map);
|
||||
if (ret) {
|
||||
test_err("error adding block group mapping to mapping tree");
|
||||
test_err("error adding chunk map to mapping tree");
|
||||
goto out_free;
|
||||
}
|
||||
|
||||
ret = btrfs_rmap_block(fs_info, em->start, btrfs_sb_offset(1),
|
||||
ret = btrfs_rmap_block(fs_info, map->start, btrfs_sb_offset(1),
|
||||
&logical, &out_ndaddrs, &out_stripe_len);
|
||||
if (ret || (out_ndaddrs == 0 && test->expected_mapped_addr)) {
|
||||
test_err("didn't rmap anything but expected %d",
|
||||
@ -938,14 +945,8 @@ static int test_rmap_block(struct btrfs_fs_info *fs_info,
|
||||
|
||||
ret = 0;
|
||||
out:
|
||||
write_lock(&fs_info->mapping_tree.lock);
|
||||
remove_extent_mapping(&fs_info->mapping_tree, em);
|
||||
write_unlock(&fs_info->mapping_tree.lock);
|
||||
/* For us */
|
||||
free_extent_map(em);
|
||||
btrfs_remove_chunk_map(fs_info, map);
|
||||
out_free:
|
||||
/* For the tree */
|
||||
free_extent_map(em);
|
||||
kfree(logical);
|
||||
return ret;
|
||||
}
|
||||
@ -1022,13 +1023,13 @@ int btrfs_test_extent_map(void)
|
||||
ret = test_case_4(fs_info, em_tree);
|
||||
if (ret)
|
||||
goto out;
|
||||
ret = test_case_5();
|
||||
ret = test_case_5(fs_info);
|
||||
if (ret)
|
||||
goto out;
|
||||
ret = test_case_6(fs_info, em_tree);
|
||||
if (ret)
|
||||
goto out;
|
||||
ret = test_case_7();
|
||||
ret = test_case_7(fs_info);
|
||||
if (ret)
|
||||
goto out;
|
||||
|
||||
|
@ -211,9 +211,9 @@ static void setup_file_extents(struct btrfs_root *root, u32 sectorsize)
|
||||
sectorsize, BTRFS_FILE_EXTENT_REG, 0, slot);
|
||||
}
|
||||
|
||||
static unsigned long prealloc_only = 0;
|
||||
static unsigned long compressed_only = 0;
|
||||
static unsigned long vacancy_only = 0;
|
||||
static u32 prealloc_only = 0;
|
||||
static u32 compressed_only = 0;
|
||||
static u32 vacancy_only = 0;
|
||||
|
||||
static noinline int test_btrfs_get_extent(u32 sectorsize, u32 nodesize)
|
||||
{
|
||||
@ -305,7 +305,7 @@ static noinline int test_btrfs_get_extent(u32 sectorsize, u32 nodesize)
|
||||
goto out;
|
||||
}
|
||||
if (em->flags != 0) {
|
||||
test_err("unexpected flags set, want 0 have %lu", em->flags);
|
||||
test_err("unexpected flags set, want 0 have %u", em->flags);
|
||||
goto out;
|
||||
}
|
||||
/*
|
||||
@ -332,7 +332,7 @@ static noinline int test_btrfs_get_extent(u32 sectorsize, u32 nodesize)
|
||||
goto out;
|
||||
}
|
||||
if (em->flags != 0) {
|
||||
test_err("unexpected flags set, want 0 have %lu", em->flags);
|
||||
test_err("unexpected flags set, want 0 have %u", em->flags);
|
||||
goto out;
|
||||
}
|
||||
offset = em->start + em->len;
|
||||
@ -355,7 +355,7 @@ static noinline int test_btrfs_get_extent(u32 sectorsize, u32 nodesize)
|
||||
goto out;
|
||||
}
|
||||
if (em->flags != 0) {
|
||||
test_err("unexpected flags set, want 0 have %lu", em->flags);
|
||||
test_err("unexpected flags set, want 0 have %u", em->flags);
|
||||
goto out;
|
||||
}
|
||||
if (em->orig_start != em->start) {
|
||||
@ -383,7 +383,7 @@ static noinline int test_btrfs_get_extent(u32 sectorsize, u32 nodesize)
|
||||
goto out;
|
||||
}
|
||||
if (em->flags != 0) {
|
||||
test_err("unexpected flags set, want 0 have %lu", em->flags);
|
||||
test_err("unexpected flags set, want 0 have %u", em->flags);
|
||||
goto out;
|
||||
}
|
||||
if (em->orig_start != em->start) {
|
||||
@ -412,7 +412,7 @@ static noinline int test_btrfs_get_extent(u32 sectorsize, u32 nodesize)
|
||||
goto out;
|
||||
}
|
||||
if (em->flags != 0) {
|
||||
test_err("unexpected flags set, want 0 have %lu", em->flags);
|
||||
test_err("unexpected flags set, want 0 have %u", em->flags);
|
||||
goto out;
|
||||
}
|
||||
offset = em->start + em->len;
|
||||
@ -434,7 +434,7 @@ static noinline int test_btrfs_get_extent(u32 sectorsize, u32 nodesize)
|
||||
goto out;
|
||||
}
|
||||
if (em->flags != 0) {
|
||||
test_err("unexpected flags set, want 0 have %lu", em->flags);
|
||||
test_err("unexpected flags set, want 0 have %u", em->flags);
|
||||
goto out;
|
||||
}
|
||||
if (em->orig_start != orig_start) {
|
||||
@ -468,7 +468,7 @@ static noinline int test_btrfs_get_extent(u32 sectorsize, u32 nodesize)
|
||||
goto out;
|
||||
}
|
||||
if (em->flags != prealloc_only) {
|
||||
test_err("unexpected flags set, want %lu have %lu",
|
||||
test_err("unexpected flags set, want %u have %u",
|
||||
prealloc_only, em->flags);
|
||||
goto out;
|
||||
}
|
||||
@ -497,7 +497,7 @@ static noinline int test_btrfs_get_extent(u32 sectorsize, u32 nodesize)
|
||||
goto out;
|
||||
}
|
||||
if (em->flags != prealloc_only) {
|
||||
test_err("unexpected flags set, want %lu have %lu",
|
||||
test_err("unexpected flags set, want %u have %u",
|
||||
prealloc_only, em->flags);
|
||||
goto out;
|
||||
}
|
||||
@ -527,7 +527,7 @@ static noinline int test_btrfs_get_extent(u32 sectorsize, u32 nodesize)
|
||||
goto out;
|
||||
}
|
||||
if (em->flags != 0) {
|
||||
test_err("unexpected flags set, want 0 have %lu", em->flags);
|
||||
test_err("unexpected flags set, want 0 have %u", em->flags);
|
||||
goto out;
|
||||
}
|
||||
if (em->orig_start != orig_start) {
|
||||
@ -560,7 +560,7 @@ static noinline int test_btrfs_get_extent(u32 sectorsize, u32 nodesize)
|
||||
goto out;
|
||||
}
|
||||
if (em->flags != prealloc_only) {
|
||||
test_err("unexpected flags set, want %lu have %lu",
|
||||
test_err("unexpected flags set, want %u have %u",
|
||||
prealloc_only, em->flags);
|
||||
goto out;
|
||||
}
|
||||
@ -595,7 +595,7 @@ static noinline int test_btrfs_get_extent(u32 sectorsize, u32 nodesize)
|
||||
goto out;
|
||||
}
|
||||
if (em->flags != compressed_only) {
|
||||
test_err("unexpected flags set, want %lu have %lu",
|
||||
test_err("unexpected flags set, want %u have %u",
|
||||
compressed_only, em->flags);
|
||||
goto out;
|
||||
}
|
||||
@ -604,9 +604,9 @@ static noinline int test_btrfs_get_extent(u32 sectorsize, u32 nodesize)
|
||||
em->start, em->orig_start);
|
||||
goto out;
|
||||
}
|
||||
if (em->compress_type != BTRFS_COMPRESS_ZLIB) {
|
||||
if (extent_map_compression(em) != BTRFS_COMPRESS_ZLIB) {
|
||||
test_err("unexpected compress type, wanted %d, got %d",
|
||||
BTRFS_COMPRESS_ZLIB, em->compress_type);
|
||||
BTRFS_COMPRESS_ZLIB, extent_map_compression(em));
|
||||
goto out;
|
||||
}
|
||||
offset = em->start + em->len;
|
||||
@ -629,7 +629,7 @@ static noinline int test_btrfs_get_extent(u32 sectorsize, u32 nodesize)
|
||||
goto out;
|
||||
}
|
||||
if (em->flags != compressed_only) {
|
||||
test_err("unexpected flags set, want %lu have %lu",
|
||||
test_err("unexpected flags set, want %u have %u",
|
||||
compressed_only, em->flags);
|
||||
goto out;
|
||||
}
|
||||
@ -638,9 +638,9 @@ static noinline int test_btrfs_get_extent(u32 sectorsize, u32 nodesize)
|
||||
em->start, em->orig_start);
|
||||
goto out;
|
||||
}
|
||||
if (em->compress_type != BTRFS_COMPRESS_ZLIB) {
|
||||
if (extent_map_compression(em) != BTRFS_COMPRESS_ZLIB) {
|
||||
test_err("unexpected compress type, wanted %d, got %d",
|
||||
BTRFS_COMPRESS_ZLIB, em->compress_type);
|
||||
BTRFS_COMPRESS_ZLIB, extent_map_compression(em));
|
||||
goto out;
|
||||
}
|
||||
disk_bytenr = em->block_start;
|
||||
@ -664,7 +664,7 @@ static noinline int test_btrfs_get_extent(u32 sectorsize, u32 nodesize)
|
||||
goto out;
|
||||
}
|
||||
if (em->flags != 0) {
|
||||
test_err("unexpected flags set, want 0 have %lu", em->flags);
|
||||
test_err("unexpected flags set, want 0 have %u", em->flags);
|
||||
goto out;
|
||||
}
|
||||
if (em->orig_start != em->start) {
|
||||
@ -692,7 +692,7 @@ static noinline int test_btrfs_get_extent(u32 sectorsize, u32 nodesize)
|
||||
goto out;
|
||||
}
|
||||
if (em->flags != compressed_only) {
|
||||
test_err("unexpected flags set, want %lu have %lu",
|
||||
test_err("unexpected flags set, want %u have %u",
|
||||
compressed_only, em->flags);
|
||||
goto out;
|
||||
}
|
||||
@ -701,9 +701,9 @@ static noinline int test_btrfs_get_extent(u32 sectorsize, u32 nodesize)
|
||||
em->start, orig_start);
|
||||
goto out;
|
||||
}
|
||||
if (em->compress_type != BTRFS_COMPRESS_ZLIB) {
|
||||
if (extent_map_compression(em) != BTRFS_COMPRESS_ZLIB) {
|
||||
test_err("unexpected compress type, wanted %d, got %d",
|
||||
BTRFS_COMPRESS_ZLIB, em->compress_type);
|
||||
BTRFS_COMPRESS_ZLIB, extent_map_compression(em));
|
||||
goto out;
|
||||
}
|
||||
offset = em->start + em->len;
|
||||
@ -726,7 +726,7 @@ static noinline int test_btrfs_get_extent(u32 sectorsize, u32 nodesize)
|
||||
goto out;
|
||||
}
|
||||
if (em->flags != 0) {
|
||||
test_err("unexpected flags set, want 0 have %lu", em->flags);
|
||||
test_err("unexpected flags set, want 0 have %u", em->flags);
|
||||
goto out;
|
||||
}
|
||||
if (em->orig_start != em->start) {
|
||||
@ -758,7 +758,7 @@ static noinline int test_btrfs_get_extent(u32 sectorsize, u32 nodesize)
|
||||
goto out;
|
||||
}
|
||||
if (em->flags != vacancy_only) {
|
||||
test_err("unexpected flags set, want %lu have %lu",
|
||||
test_err("unexpected flags set, want %u have %u",
|
||||
vacancy_only, em->flags);
|
||||
goto out;
|
||||
}
|
||||
@ -786,7 +786,7 @@ static noinline int test_btrfs_get_extent(u32 sectorsize, u32 nodesize)
|
||||
goto out;
|
||||
}
|
||||
if (em->flags != 0) {
|
||||
test_err("unexpected flags set, want 0 have %lu", em->flags);
|
||||
test_err("unexpected flags set, want 0 have %u", em->flags);
|
||||
goto out;
|
||||
}
|
||||
if (em->orig_start != em->start) {
|
||||
@ -866,7 +866,7 @@ static int test_hole_first(u32 sectorsize, u32 nodesize)
|
||||
goto out;
|
||||
}
|
||||
if (em->flags != vacancy_only) {
|
||||
test_err("wrong flags, wanted %lu, have %lu", vacancy_only,
|
||||
test_err("wrong flags, wanted %u, have %u", vacancy_only,
|
||||
em->flags);
|
||||
goto out;
|
||||
}
|
||||
@ -888,7 +888,7 @@ static int test_hole_first(u32 sectorsize, u32 nodesize)
|
||||
goto out;
|
||||
}
|
||||
if (em->flags != 0) {
|
||||
test_err("unexpected flags set, wanted 0 got %lu",
|
||||
test_err("unexpected flags set, wanted 0 got %u",
|
||||
em->flags);
|
||||
goto out;
|
||||
}
|
||||
@ -1095,8 +1095,8 @@ int btrfs_test_inodes(u32 sectorsize, u32 nodesize)
|
||||
|
||||
test_msg("running inode tests");
|
||||
|
||||
set_bit(EXTENT_FLAG_COMPRESSED, &compressed_only);
|
||||
set_bit(EXTENT_FLAG_PREALLOC, &prealloc_only);
|
||||
compressed_only |= EXTENT_FLAG_COMPRESS_ZLIB;
|
||||
prealloc_only |= EXTENT_FLAG_PREALLOC;
|
||||
|
||||
ret = test_btrfs_get_extent(sectorsize, nodesize);
|
||||
if (ret)
|
||||
|
@ -22,7 +22,7 @@ struct btrfs_tree_parent_check {
|
||||
|
||||
/*
|
||||
* Expected transid, can be 0 to skip the check, but such skip
|
||||
* should only be utlized for backref walk related code.
|
||||
* should only be utilized for backref walk related code.
|
||||
*/
|
||||
u64 transid;
|
||||
|
||||
|
@ -2575,7 +2575,6 @@ static int clean_log_buffer(struct btrfs_trans_handle *trans,
|
||||
ret = btrfs_pin_reserved_extent(trans, eb);
|
||||
if (ret)
|
||||
return ret;
|
||||
btrfs_redirty_list_add(trans->transaction, eb);
|
||||
} else {
|
||||
unaccount_log_buffer(eb->fs_info, eb->start);
|
||||
}
|
||||
@ -4520,7 +4519,7 @@ static int log_extent_csums(struct btrfs_trans_handle *trans,
|
||||
int ret = 0;
|
||||
|
||||
if (inode->flags & BTRFS_INODE_NODATASUM ||
|
||||
test_bit(EXTENT_FLAG_PREALLOC, &em->flags) ||
|
||||
(em->flags & EXTENT_FLAG_PREALLOC) ||
|
||||
em->block_start == EXTENT_MAP_HOLE)
|
||||
return 0;
|
||||
|
||||
@ -4583,7 +4582,7 @@ static int log_extent_csums(struct btrfs_trans_handle *trans,
|
||||
return 0;
|
||||
|
||||
/* If we're compressed we have to save the entire range of csums. */
|
||||
if (em->compress_type) {
|
||||
if (extent_map_is_compressed(em)) {
|
||||
csum_offset = 0;
|
||||
csum_len = max(em->block_len, em->orig_block_len);
|
||||
} else {
|
||||
@ -4623,18 +4622,20 @@ static int log_one_extent(struct btrfs_trans_handle *trans,
|
||||
struct btrfs_file_extent_item fi = { 0 };
|
||||
struct extent_buffer *leaf;
|
||||
struct btrfs_key key;
|
||||
enum btrfs_compression_type compress_type;
|
||||
u64 extent_offset = em->start - em->orig_start;
|
||||
u64 block_len;
|
||||
int ret;
|
||||
|
||||
btrfs_set_stack_file_extent_generation(&fi, trans->transid);
|
||||
if (test_bit(EXTENT_FLAG_PREALLOC, &em->flags))
|
||||
if (em->flags & EXTENT_FLAG_PREALLOC)
|
||||
btrfs_set_stack_file_extent_type(&fi, BTRFS_FILE_EXTENT_PREALLOC);
|
||||
else
|
||||
btrfs_set_stack_file_extent_type(&fi, BTRFS_FILE_EXTENT_REG);
|
||||
|
||||
block_len = max(em->block_len, em->orig_block_len);
|
||||
if (em->compress_type != BTRFS_COMPRESS_NONE) {
|
||||
compress_type = extent_map_compression(em);
|
||||
if (compress_type != BTRFS_COMPRESS_NONE) {
|
||||
btrfs_set_stack_file_extent_disk_bytenr(&fi, em->block_start);
|
||||
btrfs_set_stack_file_extent_disk_num_bytes(&fi, block_len);
|
||||
} else if (em->block_start < EXTENT_MAP_LAST_BYTE) {
|
||||
@ -4646,7 +4647,7 @@ static int log_one_extent(struct btrfs_trans_handle *trans,
|
||||
btrfs_set_stack_file_extent_offset(&fi, extent_offset);
|
||||
btrfs_set_stack_file_extent_num_bytes(&fi, em->len);
|
||||
btrfs_set_stack_file_extent_ram_bytes(&fi, em->ram_bytes);
|
||||
btrfs_set_stack_file_extent_compression(&fi, em->compress_type);
|
||||
btrfs_set_stack_file_extent_compression(&fi, compress_type);
|
||||
|
||||
ret = log_extent_csums(trans, inode, log, em, ctx);
|
||||
if (ret)
|
||||
@ -4859,13 +4860,13 @@ static int btrfs_log_changed_extents(struct btrfs_trans_handle *trans,
|
||||
continue;
|
||||
|
||||
/* We log prealloc extents beyond eof later. */
|
||||
if (test_bit(EXTENT_FLAG_PREALLOC, &em->flags) &&
|
||||
if ((em->flags & EXTENT_FLAG_PREALLOC) &&
|
||||
em->start >= i_size_read(&inode->vfs_inode))
|
||||
continue;
|
||||
|
||||
/* Need a ref to keep it from getting evicted from cache */
|
||||
refcount_inc(&em->refs);
|
||||
set_bit(EXTENT_FLAG_LOGGING, &em->flags);
|
||||
em->flags |= EXTENT_FLAG_LOGGING;
|
||||
list_add_tail(&em->list, &extents);
|
||||
num++;
|
||||
}
|
||||
|
File diff suppressed because it is too large
Load Diff
@ -426,7 +426,8 @@ struct btrfs_discard_stripe {
|
||||
struct btrfs_io_context {
|
||||
refcount_t refs;
|
||||
struct btrfs_fs_info *fs_info;
|
||||
u64 map_type; /* get from map_lookup->type */
|
||||
/* Taken from struct btrfs_chunk_map::type. */
|
||||
u64 map_type;
|
||||
struct bio *orig_bio;
|
||||
atomic_t error;
|
||||
u16 max_errors;
|
||||
@ -529,18 +530,32 @@ struct btrfs_raid_attr {
|
||||
|
||||
extern const struct btrfs_raid_attr btrfs_raid_array[BTRFS_NR_RAID_TYPES];
|
||||
|
||||
struct map_lookup {
|
||||
struct btrfs_chunk_map {
|
||||
struct rb_node rb_node;
|
||||
/* For mount time dev extent verification. */
|
||||
int verified_stripes;
|
||||
refcount_t refs;
|
||||
u64 start;
|
||||
u64 chunk_len;
|
||||
u64 stripe_size;
|
||||
u64 type;
|
||||
int io_align;
|
||||
int io_width;
|
||||
int num_stripes;
|
||||
int sub_stripes;
|
||||
int verified_stripes; /* For mount time dev extent verification */
|
||||
struct btrfs_io_stripe stripes[];
|
||||
};
|
||||
|
||||
#define map_lookup_size(n) (sizeof(struct map_lookup) + \
|
||||
(sizeof(struct btrfs_io_stripe) * (n)))
|
||||
#define btrfs_chunk_map_size(n) (sizeof(struct btrfs_chunk_map) + \
|
||||
(sizeof(struct btrfs_io_stripe) * (n)))
|
||||
|
||||
static inline void btrfs_free_chunk_map(struct btrfs_chunk_map *map)
|
||||
{
|
||||
if (map && refcount_dec_and_test(&map->refs)) {
|
||||
ASSERT(RB_EMPTY_NODE(&map->rb_node));
|
||||
kfree(map);
|
||||
}
|
||||
}
|
||||
|
||||
struct btrfs_balance_args;
|
||||
struct btrfs_balance_progress;
|
||||
@ -598,7 +613,7 @@ static inline unsigned long btrfs_chunk_item_size(int num_stripes)
|
||||
}
|
||||
|
||||
/*
|
||||
* Do the type safe converstion from stripe_nr to offset inside the chunk.
|
||||
* Do the type safe conversion from stripe_nr to offset inside the chunk.
|
||||
*
|
||||
* @stripe_nr is u32, with left shift it can overflow u32 for chunks larger
|
||||
* than 4G. This does the proper type cast to avoid overflow.
|
||||
@ -624,7 +639,7 @@ int btrfs_read_sys_array(struct btrfs_fs_info *fs_info);
|
||||
int btrfs_read_chunk_tree(struct btrfs_fs_info *fs_info);
|
||||
struct btrfs_block_group *btrfs_create_chunk(struct btrfs_trans_handle *trans,
|
||||
u64 type);
|
||||
void btrfs_mapping_tree_free(struct extent_map_tree *tree);
|
||||
void btrfs_mapping_tree_free(struct btrfs_fs_info *fs_info);
|
||||
int btrfs_open_devices(struct btrfs_fs_devices *fs_devices,
|
||||
blk_mode_t flags, void *holder);
|
||||
struct btrfs_device *btrfs_scan_one_device(const char *path, blk_mode_t flags,
|
||||
@ -680,13 +695,25 @@ int btrfs_is_parity_mirror(struct btrfs_fs_info *fs_info,
|
||||
u64 logical, u64 len);
|
||||
unsigned long btrfs_full_stripe_len(struct btrfs_fs_info *fs_info,
|
||||
u64 logical);
|
||||
u64 btrfs_calc_stripe_length(const struct extent_map *em);
|
||||
u64 btrfs_calc_stripe_length(const struct btrfs_chunk_map *map);
|
||||
int btrfs_nr_parity_stripes(u64 type);
|
||||
int btrfs_chunk_alloc_add_chunk_item(struct btrfs_trans_handle *trans,
|
||||
struct btrfs_block_group *bg);
|
||||
int btrfs_remove_chunk(struct btrfs_trans_handle *trans, u64 chunk_offset);
|
||||
struct extent_map *btrfs_get_chunk_map(struct btrfs_fs_info *fs_info,
|
||||
u64 logical, u64 length);
|
||||
|
||||
#ifdef CONFIG_BTRFS_FS_RUN_SANITY_TESTS
|
||||
struct btrfs_chunk_map *btrfs_alloc_chunk_map(int num_stripes, gfp_t gfp);
|
||||
int btrfs_add_chunk_map(struct btrfs_fs_info *fs_info, struct btrfs_chunk_map *map);
|
||||
#endif
|
||||
|
||||
struct btrfs_chunk_map *btrfs_clone_chunk_map(struct btrfs_chunk_map *map, gfp_t gfp);
|
||||
struct btrfs_chunk_map *btrfs_find_chunk_map(struct btrfs_fs_info *fs_info,
|
||||
u64 logical, u64 length);
|
||||
struct btrfs_chunk_map *btrfs_find_chunk_map_nolock(struct btrfs_fs_info *fs_info,
|
||||
u64 logical, u64 length);
|
||||
struct btrfs_chunk_map *btrfs_get_chunk_map(struct btrfs_fs_info *fs_info,
|
||||
u64 logical, u64 length);
|
||||
void btrfs_remove_chunk_map(struct btrfs_fs_info *fs_info, struct btrfs_chunk_map *map);
|
||||
void btrfs_release_disk_super(struct btrfs_super_block *super);
|
||||
|
||||
static inline void btrfs_dev_stat_inc(struct btrfs_device *dev,
|
||||
|
@ -382,6 +382,53 @@ static int btrfs_xattr_handler_set(const struct xattr_handler *handler,
|
||||
return btrfs_setxattr_trans(inode, name, buffer, size, flags);
|
||||
}
|
||||
|
||||
static int btrfs_xattr_handler_get_security(const struct xattr_handler *handler,
|
||||
struct dentry *unused,
|
||||
struct inode *inode,
|
||||
const char *name, void *buffer,
|
||||
size_t size)
|
||||
{
|
||||
int ret;
|
||||
bool is_cap = false;
|
||||
|
||||
name = xattr_full_name(handler, name);
|
||||
|
||||
/*
|
||||
* security.capability doesn't cache the results, so calls into us
|
||||
* constantly to see if there's a capability xattr. Cache the result
|
||||
* here in order to avoid wasting time doing lookups for xattrs we know
|
||||
* don't exist.
|
||||
*/
|
||||
if (strcmp(name, XATTR_NAME_CAPS) == 0) {
|
||||
is_cap = true;
|
||||
if (test_bit(BTRFS_INODE_NO_CAP_XATTR, &BTRFS_I(inode)->runtime_flags))
|
||||
return -ENODATA;
|
||||
}
|
||||
|
||||
ret = btrfs_getxattr(inode, name, buffer, size);
|
||||
if (ret == -ENODATA && is_cap)
|
||||
set_bit(BTRFS_INODE_NO_CAP_XATTR, &BTRFS_I(inode)->runtime_flags);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int btrfs_xattr_handler_set_security(const struct xattr_handler *handler,
|
||||
struct mnt_idmap *idmap,
|
||||
struct dentry *unused,
|
||||
struct inode *inode,
|
||||
const char *name,
|
||||
const void *buffer,
|
||||
size_t size, int flags)
|
||||
{
|
||||
if (btrfs_root_readonly(BTRFS_I(inode)->root))
|
||||
return -EROFS;
|
||||
|
||||
name = xattr_full_name(handler, name);
|
||||
if (strcmp(name, XATTR_NAME_CAPS) == 0)
|
||||
clear_bit(BTRFS_INODE_NO_CAP_XATTR, &BTRFS_I(inode)->runtime_flags);
|
||||
|
||||
return btrfs_setxattr_trans(inode, name, buffer, size, flags);
|
||||
}
|
||||
|
||||
static int btrfs_xattr_handler_set_prop(const struct xattr_handler *handler,
|
||||
struct mnt_idmap *idmap,
|
||||
struct dentry *unused, struct inode *inode,
|
||||
@ -420,8 +467,8 @@ static int btrfs_xattr_handler_set_prop(const struct xattr_handler *handler,
|
||||
|
||||
static const struct xattr_handler btrfs_security_xattr_handler = {
|
||||
.prefix = XATTR_SECURITY_PREFIX,
|
||||
.get = btrfs_xattr_handler_get,
|
||||
.set = btrfs_xattr_handler_set,
|
||||
.get = btrfs_xattr_handler_get_security,
|
||||
.set = btrfs_xattr_handler_set_security,
|
||||
};
|
||||
|
||||
static const struct xattr_handler btrfs_trusted_xattr_handler = {
|
||||
@ -473,6 +520,10 @@ static int btrfs_initxattrs(struct inode *inode,
|
||||
}
|
||||
strcpy(name, XATTR_SECURITY_PREFIX);
|
||||
strcpy(name + XATTR_SECURITY_PREFIX_LEN, xattr->name);
|
||||
|
||||
if (strcmp(name, XATTR_NAME_CAPS) == 0)
|
||||
clear_bit(BTRFS_INODE_NO_CAP_XATTR, &BTRFS_I(inode)->runtime_flags);
|
||||
|
||||
err = btrfs_setxattr(trans, inode, name, xattr->value,
|
||||
xattr->value_len, 0);
|
||||
kfree(name);
|
||||
|
@ -121,7 +121,7 @@ int zlib_compress_pages(struct list_head *ws, struct address_space *mapping,
|
||||
workspace->strm.total_in = 0;
|
||||
workspace->strm.total_out = 0;
|
||||
|
||||
out_page = alloc_page(GFP_NOFS);
|
||||
out_page = btrfs_alloc_compr_page();
|
||||
if (out_page == NULL) {
|
||||
ret = -ENOMEM;
|
||||
goto out;
|
||||
@ -200,7 +200,7 @@ int zlib_compress_pages(struct list_head *ws, struct address_space *mapping,
|
||||
ret = -E2BIG;
|
||||
goto out;
|
||||
}
|
||||
out_page = alloc_page(GFP_NOFS);
|
||||
out_page = btrfs_alloc_compr_page();
|
||||
if (out_page == NULL) {
|
||||
ret = -ENOMEM;
|
||||
goto out;
|
||||
@ -236,7 +236,7 @@ int zlib_compress_pages(struct list_head *ws, struct address_space *mapping,
|
||||
ret = -E2BIG;
|
||||
goto out;
|
||||
}
|
||||
out_page = alloc_page(GFP_NOFS);
|
||||
out_page = btrfs_alloc_compr_page();
|
||||
if (out_page == NULL) {
|
||||
ret = -ENOMEM;
|
||||
goto out;
|
||||
|
@ -781,7 +781,7 @@ int btrfs_check_zoned_mode(struct btrfs_fs_info *fs_info)
|
||||
* Check mount options here, because we might change fs_info->zoned
|
||||
* from fs_info->zone_size.
|
||||
*/
|
||||
ret = btrfs_check_mountopts_zoned(fs_info);
|
||||
ret = btrfs_check_mountopts_zoned(fs_info, &fs_info->mount_opt);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
@ -789,7 +789,7 @@ int btrfs_check_zoned_mode(struct btrfs_fs_info *fs_info)
|
||||
return 0;
|
||||
}
|
||||
|
||||
int btrfs_check_mountopts_zoned(struct btrfs_fs_info *info)
|
||||
int btrfs_check_mountopts_zoned(struct btrfs_fs_info *info, unsigned long *mount_opt)
|
||||
{
|
||||
if (!btrfs_is_zoned(info))
|
||||
return 0;
|
||||
@ -798,18 +798,21 @@ int btrfs_check_mountopts_zoned(struct btrfs_fs_info *info)
|
||||
* Space cache writing is not COWed. Disable that to avoid write errors
|
||||
* in sequential zones.
|
||||
*/
|
||||
if (btrfs_test_opt(info, SPACE_CACHE)) {
|
||||
if (btrfs_raw_test_opt(*mount_opt, SPACE_CACHE)) {
|
||||
btrfs_err(info, "zoned: space cache v1 is not supported");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
if (btrfs_test_opt(info, NODATACOW)) {
|
||||
if (btrfs_raw_test_opt(*mount_opt, NODATACOW)) {
|
||||
btrfs_err(info, "zoned: NODATACOW not supported");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
btrfs_clear_and_info(info, DISCARD_ASYNC,
|
||||
"zoned: async discard ignored and disabled for zoned mode");
|
||||
if (btrfs_raw_test_opt(*mount_opt, DISCARD_ASYNC)) {
|
||||
btrfs_info(info,
|
||||
"zoned: async discard ignored and disabled for zoned mode");
|
||||
btrfs_clear_opt(*mount_opt, DISCARD_ASYNC);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
@ -1290,7 +1293,7 @@ struct zone_info {
|
||||
|
||||
static int btrfs_load_zone_info(struct btrfs_fs_info *fs_info, int zone_idx,
|
||||
struct zone_info *info, unsigned long *active,
|
||||
struct map_lookup *map)
|
||||
struct btrfs_chunk_map *map)
|
||||
{
|
||||
struct btrfs_dev_replace *dev_replace = &fs_info->dev_replace;
|
||||
struct btrfs_device *device = map->stripes[zone_idx].dev;
|
||||
@ -1393,7 +1396,7 @@ static int btrfs_load_block_group_single(struct btrfs_block_group *bg,
|
||||
}
|
||||
|
||||
static int btrfs_load_block_group_dup(struct btrfs_block_group *bg,
|
||||
struct map_lookup *map,
|
||||
struct btrfs_chunk_map *map,
|
||||
struct zone_info *zone_info,
|
||||
unsigned long *active)
|
||||
{
|
||||
@ -1435,7 +1438,7 @@ static int btrfs_load_block_group_dup(struct btrfs_block_group *bg,
|
||||
}
|
||||
|
||||
static int btrfs_load_block_group_raid1(struct btrfs_block_group *bg,
|
||||
struct map_lookup *map,
|
||||
struct btrfs_chunk_map *map,
|
||||
struct zone_info *zone_info,
|
||||
unsigned long *active)
|
||||
{
|
||||
@ -1483,7 +1486,7 @@ static int btrfs_load_block_group_raid1(struct btrfs_block_group *bg,
|
||||
}
|
||||
|
||||
static int btrfs_load_block_group_raid0(struct btrfs_block_group *bg,
|
||||
struct map_lookup *map,
|
||||
struct btrfs_chunk_map *map,
|
||||
struct zone_info *zone_info,
|
||||
unsigned long *active)
|
||||
{
|
||||
@ -1515,7 +1518,7 @@ static int btrfs_load_block_group_raid0(struct btrfs_block_group *bg,
|
||||
}
|
||||
|
||||
static int btrfs_load_block_group_raid10(struct btrfs_block_group *bg,
|
||||
struct map_lookup *map,
|
||||
struct btrfs_chunk_map *map,
|
||||
struct zone_info *zone_info,
|
||||
unsigned long *active)
|
||||
{
|
||||
@ -1552,9 +1555,7 @@ static int btrfs_load_block_group_raid10(struct btrfs_block_group *bg,
|
||||
int btrfs_load_block_group_zone_info(struct btrfs_block_group *cache, bool new)
|
||||
{
|
||||
struct btrfs_fs_info *fs_info = cache->fs_info;
|
||||
struct extent_map_tree *em_tree = &fs_info->mapping_tree;
|
||||
struct extent_map *em;
|
||||
struct map_lookup *map;
|
||||
struct btrfs_chunk_map *map;
|
||||
u64 logical = cache->start;
|
||||
u64 length = cache->length;
|
||||
struct zone_info *zone_info = NULL;
|
||||
@ -1575,17 +1576,11 @@ int btrfs_load_block_group_zone_info(struct btrfs_block_group *cache, bool new)
|
||||
return -EIO;
|
||||
}
|
||||
|
||||
/* Get the chunk mapping */
|
||||
read_lock(&em_tree->lock);
|
||||
em = lookup_extent_mapping(em_tree, logical, length);
|
||||
read_unlock(&em_tree->lock);
|
||||
|
||||
if (!em)
|
||||
map = btrfs_find_chunk_map(fs_info, logical, length);
|
||||
if (!map)
|
||||
return -EINVAL;
|
||||
|
||||
map = em->map_lookup;
|
||||
|
||||
cache->physical_map = kmemdup(map, map_lookup_size(map->num_stripes), GFP_NOFS);
|
||||
cache->physical_map = btrfs_clone_chunk_map(map, GFP_NOFS);
|
||||
if (!cache->physical_map) {
|
||||
ret = -ENOMEM;
|
||||
goto out;
|
||||
@ -1687,12 +1682,11 @@ out:
|
||||
spin_unlock(&fs_info->zone_active_bgs_lock);
|
||||
}
|
||||
} else {
|
||||
kfree(cache->physical_map);
|
||||
btrfs_free_chunk_map(cache->physical_map);
|
||||
cache->physical_map = NULL;
|
||||
}
|
||||
bitmap_free(active);
|
||||
kfree(zone_info);
|
||||
free_extent_map(em);
|
||||
|
||||
return ret;
|
||||
}
|
||||
@ -1715,22 +1709,6 @@ void btrfs_calc_zone_unusable(struct btrfs_block_group *cache)
|
||||
cache->zone_unusable = unusable;
|
||||
}
|
||||
|
||||
void btrfs_redirty_list_add(struct btrfs_transaction *trans,
|
||||
struct extent_buffer *eb)
|
||||
{
|
||||
if (!btrfs_is_zoned(eb->fs_info) ||
|
||||
btrfs_header_flag(eb, BTRFS_HEADER_FLAG_WRITTEN))
|
||||
return;
|
||||
|
||||
ASSERT(!test_bit(EXTENT_BUFFER_DIRTY, &eb->bflags));
|
||||
|
||||
memzero_extent_buffer(eb, 0, eb->len);
|
||||
set_bit(EXTENT_BUFFER_NO_CHECK, &eb->bflags);
|
||||
set_extent_buffer_dirty(eb);
|
||||
set_extent_bit(&trans->dirty_pages, eb->start, eb->start + eb->len - 1,
|
||||
EXTENT_DIRTY, NULL);
|
||||
}
|
||||
|
||||
bool btrfs_use_zone_append(struct btrfs_bio *bbio)
|
||||
{
|
||||
u64 start = (bbio->bio.bi_iter.bi_sector << SECTOR_SHIFT);
|
||||
@ -2082,7 +2060,7 @@ int btrfs_sync_zone_write_pointer(struct btrfs_device *tgt_dev, u64 logical,
|
||||
bool btrfs_zone_activate(struct btrfs_block_group *block_group)
|
||||
{
|
||||
struct btrfs_fs_info *fs_info = block_group->fs_info;
|
||||
struct map_lookup *map;
|
||||
struct btrfs_chunk_map *map;
|
||||
struct btrfs_device *device;
|
||||
u64 physical;
|
||||
const bool is_data = (block_group->flags & BTRFS_BLOCK_GROUP_DATA);
|
||||
@ -2194,7 +2172,7 @@ static void wait_eb_writebacks(struct btrfs_block_group *block_group)
|
||||
static int do_zone_finish(struct btrfs_block_group *block_group, bool fully_written)
|
||||
{
|
||||
struct btrfs_fs_info *fs_info = block_group->fs_info;
|
||||
struct map_lookup *map;
|
||||
struct btrfs_chunk_map *map;
|
||||
const bool is_metadata = (block_group->flags &
|
||||
(BTRFS_BLOCK_GROUP_METADATA | BTRFS_BLOCK_GROUP_SYSTEM));
|
||||
int ret = 0;
|
||||
@ -2643,7 +2621,7 @@ void btrfs_check_active_zone_reservation(struct btrfs_fs_info *fs_info)
|
||||
/* Release reservation for currently active block groups. */
|
||||
spin_lock(&fs_info->zone_active_bgs_lock);
|
||||
list_for_each_entry(block_group, &fs_info->zone_active_bgs, active_bg_list) {
|
||||
struct map_lookup *map = block_group->physical_map;
|
||||
struct btrfs_chunk_map *map = block_group->physical_map;
|
||||
|
||||
if (!(block_group->flags &
|
||||
(BTRFS_BLOCK_GROUP_METADATA | BTRFS_BLOCK_GROUP_SYSTEM)))
|
||||
|
@ -45,7 +45,7 @@ int btrfs_get_dev_zone_info(struct btrfs_device *device, bool populate_cache);
|
||||
void btrfs_destroy_dev_zone_info(struct btrfs_device *device);
|
||||
struct btrfs_zoned_device_info *btrfs_clone_dev_zone_info(struct btrfs_device *orig_dev);
|
||||
int btrfs_check_zoned_mode(struct btrfs_fs_info *fs_info);
|
||||
int btrfs_check_mountopts_zoned(struct btrfs_fs_info *info);
|
||||
int btrfs_check_mountopts_zoned(struct btrfs_fs_info *info, unsigned long *mount_opt);
|
||||
int btrfs_sb_log_location_bdev(struct block_device *bdev, int mirror, int rw,
|
||||
u64 *bytenr_ret);
|
||||
int btrfs_sb_log_location(struct btrfs_device *device, int mirror, int rw,
|
||||
@ -59,8 +59,6 @@ int btrfs_reset_device_zone(struct btrfs_device *device, u64 physical,
|
||||
int btrfs_ensure_empty_zones(struct btrfs_device *device, u64 start, u64 size);
|
||||
int btrfs_load_block_group_zone_info(struct btrfs_block_group *cache, bool new);
|
||||
void btrfs_calc_zone_unusable(struct btrfs_block_group *cache);
|
||||
void btrfs_redirty_list_add(struct btrfs_transaction *trans,
|
||||
struct extent_buffer *eb);
|
||||
bool btrfs_use_zone_append(struct btrfs_bio *bbio);
|
||||
void btrfs_record_physical_zoned(struct btrfs_bio *bbio);
|
||||
int btrfs_check_meta_write_pointer(struct btrfs_fs_info *fs_info,
|
||||
@ -123,7 +121,8 @@ static inline int btrfs_check_zoned_mode(const struct btrfs_fs_info *fs_info)
|
||||
return -EOPNOTSUPP;
|
||||
}
|
||||
|
||||
static inline int btrfs_check_mountopts_zoned(struct btrfs_fs_info *info)
|
||||
static inline int btrfs_check_mountopts_zoned(struct btrfs_fs_info *info,
|
||||
unsigned long *mount_opt)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
@ -180,9 +179,6 @@ static inline int btrfs_load_block_group_zone_info(
|
||||
|
||||
static inline void btrfs_calc_zone_unusable(struct btrfs_block_group *cache) { }
|
||||
|
||||
static inline void btrfs_redirty_list_add(struct btrfs_transaction *trans,
|
||||
struct extent_buffer *eb) { }
|
||||
|
||||
static inline bool btrfs_use_zone_append(struct btrfs_bio *bbio)
|
||||
{
|
||||
return false;
|
||||
@ -323,7 +319,7 @@ static inline bool btrfs_check_device_zone_type(const struct btrfs_fs_info *fs_i
|
||||
(bdev_zone_sectors(bdev) << SECTOR_SHIFT);
|
||||
}
|
||||
|
||||
/* Do not allow Host Manged zoned device */
|
||||
/* Do not allow Host Managed zoned device. */
|
||||
return bdev_zoned_model(bdev) != BLK_ZONED_HM;
|
||||
}
|
||||
|
||||
|
@ -410,9 +410,8 @@ int zstd_compress_pages(struct list_head *ws, struct address_space *mapping,
|
||||
workspace->in_buf.pos = 0;
|
||||
workspace->in_buf.size = min_t(size_t, len, PAGE_SIZE);
|
||||
|
||||
|
||||
/* Allocate and map in the output buffer */
|
||||
out_page = alloc_page(GFP_NOFS);
|
||||
out_page = btrfs_alloc_compr_page();
|
||||
if (out_page == NULL) {
|
||||
ret = -ENOMEM;
|
||||
goto out;
|
||||
@ -457,7 +456,7 @@ int zstd_compress_pages(struct list_head *ws, struct address_space *mapping,
|
||||
ret = -E2BIG;
|
||||
goto out;
|
||||
}
|
||||
out_page = alloc_page(GFP_NOFS);
|
||||
out_page = btrfs_alloc_compr_page();
|
||||
if (out_page == NULL) {
|
||||
ret = -ENOMEM;
|
||||
goto out;
|
||||
@ -514,7 +513,7 @@ int zstd_compress_pages(struct list_head *ws, struct address_space *mapping,
|
||||
ret = -E2BIG;
|
||||
goto out;
|
||||
}
|
||||
out_page = alloc_page(GFP_NOFS);
|
||||
out_page = btrfs_alloc_compr_page();
|
||||
if (out_page == NULL) {
|
||||
ret = -ENOMEM;
|
||||
goto out;
|
||||
|
@ -2878,7 +2878,12 @@ static int do_remount(struct path *path, int ms_flags, int sb_flags,
|
||||
if (IS_ERR(fc))
|
||||
return PTR_ERR(fc);
|
||||
|
||||
/*
|
||||
* Indicate to the filesystem that the remount request is coming
|
||||
* from the legacy mount system call.
|
||||
*/
|
||||
fc->oldapi = true;
|
||||
|
||||
err = parse_monolithic_mount_data(fc, data);
|
||||
if (!err) {
|
||||
down_write(&sb->s_umount);
|
||||
@ -3328,6 +3333,12 @@ static int do_new_mount(struct path *path, const char *fstype, int sb_flags,
|
||||
if (IS_ERR(fc))
|
||||
return PTR_ERR(fc);
|
||||
|
||||
/*
|
||||
* Indicate to the filesystem that the mount request is coming
|
||||
* from the legacy mount system call.
|
||||
*/
|
||||
fc->oldapi = true;
|
||||
|
||||
if (subtype)
|
||||
err = vfs_parse_fs_string(fc, "subtype",
|
||||
subtype, strlen(subtype));
|
||||
|
@ -21,7 +21,7 @@ struct btrfs_delayed_data_ref;
|
||||
struct btrfs_delayed_ref_head;
|
||||
struct btrfs_block_group;
|
||||
struct btrfs_free_cluster;
|
||||
struct map_lookup;
|
||||
struct btrfs_chunk_map;
|
||||
struct extent_buffer;
|
||||
struct btrfs_work;
|
||||
struct btrfs_workqueue;
|
||||
@ -265,20 +265,20 @@ DEFINE_EVENT(btrfs__inode, btrfs_inode_evict,
|
||||
__print_symbolic_u64(type, \
|
||||
{ EXTENT_MAP_LAST_BYTE, "LAST_BYTE" }, \
|
||||
{ EXTENT_MAP_HOLE, "HOLE" }, \
|
||||
{ EXTENT_MAP_INLINE, "INLINE" }, \
|
||||
{ EXTENT_MAP_DELALLOC, "DELALLOC" })
|
||||
{ EXTENT_MAP_INLINE, "INLINE" })
|
||||
|
||||
#define show_map_type(type) \
|
||||
type, (type >= EXTENT_MAP_LAST_BYTE) ? "-" : __show_map_type(type)
|
||||
|
||||
#define show_map_flags(flag) \
|
||||
__print_flags(flag, "|", \
|
||||
{ (1 << EXTENT_FLAG_PINNED), "PINNED" },\
|
||||
{ (1 << EXTENT_FLAG_COMPRESSED), "COMPRESSED" },\
|
||||
{ (1 << EXTENT_FLAG_PREALLOC), "PREALLOC" },\
|
||||
{ (1 << EXTENT_FLAG_LOGGING), "LOGGING" },\
|
||||
{ (1 << EXTENT_FLAG_FILLING), "FILLING" },\
|
||||
{ (1 << EXTENT_FLAG_FS_MAPPING), "FS_MAPPING" })
|
||||
{ EXTENT_FLAG_PINNED, "PINNED" },\
|
||||
{ EXTENT_FLAG_COMPRESS_ZLIB, "COMPRESS_ZLIB" },\
|
||||
{ EXTENT_FLAG_COMPRESS_LZO, "COMPRESS_LZO" },\
|
||||
{ EXTENT_FLAG_COMPRESS_ZSTD, "COMPRESS_ZSTD" },\
|
||||
{ EXTENT_FLAG_PREALLOC, "PREALLOC" },\
|
||||
{ EXTENT_FLAG_LOGGING, "LOGGING" },\
|
||||
{ EXTENT_FLAG_FILLING, "FILLING" })
|
||||
|
||||
TRACE_EVENT_CONDITION(btrfs_get_extent,
|
||||
|
||||
@ -297,9 +297,8 @@ TRACE_EVENT_CONDITION(btrfs_get_extent,
|
||||
__field( u64, orig_start )
|
||||
__field( u64, block_start )
|
||||
__field( u64, block_len )
|
||||
__field( unsigned long, flags )
|
||||
__field( u32, flags )
|
||||
__field( int, refs )
|
||||
__field( unsigned int, compress_type )
|
||||
),
|
||||
|
||||
TP_fast_assign_btrfs(root->fs_info,
|
||||
@ -312,13 +311,11 @@ TRACE_EVENT_CONDITION(btrfs_get_extent,
|
||||
__entry->block_len = map->block_len;
|
||||
__entry->flags = map->flags;
|
||||
__entry->refs = refcount_read(&map->refs);
|
||||
__entry->compress_type = map->compress_type;
|
||||
),
|
||||
|
||||
TP_printk_btrfs("root=%llu(%s) ino=%llu start=%llu len=%llu "
|
||||
"orig_start=%llu block_start=%llu(%s) "
|
||||
"block_len=%llu flags=%s refs=%u "
|
||||
"compress_type=%u",
|
||||
"block_len=%llu flags=%s refs=%u",
|
||||
show_root_type(__entry->root_objectid),
|
||||
__entry->ino,
|
||||
__entry->start,
|
||||
@ -327,7 +324,7 @@ TRACE_EVENT_CONDITION(btrfs_get_extent,
|
||||
show_map_type(__entry->block_start),
|
||||
__entry->block_len,
|
||||
show_map_flags(__entry->flags),
|
||||
__entry->refs, __entry->compress_type)
|
||||
__entry->refs)
|
||||
);
|
||||
|
||||
TRACE_EVENT(btrfs_handle_em_exist,
|
||||
@ -1061,7 +1058,7 @@ DEFINE_EVENT(btrfs_delayed_ref_head, run_delayed_ref_head,
|
||||
DECLARE_EVENT_CLASS(btrfs__chunk,
|
||||
|
||||
TP_PROTO(const struct btrfs_fs_info *fs_info,
|
||||
const struct map_lookup *map, u64 offset, u64 size),
|
||||
const struct btrfs_chunk_map *map, u64 offset, u64 size),
|
||||
|
||||
TP_ARGS(fs_info, map, offset, size),
|
||||
|
||||
@ -1095,7 +1092,7 @@ DECLARE_EVENT_CLASS(btrfs__chunk,
|
||||
DEFINE_EVENT(btrfs__chunk, btrfs_chunk_alloc,
|
||||
|
||||
TP_PROTO(const struct btrfs_fs_info *fs_info,
|
||||
const struct map_lookup *map, u64 offset, u64 size),
|
||||
const struct btrfs_chunk_map *map, u64 offset, u64 size),
|
||||
|
||||
TP_ARGS(fs_info, map, offset, size)
|
||||
);
|
||||
@ -1103,7 +1100,7 @@ DEFINE_EVENT(btrfs__chunk, btrfs_chunk_alloc,
|
||||
DEFINE_EVENT(btrfs__chunk, btrfs_chunk_free,
|
||||
|
||||
TP_PROTO(const struct btrfs_fs_info *fs_info,
|
||||
const struct map_lookup *map, u64 offset, u64 size),
|
||||
const struct btrfs_chunk_map *map, u64 offset, u64 size),
|
||||
|
||||
TP_ARGS(fs_info, map, offset, size)
|
||||
);
|
||||
@ -2099,17 +2096,12 @@ TRACE_EVENT(btrfs_set_extent_bit,
|
||||
__field( unsigned, set_bits)
|
||||
),
|
||||
|
||||
TP_fast_assign_btrfs(tree->fs_info,
|
||||
__entry->owner = tree->owner;
|
||||
if (tree->inode) {
|
||||
const struct btrfs_inode *inode = tree->inode;
|
||||
TP_fast_assign_btrfs(extent_io_tree_to_fs_info(tree),
|
||||
const struct btrfs_inode *inode = extent_io_tree_to_inode_const(tree);
|
||||
|
||||
__entry->ino = btrfs_ino(inode);
|
||||
__entry->rootid = inode->root->root_key.objectid;
|
||||
} else {
|
||||
__entry->ino = 0;
|
||||
__entry->rootid = 0;
|
||||
}
|
||||
__entry->owner = tree->owner;
|
||||
__entry->ino = inode ? btrfs_ino(inode) : 0;
|
||||
__entry->rootid = inode ? inode->root->root_key.objectid : 0;
|
||||
__entry->start = start;
|
||||
__entry->len = len;
|
||||
__entry->set_bits = set_bits;
|
||||
@ -2137,17 +2129,12 @@ TRACE_EVENT(btrfs_clear_extent_bit,
|
||||
__field( unsigned, clear_bits)
|
||||
),
|
||||
|
||||
TP_fast_assign_btrfs(tree->fs_info,
|
||||
__entry->owner = tree->owner;
|
||||
if (tree->inode) {
|
||||
const struct btrfs_inode *inode = tree->inode;
|
||||
TP_fast_assign_btrfs(extent_io_tree_to_fs_info(tree),
|
||||
const struct btrfs_inode *inode = extent_io_tree_to_inode_const(tree);
|
||||
|
||||
__entry->ino = btrfs_ino(inode);
|
||||
__entry->rootid = inode->root->root_key.objectid;
|
||||
} else {
|
||||
__entry->ino = 0;
|
||||
__entry->rootid = 0;
|
||||
}
|
||||
__entry->owner = tree->owner;
|
||||
__entry->ino = inode ? btrfs_ino(inode) : 0;
|
||||
__entry->rootid = inode ? inode->root->root_key.objectid : 0;
|
||||
__entry->start = start;
|
||||
__entry->len = len;
|
||||
__entry->clear_bits = clear_bits;
|
||||
@ -2176,17 +2163,12 @@ TRACE_EVENT(btrfs_convert_extent_bit,
|
||||
__field( unsigned, clear_bits)
|
||||
),
|
||||
|
||||
TP_fast_assign_btrfs(tree->fs_info,
|
||||
__entry->owner = tree->owner;
|
||||
if (tree->inode) {
|
||||
const struct btrfs_inode *inode = tree->inode;
|
||||
TP_fast_assign_btrfs(extent_io_tree_to_fs_info(tree),
|
||||
const struct btrfs_inode *inode = extent_io_tree_to_inode_const(tree);
|
||||
|
||||
__entry->ino = btrfs_ino(inode);
|
||||
__entry->rootid = inode->root->root_key.objectid;
|
||||
} else {
|
||||
__entry->ino = 0;
|
||||
__entry->rootid = 0;
|
||||
}
|
||||
__entry->owner = tree->owner;
|
||||
__entry->ino = inode ? btrfs_ino(inode) : 0;
|
||||
__entry->rootid = inode ? inode->root->root_key.objectid : 0;
|
||||
__entry->start = start;
|
||||
__entry->len = len;
|
||||
__entry->set_bits = set_bits;
|
||||
|
Loading…
x
Reference in New Issue
Block a user