Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
L
linux
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
linux
Commits
7f5e31e1
Commit
7f5e31e1
authored
Jul 29, 2019
by
Kent Overstreet
Committed by
Kent Overstreet
Oct 22, 2023
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
bcachefs: Change __bch2_writepage() to not write to holes
Signed-off-by:
Kent Overstreet
<
kent.overstreet@linux.dev
>
parent
e10d3094
Changes
1
Hide whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
86 additions
and
56 deletions
+86
-56
fs/bcachefs/fs-io.c
fs/bcachefs/fs-io.c
+86
-56
No files found.
fs/bcachefs/fs-io.c
View file @
7f5e31e1
...
@@ -31,6 +31,15 @@
...
@@ -31,6 +31,15 @@
#include <trace/events/writeback.h>
#include <trace/events/writeback.h>
static
inline
bool
bio_full
(
struct
bio
*
bio
,
unsigned
len
)
{
if
(
bio
->
bi_vcnt
>=
bio
->
bi_max_vecs
)
return
true
;
if
(
bio
->
bi_iter
.
bi_size
>
UINT_MAX
-
len
)
return
true
;
return
false
;
}
struct
quota_res
{
struct
quota_res
{
u64
sectors
;
u64
sectors
;
};
};
...
@@ -517,6 +526,7 @@ struct bch_page_sector {
...
@@ -517,6 +526,7 @@ struct bch_page_sector {
};
};
struct
bch_page_state
{
struct
bch_page_state
{
atomic_t
write_count
;
struct
bch_page_sector
s
[
PAGE_SECTORS
];
struct
bch_page_sector
s
[
PAGE_SECTORS
];
};
};
...
@@ -835,31 +845,6 @@ bool bch2_release_folio(struct folio *folio, gfp_t gfp_mask)
...
@@ -835,31 +845,6 @@ bool bch2_release_folio(struct folio *folio, gfp_t gfp_mask)
return
true
;
return
true
;
}
}
/* readpages/writepages: */
static
bool
bio_can_add_page_contig
(
struct
bio
*
bio
,
struct
page
*
page
)
{
sector_t
offset
=
(
sector_t
)
page
->
index
<<
PAGE_SECTOR_SHIFT
;
return
bio
->
bi_vcnt
<
bio
->
bi_max_vecs
&&
bio_end_sector
(
bio
)
==
offset
;
}
static
int
bio_add_page_contig
(
struct
bio
*
bio
,
struct
page
*
page
)
{
sector_t
offset
=
(
sector_t
)
page
->
index
<<
PAGE_SECTOR_SHIFT
;
EBUG_ON
(
!
bio
->
bi_max_vecs
);
if
(
!
bio
->
bi_vcnt
)
bio
->
bi_iter
.
bi_sector
=
offset
;
else
if
(
!
bio_can_add_page_contig
(
bio
,
page
))
return
-
1
;
BUG_ON
(
!
bio_add_page
(
bio
,
page
,
PAGE_SIZE
,
0
));
return
0
;
}
/* readpage(s): */
/* readpage(s): */
static
void
bch2_readpages_end_io
(
struct
bio
*
bio
)
static
void
bch2_readpages_end_io
(
struct
bio
*
bio
)
...
@@ -1132,7 +1117,9 @@ static void __bchfs_readpage(struct bch_fs *c, struct bch_read_bio *rbio,
...
@@ -1132,7 +1117,9 @@ static void __bchfs_readpage(struct bch_fs *c, struct bch_read_bio *rbio,
bch2_page_state_create
(
page
,
__GFP_NOFAIL
);
bch2_page_state_create
(
page
,
__GFP_NOFAIL
);
rbio
->
bio
.
bi_opf
=
REQ_OP_READ
|
REQ_SYNC
;
rbio
->
bio
.
bi_opf
=
REQ_OP_READ
|
REQ_SYNC
;
bio_add_page_contig
(
&
rbio
->
bio
,
page
);
rbio
->
bio
.
bi_iter
.
bi_sector
=
(
sector_t
)
page
->
index
<<
PAGE_SECTOR_SHIFT
;
BUG_ON
(
!
bio_add_page
(
&
rbio
->
bio
,
page
,
PAGE_SIZE
,
0
));
bch2_trans_init
(
&
trans
,
c
,
0
,
0
);
bch2_trans_init
(
&
trans
,
c
,
0
,
0
);
iter
=
bch2_trans_get_iter
(
&
trans
,
BTREE_ID_EXTENTS
,
POS_MIN
,
iter
=
bch2_trans_get_iter
(
&
trans
,
BTREE_ID_EXTENTS
,
POS_MIN
,
...
@@ -1243,8 +1230,12 @@ static void bch2_writepage_io_done(struct closure *cl)
...
@@ -1243,8 +1230,12 @@ static void bch2_writepage_io_done(struct closure *cl)
i_sectors_acct
(
c
,
io
->
op
.
inode
,
NULL
,
i_sectors_acct
(
c
,
io
->
op
.
inode
,
NULL
,
io
->
op
.
sectors_added
-
(
s64
)
io
->
new_sectors
);
io
->
op
.
sectors_added
-
(
s64
)
io
->
new_sectors
);
bio_for_each_segment_all
(
bvec
,
bio
,
iter
)
bio_for_each_segment_all
(
bvec
,
bio
,
iter
)
{
end_page_writeback
(
bvec
->
bv_page
);
struct
bch_page_state
*
s
=
__bch2_page_state
(
bvec
->
bv_page
);
if
(
atomic_dec_and_test
(
&
s
->
write_count
))
end_page_writeback
(
bvec
->
bv_page
);
}
closure_return_with_destructor
(
&
io
->
cl
,
bch2_writepage_io_free
);
closure_return_with_destructor
(
&
io
->
cl
,
bch2_writepage_io_free
);
}
}
...
@@ -1265,11 +1256,10 @@ static void bch2_writepage_do_io(struct bch_writepage_state *w)
...
@@ -1265,11 +1256,10 @@ static void bch2_writepage_do_io(struct bch_writepage_state *w)
static
void
bch2_writepage_io_alloc
(
struct
bch_fs
*
c
,
static
void
bch2_writepage_io_alloc
(
struct
bch_fs
*
c
,
struct
bch_writepage_state
*
w
,
struct
bch_writepage_state
*
w
,
struct
bch_inode_info
*
inode
,
struct
bch_inode_info
*
inode
,
struct
page
*
page
,
u64
sector
,
unsigned
nr_replicas
)
unsigned
nr_replicas
)
{
{
struct
bch_write_op
*
op
;
struct
bch_write_op
*
op
;
u64
offset
=
(
u64
)
page
->
index
<<
PAGE_SECTOR_SHIFT
;
w
->
io
=
container_of
(
bio_alloc_bioset
(
NULL
,
BIO_MAX_VECS
,
w
->
io
=
container_of
(
bio_alloc_bioset
(
NULL
,
BIO_MAX_VECS
,
REQ_OP_WRITE
,
REQ_OP_WRITE
,
...
@@ -1284,8 +1274,8 @@ static void bch2_writepage_io_alloc(struct bch_fs *c,
...
@@ -1284,8 +1274,8 @@ static void bch2_writepage_io_alloc(struct bch_fs *c,
op
->
nr_replicas
=
nr_replicas
;
op
->
nr_replicas
=
nr_replicas
;
op
->
res
.
nr_replicas
=
nr_replicas
;
op
->
res
.
nr_replicas
=
nr_replicas
;
op
->
write_point
=
writepoint_hashed
(
inode
->
ei_last_dirtied
);
op
->
write_point
=
writepoint_hashed
(
inode
->
ei_last_dirtied
);
op
->
pos
=
POS
(
inode
->
v
.
i_ino
,
offset
);
op
->
pos
=
POS
(
inode
->
v
.
i_ino
,
sector
);
op
->
wbio
.
bio
.
bi_iter
.
bi_sector
=
offset
;
op
->
wbio
.
bio
.
bi_iter
.
bi_sector
=
sector
;
}
}
static
int
__bch2_writepage
(
struct
folio
*
folio
,
static
int
__bch2_writepage
(
struct
folio
*
folio
,
...
@@ -1296,12 +1286,10 @@ static int __bch2_writepage(struct folio *folio,
...
@@ -1296,12 +1286,10 @@ static int __bch2_writepage(struct folio *folio,
struct
bch_inode_info
*
inode
=
to_bch_ei
(
page
->
mapping
->
host
);
struct
bch_inode_info
*
inode
=
to_bch_ei
(
page
->
mapping
->
host
);
struct
bch_fs
*
c
=
inode
->
v
.
i_sb
->
s_fs_info
;
struct
bch_fs
*
c
=
inode
->
v
.
i_sb
->
s_fs_info
;
struct
bch_writepage_state
*
w
=
data
;
struct
bch_writepage_state
*
w
=
data
;
struct
bch_page_state
*
s
;
struct
bch_page_state
*
s
,
orig
;
unsigned
offset
,
nr_replicas_this_write
=
U32_MAX
;
unsigned
i
,
offset
,
nr_replicas_this_write
=
U32_MAX
;
unsigned
dirty_sectors
=
0
,
reserved_sectors
=
0
;
loff_t
i_size
=
i_size_read
(
&
inode
->
v
);
loff_t
i_size
=
i_size_read
(
&
inode
->
v
);
pgoff_t
end_index
=
i_size
>>
PAGE_SHIFT
;
pgoff_t
end_index
=
i_size
>>
PAGE_SHIFT
;
unsigned
i
;
int
ret
;
int
ret
;
EBUG_ON
(
!
PageUptodate
(
page
));
EBUG_ON
(
!
PageUptodate
(
page
));
...
@@ -1336,48 +1324,90 @@ static int __bch2_writepage(struct folio *folio,
...
@@ -1336,48 +1324,90 @@ static int __bch2_writepage(struct folio *folio,
return
0
;
return
0
;
}
}
for
(
i
=
0
;
i
<
PAGE_SECTORS
;
i
++
)
/* Before unlocking the page, get copy of reservations: */
orig
=
*
s
;
for
(
i
=
0
;
i
<
PAGE_SECTORS
;
i
++
)
{
if
(
s
->
s
[
i
].
state
==
SECTOR_UNALLOCATED
)
continue
;
nr_replicas_this_write
=
nr_replicas_this_write
=
min_t
(
unsigned
,
nr_replicas_this_write
,
min_t
(
unsigned
,
nr_replicas_this_write
,
s
->
s
[
i
].
nr_replicas
+
s
->
s
[
i
].
nr_replicas
+
s
->
s
[
i
].
replicas_reserved
);
s
->
s
[
i
].
replicas_reserved
);
}
/* Before unlocking the page, transfer reservation to w->io: */
for
(
i
=
0
;
i
<
PAGE_SECTORS
;
i
++
)
{
for
(
i
=
0
;
i
<
PAGE_SECTORS
;
i
++
)
{
if
(
s
->
s
[
i
].
state
==
SECTOR_UNALLOCATED
)
continue
;
s
->
s
[
i
].
nr_replicas
=
w
->
opts
.
compression
s
->
s
[
i
].
nr_replicas
=
w
->
opts
.
compression
?
0
:
nr_replicas_this_write
;
?
0
:
nr_replicas_this_write
;
reserved_sectors
+=
s
->
s
[
i
].
replicas_reserved
;
s
->
s
[
i
].
replicas_reserved
=
0
;
s
->
s
[
i
].
replicas_reserved
=
0
;
dirty_sectors
+=
s
->
s
[
i
].
state
==
SECTOR_DIRTY
;
s
->
s
[
i
].
state
=
SECTOR_ALLOCATED
;
s
->
s
[
i
].
state
=
SECTOR_ALLOCATED
;
}
}
BUG_ON
(
atomic_read
(
&
s
->
write_count
));
atomic_set
(
&
s
->
write_count
,
1
);
BUG_ON
(
PageWriteback
(
page
));
BUG_ON
(
PageWriteback
(
page
));
set_page_writeback
(
page
);
set_page_writeback
(
page
);
unlock_page
(
page
);
unlock_page
(
page
);
if
(
w
->
io
&&
offset
=
0
;
(
w
->
io
->
op
.
op
.
res
.
nr_replicas
!=
nr_replicas_this_write
||
while
(
1
)
{
!
bio_can_add_page_contig
(
&
w
->
io
->
op
.
op
.
wbio
.
bio
,
page
)))
unsigned
sectors
=
1
,
dirty_sectors
=
0
,
reserved_sectors
=
0
;
bch2_writepage_do_io
(
w
);
u64
sector
;
while
(
offset
<
PAGE_SECTORS
&&
orig
.
s
[
offset
].
state
==
SECTOR_UNALLOCATED
)
offset
++
;
if
(
offset
==
PAGE_SECTORS
)
break
;
sector
=
((
u64
)
page
->
index
<<
PAGE_SECTOR_SHIFT
)
+
offset
;
while
(
offset
+
sectors
<
PAGE_SECTORS
&&
orig
.
s
[
offset
+
sectors
].
state
!=
SECTOR_UNALLOCATED
)
sectors
++
;
for
(
i
=
offset
;
i
<
offset
+
sectors
;
i
++
)
{
reserved_sectors
+=
orig
.
s
[
i
].
replicas_reserved
;
dirty_sectors
+=
orig
.
s
[
i
].
state
==
SECTOR_DIRTY
;
}
if
(
w
->
io
&&
(
w
->
io
->
op
.
op
.
res
.
nr_replicas
!=
nr_replicas_this_write
||
bio_full
(
&
w
->
io
->
op
.
op
.
wbio
.
bio
,
PAGE_SIZE
)
||
bio_end_sector
(
&
w
->
io
->
op
.
op
.
wbio
.
bio
)
!=
sector
))
bch2_writepage_do_io
(
w
);
if
(
!
w
->
io
)
bch2_writepage_io_alloc
(
c
,
w
,
inode
,
sector
,
nr_replicas_this_write
);
if
(
!
w
->
io
)
w
->
io
->
new_sectors
+=
dirty_sectors
;
bch2_writepage_io_alloc
(
c
,
w
,
inode
,
page
,
nr_replicas_this_write
);
w
->
io
->
new_sectors
+=
dirty_sectors
;
atomic_inc
(
&
s
->
write_count
)
;
BUG_ON
(
inode
!=
w
->
io
->
op
.
inode
);
BUG_ON
(
inode
!=
w
->
io
->
op
.
inode
);
BUG_ON
(
bio_add_page_contig
(
&
w
->
io
->
op
.
op
.
wbio
.
bio
,
page
));
BUG_ON
(
!
bio_add_page
(
&
w
->
io
->
op
.
op
.
wbio
.
bio
,
page
,
sectors
<<
9
,
offset
<<
9
));
w
->
io
->
op
.
op
.
res
.
sectors
+=
reserved_sectors
;
w
->
io
->
op
.
op
.
res
.
sectors
+=
reserved_sectors
;
w
->
io
->
op
.
new_i_size
=
i_size
;
w
->
io
->
op
.
new_i_size
=
i_size
;
if
(
wbc
->
sync_mode
==
WB_SYNC_ALL
)
w
->
io
->
op
.
op
.
wbio
.
bio
.
bi_opf
|=
REQ_SYNC
;
offset
+=
sectors
;
}
if
(
wbc
->
sync_mode
==
WB_SYNC_ALL
)
if
(
atomic_dec_and_test
(
&
s
->
write_count
)
)
w
->
io
->
op
.
op
.
wbio
.
bio
.
bi_opf
|=
REQ_SYNC
;
end_page_writeback
(
page
)
;
return
0
;
return
0
;
}
}
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment