Skip to content

Commit 4a79616

Browse files
committed
netfs: Provide a launder_folio implementation
Provide a launder_folio implementation for netfslib. Signed-off-by: David Howells <dhowells@redhat.com> Reviewed-by: Jeff Layton <jlayton@kernel.org> cc: linux-cachefs@redhat.com cc: linux-fsdevel@vger.kernel.org cc: linux-mm@kvack.org
1 parent 62c3b74 commit 4a79616

File tree

4 files changed

+80
-0
lines changed

4 files changed

+80
-0
lines changed

fs/netfs/buffered_write.c

Lines changed: 74 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -1111,3 +1111,77 @@ int netfs_writepages(struct address_space *mapping,
11111111
return ret;
11121112
}
11131113
EXPORT_SYMBOL(netfs_writepages);
1114+
1115+
/*
1116+
* Deal with the disposition of a laundered folio.
1117+
*/
1118+
static void netfs_cleanup_launder_folio(struct netfs_io_request *wreq)
1119+
{
1120+
if (wreq->error) {
1121+
pr_notice("R=%08x Laundering error %d\n", wreq->debug_id, wreq->error);
1122+
mapping_set_error(wreq->mapping, wreq->error);
1123+
}
1124+
}
1125+
1126+
/**
1127+
* netfs_launder_folio - Clean up a dirty folio that's being invalidated
1128+
* @folio: The folio to clean
1129+
*
1130+
* This is called to write back a folio that's being invalidated when an inode
1131+
* is getting torn down. Ideally, writepages would be used instead.
1132+
*/
1133+
int netfs_launder_folio(struct folio *folio)
1134+
{
1135+
struct netfs_io_request *wreq;
1136+
struct address_space *mapping = folio->mapping;
1137+
struct netfs_folio *finfo = netfs_folio_info(folio);
1138+
struct netfs_group *group = netfs_folio_group(folio);
1139+
struct bio_vec bvec;
1140+
unsigned long long i_size = i_size_read(mapping->host);
1141+
unsigned long long start = folio_pos(folio);
1142+
size_t offset = 0, len;
1143+
int ret = 0;
1144+
1145+
if (finfo) {
1146+
offset = finfo->dirty_offset;
1147+
start += offset;
1148+
len = finfo->dirty_len;
1149+
} else {
1150+
len = folio_size(folio);
1151+
}
1152+
len = min_t(unsigned long long, len, i_size - start);
1153+
1154+
wreq = netfs_alloc_request(mapping, NULL, start, len, NETFS_LAUNDER_WRITE);
1155+
if (IS_ERR(wreq)) {
1156+
ret = PTR_ERR(wreq);
1157+
goto out;
1158+
}
1159+
1160+
if (!folio_clear_dirty_for_io(folio))
1161+
goto out_put;
1162+
1163+
trace_netfs_folio(folio, netfs_folio_trace_launder);
1164+
1165+
_debug("launder %llx-%llx", start, start + len - 1);
1166+
1167+
/* Speculatively write to the cache. We have to fix this up later if
1168+
* the store fails.
1169+
*/
1170+
wreq->cleanup = netfs_cleanup_launder_folio;
1171+
1172+
bvec_set_folio(&bvec, folio, len, offset);
1173+
iov_iter_bvec(&wreq->iter, ITER_SOURCE, &bvec, 1, len);
1174+
__set_bit(NETFS_RREQ_UPLOAD_TO_SERVER, &wreq->flags);
1175+
ret = netfs_begin_write(wreq, true, netfs_write_trace_launder);
1176+
1177+
out_put:
1178+
folio_detach_private(folio);
1179+
netfs_put_group(group);
1180+
kfree(finfo);
1181+
netfs_put_request(wreq, false, netfs_rreq_trace_put_return);
1182+
out:
1183+
folio_wait_fscache(folio);
1184+
_leave(" = %d", ret);
1185+
return ret;
1186+
}
1187+
EXPORT_SYMBOL(netfs_launder_folio);

fs/netfs/main.c

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -30,6 +30,7 @@ static const char *netfs_origins[nr__netfs_io_origin] = {
3030
[NETFS_READPAGE] = "RP",
3131
[NETFS_READ_FOR_WRITE] = "RW",
3232
[NETFS_WRITEBACK] = "WB",
33+
[NETFS_LAUNDER_WRITE] = "LW",
3334
[NETFS_UNBUFFERED_WRITE] = "UW",
3435
[NETFS_DIO_READ] = "DR",
3536
[NETFS_DIO_WRITE] = "DW",

include/linux/netfs.h

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -227,6 +227,7 @@ enum netfs_io_origin {
227227
NETFS_READPAGE, /* This read is a synchronous read */
228228
NETFS_READ_FOR_WRITE, /* This read is to prepare a write */
229229
NETFS_WRITEBACK, /* This write was triggered by writepages */
230+
NETFS_LAUNDER_WRITE, /* This is triggered by ->launder_folio() */
230231
NETFS_UNBUFFERED_WRITE, /* This is an unbuffered write */
231232
NETFS_DIO_READ, /* This is a direct I/O read */
232233
NETFS_DIO_WRITE, /* This is a direct I/O write */
@@ -404,6 +405,7 @@ int netfs_unpin_writeback(struct inode *inode, struct writeback_control *wbc);
404405
void netfs_clear_inode_writeback(struct inode *inode, const void *aux);
405406
void netfs_invalidate_folio(struct folio *folio, size_t offset, size_t length);
406407
bool netfs_release_folio(struct folio *folio, gfp_t gfp);
408+
int netfs_launder_folio(struct folio *folio);
407409

408410
/* VMA operations API. */
409411
vm_fault_t netfs_page_mkwrite(struct vm_fault *vmf, struct netfs_group *netfs_group);

include/trace/events/netfs.h

Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,7 @@
2525

2626
#define netfs_write_traces \
2727
EM(netfs_write_trace_dio_write, "DIO-WRITE") \
28+
EM(netfs_write_trace_launder, "LAUNDER ") \
2829
EM(netfs_write_trace_unbuffered_write, "UNB-WRITE") \
2930
E_(netfs_write_trace_writeback, "WRITEBACK")
3031

@@ -33,6 +34,7 @@
3334
EM(NETFS_READPAGE, "RP") \
3435
EM(NETFS_READ_FOR_WRITE, "RW") \
3536
EM(NETFS_WRITEBACK, "WB") \
37+
EM(NETFS_LAUNDER_WRITE, "LW") \
3638
EM(NETFS_UNBUFFERED_WRITE, "UW") \
3739
EM(NETFS_DIO_READ, "DR") \
3840
E_(NETFS_DIO_WRITE, "DW")
@@ -127,6 +129,7 @@
127129
EM(netfs_folio_trace_end_copy, "end-copy") \
128130
EM(netfs_folio_trace_filled_gaps, "filled-gaps") \
129131
EM(netfs_folio_trace_kill, "kill") \
132+
EM(netfs_folio_trace_launder, "launder") \
130133
EM(netfs_folio_trace_mkwrite, "mkwrite") \
131134
EM(netfs_folio_trace_mkwrite_plus, "mkwrite+") \
132135
EM(netfs_folio_trace_read_gaps, "read-gaps") \

0 commit comments

Comments
 (0)