netfs: Change the read result collector to only use one work item

Change the way netfslib collects read results to do all the collection for
a particular read request using a single work item that walks along the
subrequest queue as subrequests make progress or complete, unlocking folios
progressively rather than doing the unlock in parallel as parallel requests
come in.

The code is remodelled to be more like the write-side code, though only
using a single stream.  This makes it more directly comparable and thus
easier to duplicate fixes between the two sides.

This has a number of advantages:

 (1) It's simpler.  There doesn't need to be a complex donation mechanism
     to handle mismatches between the size and alignment of subrequests and
     folios.  The collector unlocks folios as the subrequests covering each
     complete.

 (2) It should cause less scheduler overhead as there's a single work item
     in play unlocking pages in parallel when a read gets split up into a
     lot of subrequests instead of one per subrequest.

     Whilst the parallellism is nice in theory, in practice, the vast
     majority of loads are sequential reads of the whole file, so
     committing a bunch of threads to unlocking folios out of order doesn't
     help in those cases.

 (3) It should make it easier to implement content decryption.  A folio
     cannot be decrypted until all the requests that contribute to it have
     completed - and, again, most loads are sequential and so, most of the
     time, we want to begin decryption sequentially (though it's great if
     the decryption can happen in parallel).

There is a disadvantage in that we're losing the ability to decrypt and
unlock things on an as-things-arrive basis which may affect some
applications.

Signed-off-by: David Howells <dhowells@redhat.com>
Link: https://lore.kernel.org/r/20241108173236.1382366-29-dhowells@redhat.com
cc: Jeff Layton <jlayton@kernel.org>
cc: netfs@lists.linux.dev
cc: linux-fsdevel@vger.kernel.org
Signed-off-by: Christian Brauner <brauner@kernel.org>
This commit is contained in:
David Howells 2024-11-08 17:32:29 +00:00 committed by Christian Brauner
parent 8942f9db28
commit b4f239c91f
No known key found for this signature in database
GPG Key ID: 91C61BC06578DCA2
16 changed files with 669 additions and 617 deletions

View File

@ -79,9 +79,10 @@ static void v9fs_issue_read(struct netfs_io_subrequest *subreq)
__set_bit(NETFS_SREQ_CLEAR_TAIL, &subreq->flags); __set_bit(NETFS_SREQ_CLEAR_TAIL, &subreq->flags);
if (pos + total >= i_size_read(rreq->inode)) if (pos + total >= i_size_read(rreq->inode))
__set_bit(NETFS_SREQ_HIT_EOF, &subreq->flags); __set_bit(NETFS_SREQ_HIT_EOF, &subreq->flags);
if (!err && total) {
if (!err) __set_bit(NETFS_SREQ_MADE_PROGRESS, &subreq->flags);
subreq->transferred += total; subreq->transferred += total;
}
subreq->error = err; subreq->error = err;
netfs_read_subreq_terminated(subreq); netfs_read_subreq_terminated(subreq);

View File

@ -325,8 +325,10 @@ static ssize_t afs_read_dir(struct afs_vnode *dvnode, struct file *file)
* haven't read it yet. * haven't read it yet.
*/ */
if (test_bit(AFS_VNODE_DIR_VALID, &dvnode->flags) && if (test_bit(AFS_VNODE_DIR_VALID, &dvnode->flags) &&
test_bit(AFS_VNODE_DIR_READ, &dvnode->flags)) test_bit(AFS_VNODE_DIR_READ, &dvnode->flags)) {
ret = i_size;
goto valid; goto valid;
}
up_read(&dvnode->validate_lock); up_read(&dvnode->validate_lock);
if (down_write_killable(&dvnode->validate_lock) < 0) if (down_write_killable(&dvnode->validate_lock) < 0)
@ -346,11 +348,13 @@ static ssize_t afs_read_dir(struct afs_vnode *dvnode, struct file *file)
set_bit(AFS_VNODE_DIR_VALID, &dvnode->flags); set_bit(AFS_VNODE_DIR_VALID, &dvnode->flags);
set_bit(AFS_VNODE_DIR_READ, &dvnode->flags); set_bit(AFS_VNODE_DIR_READ, &dvnode->flags);
} else {
ret = i_size;
} }
downgrade_write(&dvnode->validate_lock); downgrade_write(&dvnode->validate_lock);
valid: valid:
return i_size; return ret;
error_unlock: error_unlock:
up_write(&dvnode->validate_lock); up_write(&dvnode->validate_lock);

View File

@ -121,12 +121,6 @@ static ssize_t netfs_prepare_read_iterator(struct netfs_io_subrequest *subreq)
subreq->io_iter = rreq->buffer.iter; subreq->io_iter = rreq->buffer.iter;
if (iov_iter_is_folioq(&subreq->io_iter)) {
subreq->curr_folioq = (struct folio_queue *)subreq->io_iter.folioq;
subreq->curr_folioq_slot = subreq->io_iter.folioq_slot;
subreq->curr_folio_order = subreq->curr_folioq->orders[subreq->curr_folioq_slot];
}
iov_iter_truncate(&subreq->io_iter, subreq->len); iov_iter_truncate(&subreq->io_iter, subreq->len);
rolling_buffer_advance(&rreq->buffer, subreq->len); rolling_buffer_advance(&rreq->buffer, subreq->len);
return subreq->len; return subreq->len;
@ -147,19 +141,6 @@ static enum netfs_io_source netfs_cache_prepare_read(struct netfs_io_request *rr
} }
void netfs_cache_read_terminated(void *priv, ssize_t transferred_or_error, bool was_async)
{
struct netfs_io_subrequest *subreq = priv;
if (transferred_or_error > 0) {
subreq->transferred += transferred_or_error;
subreq->error = 0;
} else {
subreq->error = transferred_or_error;
}
schedule_work(&subreq->work);
}
/* /*
* Issue a read against the cache. * Issue a read against the cache.
* - Eats the caller's ref on subreq. * - Eats the caller's ref on subreq.
@ -174,6 +155,47 @@ static void netfs_read_cache_to_pagecache(struct netfs_io_request *rreq,
netfs_cache_read_terminated, subreq); netfs_cache_read_terminated, subreq);
} }
static void netfs_issue_read(struct netfs_io_request *rreq,
struct netfs_io_subrequest *subreq)
{
struct netfs_io_stream *stream = &rreq->io_streams[0];
__set_bit(NETFS_SREQ_IN_PROGRESS, &subreq->flags);
/* We add to the end of the list whilst the collector may be walking
* the list. The collector only goes nextwards and uses the lock to
* remove entries off of the front.
*/
spin_lock(&rreq->lock);
list_add_tail(&subreq->rreq_link, &stream->subrequests);
if (list_is_first(&subreq->rreq_link, &stream->subrequests)) {
stream->front = subreq;
if (!stream->active) {
stream->collected_to = stream->front->start;
/* Store list pointers before active flag */
smp_store_release(&stream->active, true);
}
}
spin_unlock(&rreq->lock);
switch (subreq->source) {
case NETFS_DOWNLOAD_FROM_SERVER:
rreq->netfs_ops->issue_read(subreq);
break;
case NETFS_READ_FROM_CACHE:
netfs_read_cache_to_pagecache(rreq, subreq);
break;
default:
__set_bit(NETFS_SREQ_CLEAR_TAIL, &subreq->flags);
subreq->error = 0;
iov_iter_zero(subreq->len, &subreq->io_iter);
subreq->transferred = subreq->len;
netfs_read_subreq_terminated(subreq);
break;
}
}
/* /*
* Perform a read to the pagecache from a series of sources of different types, * Perform a read to the pagecache from a series of sources of different types,
* slicing up the region to be read according to available cache blocks and * slicing up the region to be read according to available cache blocks and
@ -186,8 +208,6 @@ static void netfs_read_to_pagecache(struct netfs_io_request *rreq)
ssize_t size = rreq->len; ssize_t size = rreq->len;
int ret = 0; int ret = 0;
atomic_inc(&rreq->nr_outstanding);
do { do {
struct netfs_io_subrequest *subreq; struct netfs_io_subrequest *subreq;
enum netfs_io_source source = NETFS_DOWNLOAD_FROM_SERVER; enum netfs_io_source source = NETFS_DOWNLOAD_FROM_SERVER;
@ -202,14 +222,6 @@ static void netfs_read_to_pagecache(struct netfs_io_request *rreq)
subreq->start = start; subreq->start = start;
subreq->len = size; subreq->len = size;
atomic_inc(&rreq->nr_outstanding);
spin_lock(&rreq->lock);
list_add_tail(&subreq->rreq_link, &rreq->subrequests);
subreq->prev_donated = rreq->prev_donated;
rreq->prev_donated = 0;
trace_netfs_sreq(subreq, netfs_sreq_trace_added);
spin_unlock(&rreq->lock);
source = netfs_cache_prepare_read(rreq, subreq, rreq->i_size); source = netfs_cache_prepare_read(rreq, subreq, rreq->i_size);
subreq->source = source; subreq->source = source;
if (source == NETFS_DOWNLOAD_FROM_SERVER) { if (source == NETFS_DOWNLOAD_FROM_SERVER) {
@ -238,24 +250,13 @@ static void netfs_read_to_pagecache(struct netfs_io_request *rreq)
if (rreq->netfs_ops->prepare_read) { if (rreq->netfs_ops->prepare_read) {
ret = rreq->netfs_ops->prepare_read(subreq); ret = rreq->netfs_ops->prepare_read(subreq);
if (ret < 0) { if (ret < 0) {
atomic_dec(&rreq->nr_outstanding);
netfs_put_subrequest(subreq, false, netfs_put_subrequest(subreq, false,
netfs_sreq_trace_put_cancel); netfs_sreq_trace_put_cancel);
break; break;
} }
trace_netfs_sreq(subreq, netfs_sreq_trace_prepare); trace_netfs_sreq(subreq, netfs_sreq_trace_prepare);
} }
goto issue;
slice = netfs_prepare_read_iterator(subreq);
if (slice < 0) {
atomic_dec(&rreq->nr_outstanding);
netfs_put_subrequest(subreq, false, netfs_sreq_trace_put_cancel);
ret = slice;
break;
}
rreq->netfs_ops->issue_read(subreq);
goto done;
} }
fill_with_zeroes: fill_with_zeroes:
@ -263,55 +264,46 @@ static void netfs_read_to_pagecache(struct netfs_io_request *rreq)
subreq->source = NETFS_FILL_WITH_ZEROES; subreq->source = NETFS_FILL_WITH_ZEROES;
trace_netfs_sreq(subreq, netfs_sreq_trace_submit); trace_netfs_sreq(subreq, netfs_sreq_trace_submit);
netfs_stat(&netfs_n_rh_zero); netfs_stat(&netfs_n_rh_zero);
slice = netfs_prepare_read_iterator(subreq); goto issue;
__set_bit(NETFS_SREQ_CLEAR_TAIL, &subreq->flags);
subreq->error = 0;
netfs_read_subreq_terminated(subreq);
goto done;
} }
if (source == NETFS_READ_FROM_CACHE) { if (source == NETFS_READ_FROM_CACHE) {
trace_netfs_sreq(subreq, netfs_sreq_trace_submit); trace_netfs_sreq(subreq, netfs_sreq_trace_submit);
slice = netfs_prepare_read_iterator(subreq); goto issue;
netfs_read_cache_to_pagecache(rreq, subreq);
goto done;
} }
pr_err("Unexpected read source %u\n", source); pr_err("Unexpected read source %u\n", source);
WARN_ON_ONCE(1); WARN_ON_ONCE(1);
break; break;
done: issue:
slice = netfs_prepare_read_iterator(subreq);
if (slice < 0) {
netfs_put_subrequest(subreq, false, netfs_sreq_trace_put_cancel);
ret = slice;
break;
}
size -= slice; size -= slice;
start += slice; start += slice;
if (size <= 0) {
smp_wmb(); /* Write lists before ALL_QUEUED. */
set_bit(NETFS_RREQ_ALL_QUEUED, &rreq->flags);
}
netfs_issue_read(rreq, subreq);
cond_resched(); cond_resched();
} while (size > 0); } while (size > 0);
if (atomic_dec_and_test(&rreq->nr_outstanding)) if (unlikely(size > 0)) {
netfs_rreq_terminated(rreq); smp_wmb(); /* Write lists before ALL_QUEUED. */
set_bit(NETFS_RREQ_ALL_QUEUED, &rreq->flags);
netfs_wake_read_collector(rreq);
}
/* Defer error return as we may need to wait for outstanding I/O. */ /* Defer error return as we may need to wait for outstanding I/O. */
cmpxchg(&rreq->error, 0, ret); cmpxchg(&rreq->error, 0, ret);
} }
/*
* Wait for the read operation to complete, successfully or otherwise.
*/
static int netfs_wait_for_read(struct netfs_io_request *rreq)
{
int ret;
trace_netfs_rreq(rreq, netfs_rreq_trace_wait_ip);
wait_on_bit(&rreq->flags, NETFS_RREQ_IN_PROGRESS, TASK_UNINTERRUPTIBLE);
ret = rreq->error;
if (ret == 0 && rreq->submitted < rreq->len) {
trace_netfs_failure(rreq, NULL, ret, netfs_fail_short_read);
ret = -EIO;
}
return ret;
}
/** /**
* netfs_readahead - Helper to manage a read request * netfs_readahead - Helper to manage a read request
* @ractl: The description of the readahead request * @ractl: The description of the readahead request
@ -340,6 +332,8 @@ void netfs_readahead(struct readahead_control *ractl)
if (IS_ERR(rreq)) if (IS_ERR(rreq))
return; return;
__set_bit(NETFS_RREQ_OFFLOAD_COLLECTION, &rreq->flags);
ret = netfs_begin_cache_read(rreq, ictx); ret = netfs_begin_cache_read(rreq, ictx);
if (ret == -ENOMEM || ret == -EINTR || ret == -ERESTARTSYS) if (ret == -ENOMEM || ret == -EINTR || ret == -ERESTARTSYS)
goto cleanup_free; goto cleanup_free;
@ -456,7 +450,7 @@ static int netfs_read_gaps(struct file *file, struct folio *folio)
folio_put(sink); folio_put(sink);
ret = netfs_wait_for_read(rreq); ret = netfs_wait_for_read(rreq);
if (ret == 0) { if (ret >= 0) {
flush_dcache_folio(folio); flush_dcache_folio(folio);
folio_mark_uptodate(folio); folio_mark_uptodate(folio);
} }
@ -744,7 +738,7 @@ int netfs_prefetch_for_write(struct file *file, struct folio *folio,
netfs_read_to_pagecache(rreq); netfs_read_to_pagecache(rreq);
ret = netfs_wait_for_read(rreq); ret = netfs_wait_for_read(rreq);
netfs_put_request(rreq, false, netfs_rreq_trace_put_return); netfs_put_request(rreq, false, netfs_rreq_trace_put_return);
return ret; return ret < 0 ? ret : 0;
error_put: error_put:
netfs_put_request(rreq, false, netfs_rreq_trace_put_discard); netfs_put_request(rreq, false, netfs_rreq_trace_put_discard);

View File

@ -47,12 +47,11 @@ static void netfs_prepare_dio_read_iterator(struct netfs_io_subrequest *subreq)
*/ */
static int netfs_dispatch_unbuffered_reads(struct netfs_io_request *rreq) static int netfs_dispatch_unbuffered_reads(struct netfs_io_request *rreq)
{ {
struct netfs_io_stream *stream = &rreq->io_streams[0];
unsigned long long start = rreq->start; unsigned long long start = rreq->start;
ssize_t size = rreq->len; ssize_t size = rreq->len;
int ret = 0; int ret = 0;
atomic_set(&rreq->nr_outstanding, 1);
do { do {
struct netfs_io_subrequest *subreq; struct netfs_io_subrequest *subreq;
ssize_t slice; ssize_t slice;
@ -67,11 +66,18 @@ static int netfs_dispatch_unbuffered_reads(struct netfs_io_request *rreq)
subreq->start = start; subreq->start = start;
subreq->len = size; subreq->len = size;
atomic_inc(&rreq->nr_outstanding); __set_bit(NETFS_SREQ_IN_PROGRESS, &subreq->flags);
spin_lock(&rreq->lock); spin_lock(&rreq->lock);
list_add_tail(&subreq->rreq_link, &rreq->subrequests); list_add_tail(&subreq->rreq_link, &stream->subrequests);
subreq->prev_donated = rreq->prev_donated; if (list_is_first(&subreq->rreq_link, &stream->subrequests)) {
rreq->prev_donated = 0; stream->front = subreq;
if (!stream->active) {
stream->collected_to = stream->front->start;
/* Store list pointers before active flag */
smp_store_release(&stream->active, true);
}
}
trace_netfs_sreq(subreq, netfs_sreq_trace_added); trace_netfs_sreq(subreq, netfs_sreq_trace_added);
spin_unlock(&rreq->lock); spin_unlock(&rreq->lock);
@ -79,7 +85,6 @@ static int netfs_dispatch_unbuffered_reads(struct netfs_io_request *rreq)
if (rreq->netfs_ops->prepare_read) { if (rreq->netfs_ops->prepare_read) {
ret = rreq->netfs_ops->prepare_read(subreq); ret = rreq->netfs_ops->prepare_read(subreq);
if (ret < 0) { if (ret < 0) {
atomic_dec(&rreq->nr_outstanding);
netfs_put_subrequest(subreq, false, netfs_sreq_trace_put_cancel); netfs_put_subrequest(subreq, false, netfs_sreq_trace_put_cancel);
break; break;
} }
@ -87,20 +92,34 @@ static int netfs_dispatch_unbuffered_reads(struct netfs_io_request *rreq)
netfs_prepare_dio_read_iterator(subreq); netfs_prepare_dio_read_iterator(subreq);
slice = subreq->len; slice = subreq->len;
rreq->netfs_ops->issue_read(subreq);
size -= slice; size -= slice;
start += slice; start += slice;
rreq->submitted += slice; rreq->submitted += slice;
if (size <= 0) {
smp_wmb(); /* Write lists before ALL_QUEUED. */
set_bit(NETFS_RREQ_ALL_QUEUED, &rreq->flags);
}
rreq->netfs_ops->issue_read(subreq);
if (test_bit(NETFS_RREQ_PAUSE, &rreq->flags)) {
trace_netfs_rreq(rreq, netfs_rreq_trace_wait_pause);
wait_on_bit(&rreq->flags, NETFS_RREQ_PAUSE, TASK_UNINTERRUPTIBLE);
}
if (test_bit(NETFS_RREQ_FAILED, &rreq->flags))
break;
if (test_bit(NETFS_RREQ_BLOCKED, &rreq->flags) && if (test_bit(NETFS_RREQ_BLOCKED, &rreq->flags) &&
test_bit(NETFS_RREQ_NONBLOCK, &rreq->flags)) test_bit(NETFS_RREQ_NONBLOCK, &rreq->flags))
break; break;
cond_resched(); cond_resched();
} while (size > 0); } while (size > 0);
if (atomic_dec_and_test(&rreq->nr_outstanding)) if (unlikely(size > 0)) {
netfs_rreq_terminated(rreq); smp_wmb(); /* Write lists before ALL_QUEUED. */
set_bit(NETFS_RREQ_ALL_QUEUED, &rreq->flags);
netfs_wake_read_collector(rreq);
}
return ret; return ret;
} }
@ -133,21 +152,10 @@ static int netfs_unbuffered_read(struct netfs_io_request *rreq, bool sync)
goto out; goto out;
} }
if (sync) { if (sync)
trace_netfs_rreq(rreq, netfs_rreq_trace_wait_ip); ret = netfs_wait_for_read(rreq);
wait_on_bit(&rreq->flags, NETFS_RREQ_IN_PROGRESS, else
TASK_UNINTERRUPTIBLE);
ret = rreq->error;
if (ret == 0 && rreq->submitted < rreq->len &&
rreq->origin != NETFS_DIO_READ) {
trace_netfs_failure(rreq, NULL, ret, netfs_fail_short_read);
ret = -EIO;
}
} else {
ret = -EIOCBQUEUED; ret = -EIOCBQUEUED;
}
out: out:
_leave(" = %d", ret); _leave(" = %d", ret);
return ret; return ret;
@ -215,8 +223,10 @@ ssize_t netfs_unbuffered_read_iter_locked(struct kiocb *iocb, struct iov_iter *i
// TODO: Set up bounce buffer if needed // TODO: Set up bounce buffer if needed
if (!sync) if (!sync) {
rreq->iocb = iocb; rreq->iocb = iocb;
__set_bit(NETFS_RREQ_OFFLOAD_COLLECTION, &rreq->flags);
}
ret = netfs_unbuffered_read(rreq, sync); ret = netfs_unbuffered_read(rreq, sync);
if (ret < 0) if (ret < 0)

View File

@ -82,17 +82,25 @@ static inline void netfs_see_request(struct netfs_io_request *rreq,
trace_netfs_rreq_ref(rreq->debug_id, refcount_read(&rreq->ref), what); trace_netfs_rreq_ref(rreq->debug_id, refcount_read(&rreq->ref), what);
} }
static inline void netfs_see_subrequest(struct netfs_io_subrequest *subreq,
enum netfs_sreq_ref_trace what)
{
trace_netfs_sreq_ref(subreq->rreq->debug_id, subreq->debug_index,
refcount_read(&subreq->ref), what);
}
/* /*
* read_collect.c * read_collect.c
*/ */
void netfs_read_termination_worker(struct work_struct *work); void netfs_read_collection_worker(struct work_struct *work);
void netfs_rreq_terminated(struct netfs_io_request *rreq); void netfs_wake_read_collector(struct netfs_io_request *rreq);
void netfs_cache_read_terminated(void *priv, ssize_t transferred_or_error, bool was_async);
ssize_t netfs_wait_for_read(struct netfs_io_request *rreq);
/* /*
* read_pgpriv2.c * read_pgpriv2.c
*/ */
void netfs_pgpriv2_mark_copy_to_cache(struct netfs_io_subrequest *subreq, void netfs_pgpriv2_mark_copy_to_cache(struct netfs_io_request *rreq,
struct netfs_io_request *rreq,
struct folio_queue *folioq, struct folio_queue *folioq,
int slot); int slot);
void netfs_pgpriv2_write_to_the_cache(struct netfs_io_request *rreq); void netfs_pgpriv2_write_to_the_cache(struct netfs_io_request *rreq);

View File

@ -71,7 +71,7 @@ static int netfs_requests_seq_show(struct seq_file *m, void *v)
refcount_read(&rreq->ref), refcount_read(&rreq->ref),
rreq->flags, rreq->flags,
rreq->error, rreq->error,
atomic_read(&rreq->nr_outstanding), 0,
rreq->start, rreq->submitted, rreq->len); rreq->start, rreq->submitted, rreq->len);
seq_putc(m, '\n'); seq_putc(m, '\n');
return 0; return 0;

View File

@ -48,7 +48,7 @@ struct netfs_io_request *netfs_alloc_request(struct address_space *mapping,
spin_lock_init(&rreq->lock); spin_lock_init(&rreq->lock);
INIT_LIST_HEAD(&rreq->io_streams[0].subrequests); INIT_LIST_HEAD(&rreq->io_streams[0].subrequests);
INIT_LIST_HEAD(&rreq->io_streams[1].subrequests); INIT_LIST_HEAD(&rreq->io_streams[1].subrequests);
INIT_LIST_HEAD(&rreq->subrequests); init_waitqueue_head(&rreq->waitq);
refcount_set(&rreq->ref, 1); refcount_set(&rreq->ref, 1);
if (origin == NETFS_READAHEAD || if (origin == NETFS_READAHEAD ||
@ -56,10 +56,12 @@ struct netfs_io_request *netfs_alloc_request(struct address_space *mapping,
origin == NETFS_READ_GAPS || origin == NETFS_READ_GAPS ||
origin == NETFS_READ_SINGLE || origin == NETFS_READ_SINGLE ||
origin == NETFS_READ_FOR_WRITE || origin == NETFS_READ_FOR_WRITE ||
origin == NETFS_DIO_READ) origin == NETFS_DIO_READ) {
INIT_WORK(&rreq->work, NULL); INIT_WORK(&rreq->work, netfs_read_collection_worker);
else rreq->io_streams[0].avail = true;
} else {
INIT_WORK(&rreq->work, netfs_write_collection_worker); INIT_WORK(&rreq->work, netfs_write_collection_worker);
}
__set_bit(NETFS_RREQ_IN_PROGRESS, &rreq->flags); __set_bit(NETFS_RREQ_IN_PROGRESS, &rreq->flags);
if (file && file->f_flags & O_NONBLOCK) if (file && file->f_flags & O_NONBLOCK)
@ -93,14 +95,6 @@ void netfs_clear_subrequests(struct netfs_io_request *rreq, bool was_async)
struct netfs_io_stream *stream; struct netfs_io_stream *stream;
int s; int s;
while (!list_empty(&rreq->subrequests)) {
subreq = list_first_entry(&rreq->subrequests,
struct netfs_io_subrequest, rreq_link);
list_del(&subreq->rreq_link);
netfs_put_subrequest(subreq, was_async,
netfs_sreq_trace_put_clear);
}
for (s = 0; s < ARRAY_SIZE(rreq->io_streams); s++) { for (s = 0; s < ARRAY_SIZE(rreq->io_streams); s++) {
stream = &rreq->io_streams[s]; stream = &rreq->io_streams[s];
while (!list_empty(&stream->subrequests)) { while (!list_empty(&stream->subrequests)) {
@ -192,21 +186,7 @@ struct netfs_io_subrequest *netfs_alloc_subrequest(struct netfs_io_request *rreq
} }
memset(subreq, 0, kmem_cache_size(cache)); memset(subreq, 0, kmem_cache_size(cache));
INIT_WORK(&subreq->work, NULL);
switch (rreq->origin) {
case NETFS_READAHEAD:
case NETFS_READPAGE:
case NETFS_READ_GAPS:
case NETFS_READ_SINGLE:
case NETFS_READ_FOR_WRITE:
case NETFS_DIO_READ:
INIT_WORK(&subreq->work, netfs_read_subreq_termination_worker);
break;
default:
INIT_WORK(&subreq->work, NULL);
break;
}
INIT_LIST_HEAD(&subreq->rreq_link); INIT_LIST_HEAD(&subreq->rreq_link);
refcount_set(&subreq->ref, 2); refcount_set(&subreq->ref, 2);
subreq->rreq = rreq; subreq->rreq = rreq;

View File

@ -14,6 +14,14 @@
#include <linux/task_io_accounting_ops.h> #include <linux/task_io_accounting_ops.h>
#include "internal.h" #include "internal.h"
/* Notes made in the collector */
#define HIT_PENDING 0x01 /* A front op was still pending */
#define MADE_PROGRESS 0x04 /* Made progress cleaning up a stream or the folio set */
#define BUFFERED 0x08 /* The pagecache needs cleaning up */
#define NEED_RETRY 0x10 /* A front op requests retrying */
#define COPY_TO_CACHE 0x40 /* Need to copy subrequest to cache */
#define ABANDON_SREQ 0x80 /* Need to abandon untransferred part of subrequest */
/* /*
* Clear the unread part of an I/O request. * Clear the unread part of an I/O request.
*/ */
@ -31,14 +39,18 @@ static void netfs_clear_unread(struct netfs_io_subrequest *subreq)
* cache the folio, we set the group to NETFS_FOLIO_COPY_TO_CACHE, mark it * cache the folio, we set the group to NETFS_FOLIO_COPY_TO_CACHE, mark it
* dirty and let writeback handle it. * dirty and let writeback handle it.
*/ */
static void netfs_unlock_read_folio(struct netfs_io_subrequest *subreq, static void netfs_unlock_read_folio(struct netfs_io_request *rreq,
struct netfs_io_request *rreq,
struct folio_queue *folioq, struct folio_queue *folioq,
int slot) int slot)
{ {
struct netfs_folio *finfo; struct netfs_folio *finfo;
struct folio *folio = folioq_folio(folioq, slot); struct folio *folio = folioq_folio(folioq, slot);
if (unlikely(folio_pos(folio) < rreq->abandon_to)) {
trace_netfs_folio(folio, netfs_folio_trace_abandon);
goto just_unlock;
}
flush_dcache_folio(folio); flush_dcache_folio(folio);
folio_mark_uptodate(folio); folio_mark_uptodate(folio);
@ -53,7 +65,7 @@ static void netfs_unlock_read_folio(struct netfs_io_subrequest *subreq,
kfree(finfo); kfree(finfo);
} }
if (test_bit(NETFS_SREQ_COPY_TO_CACHE, &subreq->flags)) { if (test_bit(NETFS_RREQ_FOLIO_COPY_TO_CACHE, &rreq->flags)) {
if (!WARN_ON_ONCE(folio_get_private(folio) != NULL)) { if (!WARN_ON_ONCE(folio_get_private(folio) != NULL)) {
trace_netfs_folio(folio, netfs_folio_trace_copy_to_cache); trace_netfs_folio(folio, netfs_folio_trace_copy_to_cache);
folio_attach_private(folio, NETFS_FOLIO_COPY_TO_CACHE); folio_attach_private(folio, NETFS_FOLIO_COPY_TO_CACHE);
@ -64,10 +76,11 @@ static void netfs_unlock_read_folio(struct netfs_io_subrequest *subreq,
} }
} else { } else {
// TODO: Use of PG_private_2 is deprecated. // TODO: Use of PG_private_2 is deprecated.
if (test_bit(NETFS_SREQ_COPY_TO_CACHE, &subreq->flags)) if (test_bit(NETFS_RREQ_FOLIO_COPY_TO_CACHE, &rreq->flags))
netfs_pgpriv2_mark_copy_to_cache(subreq, rreq, folioq, slot); netfs_pgpriv2_mark_copy_to_cache(rreq, folioq, slot);
} }
just_unlock:
if (!test_bit(NETFS_RREQ_DONT_UNLOCK_FOLIOS, &rreq->flags)) { if (!test_bit(NETFS_RREQ_DONT_UNLOCK_FOLIOS, &rreq->flags)) {
if (folio->index == rreq->no_unlock_folio && if (folio->index == rreq->no_unlock_folio &&
test_bit(NETFS_RREQ_NO_UNLOCK_FOLIO, &rreq->flags)) { test_bit(NETFS_RREQ_NO_UNLOCK_FOLIO, &rreq->flags)) {
@ -82,237 +95,243 @@ static void netfs_unlock_read_folio(struct netfs_io_subrequest *subreq,
} }
/* /*
* Unlock any folios that are now completely read. Returns true if the * Unlock any folios we've finished with.
* subrequest is removed from the list.
*/ */
static bool netfs_consume_read_data(struct netfs_io_subrequest *subreq) static void netfs_read_unlock_folios(struct netfs_io_request *rreq,
unsigned int *notes)
{ {
struct netfs_io_subrequest *prev, *next; struct folio_queue *folioq = rreq->buffer.tail;
struct netfs_io_request *rreq = subreq->rreq; unsigned long long collected_to = rreq->collected_to;
struct folio_queue *folioq = subreq->curr_folioq; unsigned int slot = rreq->buffer.first_tail_slot;
size_t avail, prev_donated, next_donated, fsize, part, excess;
loff_t fpos, start;
loff_t fend;
int slot = subreq->curr_folioq_slot;
if (WARN(subreq->transferred > subreq->len, if (rreq->cleaned_to >= rreq->collected_to)
"Subreq overread: R%x[%x] %zu > %zu", return;
rreq->debug_id, subreq->debug_index,
subreq->transferred, subreq->len))
subreq->transferred = subreq->len;
trace_netfs_folioq(folioq, netfs_trace_folioq_read_progress); // TODO: Begin decryption
next_folio:
fsize = PAGE_SIZE << subreq->curr_folio_order;
fpos = round_down(subreq->start + subreq->consumed, fsize);
fend = fpos + fsize;
if (WARN_ON_ONCE(!folioq) || if (slot >= folioq_nr_slots(folioq)) {
WARN_ON_ONCE(!folioq_folio(folioq, slot)) || folioq = rolling_buffer_delete_spent(&rreq->buffer);
WARN_ON_ONCE(folioq_folio(folioq, slot)->index != fpos / PAGE_SIZE)) { if (!folioq) {
pr_err("R=%08x[%x] s=%llx-%llx ctl=%zx/%zx/%zx sl=%u\n", rreq->front_folio_order = 0;
rreq->debug_id, subreq->debug_index, return;
subreq->start, subreq->start + subreq->transferred - 1,
subreq->consumed, subreq->transferred, subreq->len,
slot);
if (folioq) {
struct folio *folio = folioq_folio(folioq, slot);
pr_err("folioq: fq=%x orders=%02x%02x%02x%02x %px\n",
folioq->debug_id,
folioq->orders[0], folioq->orders[1],
folioq->orders[2], folioq->orders[3],
folioq);
if (folio)
pr_err("folio: %llx-%llx ix=%llx o=%u qo=%u\n",
fpos, fend - 1, folio_pos(folio), folio_order(folio),
folioq_folio_order(folioq, slot));
} }
slot = 0;
} }
donation_changed: for (;;) {
/* Try to consume the current folio if we've hit or passed the end of struct folio *folio;
* it. There's a possibility that this subreq doesn't start at the unsigned long long fpos, fend;
* beginning of the folio, in which case we need to donate to/from the unsigned int order;
* preceding subreq. size_t fsize;
*
* We also need to include any potential donation back from the
* following subreq.
*/
prev_donated = READ_ONCE(subreq->prev_donated);
next_donated = READ_ONCE(subreq->next_donated);
if (prev_donated || next_donated) {
spin_lock(&rreq->lock);
prev_donated = subreq->prev_donated;
next_donated = subreq->next_donated;
subreq->start -= prev_donated;
subreq->len += prev_donated;
subreq->transferred += prev_donated;
prev_donated = subreq->prev_donated = 0;
if (subreq->transferred == subreq->len) {
subreq->len += next_donated;
subreq->transferred += next_donated;
next_donated = subreq->next_donated = 0;
}
trace_netfs_sreq(subreq, netfs_sreq_trace_add_donations);
spin_unlock(&rreq->lock);
}
avail = subreq->transferred; if (*notes & COPY_TO_CACHE)
if (avail == subreq->len) set_bit(NETFS_RREQ_FOLIO_COPY_TO_CACHE, &rreq->flags);
avail += next_donated;
start = subreq->start;
if (subreq->consumed == 0) {
start -= prev_donated;
avail += prev_donated;
} else {
start += subreq->consumed;
avail -= subreq->consumed;
}
part = umin(avail, fsize);
trace_netfs_progress(subreq, start, avail, part); folio = folioq_folio(folioq, slot);
if (WARN_ONCE(!folio_test_locked(folio),
"R=%08x: folio %lx is not locked\n",
rreq->debug_id, folio->index))
trace_netfs_folio(folio, netfs_folio_trace_not_locked);
if (start + avail >= fend) { order = folioq_folio_order(folioq, slot);
if (fpos == start) { rreq->front_folio_order = order;
/* Flush, unlock and mark for caching any folio we've just read. */ fsize = PAGE_SIZE << order;
subreq->consumed = fend - subreq->start; fpos = folio_pos(folio);
netfs_unlock_read_folio(subreq, rreq, folioq, slot); fend = umin(fpos + fsize, rreq->i_size);
folioq_mark2(folioq, slot);
if (subreq->consumed >= subreq->len)
goto remove_subreq;
} else if (fpos < start) {
excess = fend - subreq->start;
spin_lock(&rreq->lock); trace_netfs_collect_folio(rreq, folio, fend, collected_to);
/* If we complete first on a folio split with the
* preceding subreq, donate to that subreq - otherwise
* we get the responsibility.
*/
if (subreq->prev_donated != prev_donated) {
spin_unlock(&rreq->lock);
goto donation_changed;
}
if (list_is_first(&subreq->rreq_link, &rreq->subrequests)) { /* Unlock any folio we've transferred all of. */
spin_unlock(&rreq->lock); if (collected_to < fend)
pr_err("Can't donate prior to front\n"); break;
goto bad;
}
prev = list_prev_entry(subreq, rreq_link); netfs_unlock_read_folio(rreq, folioq, slot);
WRITE_ONCE(prev->next_donated, prev->next_donated + excess); WRITE_ONCE(rreq->cleaned_to, fpos + fsize);
subreq->start += excess; *notes |= MADE_PROGRESS;
subreq->len -= excess;
subreq->transferred -= excess;
trace_netfs_donate(rreq, subreq, prev, excess,
netfs_trace_donate_tail_to_prev);
trace_netfs_sreq(subreq, netfs_sreq_trace_donate_to_prev);
if (subreq->consumed >= subreq->len) clear_bit(NETFS_RREQ_FOLIO_COPY_TO_CACHE, &rreq->flags);
goto remove_subreq_locked;
spin_unlock(&rreq->lock);
} else {
pr_err("fpos > start\n");
goto bad;
}
/* Advance the rolling buffer to the next folio. */ /* Clean up the head folioq. If we clear an entire folioq, then
* we can get rid of it provided it's not also the tail folioq
* being filled by the issuer.
*/
folioq_clear(folioq, slot);
slot++; slot++;
if (slot >= folioq_nr_slots(folioq)) { if (slot >= folioq_nr_slots(folioq)) {
folioq = rolling_buffer_delete_spent(&rreq->buffer);
if (!folioq)
goto done;
slot = 0; slot = 0;
folioq = folioq->next;
subreq->curr_folioq = folioq;
trace_netfs_folioq(folioq, netfs_trace_folioq_read_progress); trace_netfs_folioq(folioq, netfs_trace_folioq_read_progress);
} }
subreq->curr_folioq_slot = slot;
if (folioq && folioq_folio(folioq, slot)) if (fpos + fsize >= collected_to)
subreq->curr_folio_order = folioq->orders[slot]; break;
cond_resched();
goto next_folio;
} }
/* Deal with partial progress. */ rreq->buffer.tail = folioq;
if (subreq->transferred < subreq->len) done:
return false; rreq->buffer.first_tail_slot = slot;
}
/* Donate the remaining downloaded data to one of the neighbouring /*
* subrequests. Note that we may race with them doing the same thing. * Collect and assess the results of various read subrequests. We may need to
* retry some of the results.
*
* Note that we have a sequence of subrequests, which may be drawing on
* different sources and may or may not be the same size or starting position
* and may not even correspond in boundary alignment.
*/
static void netfs_collect_read_results(struct netfs_io_request *rreq)
{
struct netfs_io_subrequest *front, *remove;
struct netfs_io_stream *stream = &rreq->io_streams[0];
unsigned int notes;
_enter("%llx-%llx", rreq->start, rreq->start + rreq->len);
trace_netfs_rreq(rreq, netfs_rreq_trace_collect);
trace_netfs_collect(rreq);
reassess:
if (rreq->origin == NETFS_READAHEAD ||
rreq->origin == NETFS_READPAGE ||
rreq->origin == NETFS_READ_FOR_WRITE)
notes = BUFFERED;
else
notes = 0;
/* Remove completed subrequests from the front of the stream and
* advance the completion point. We stop when we hit something that's
* in progress. The issuer thread may be adding stuff to the tail
* whilst we're doing this.
*/ */
spin_lock(&rreq->lock); front = READ_ONCE(stream->front);
while (front) {
size_t transferred;
if (subreq->prev_donated != prev_donated || trace_netfs_collect_sreq(rreq, front);
subreq->next_donated != next_donated) { _debug("sreq [%x] %llx %zx/%zx",
front->debug_index, front->start, front->transferred, front->len);
if (stream->collected_to < front->start) {
trace_netfs_collect_gap(rreq, stream, front->start, 'F');
stream->collected_to = front->start;
}
if (test_bit(NETFS_SREQ_IN_PROGRESS, &front->flags))
notes |= HIT_PENDING;
smp_rmb(); /* Read counters after IN_PROGRESS flag. */
transferred = READ_ONCE(front->transferred);
/* If we can now collect the next folio, do so. We don't want
* to defer this as we have to decide whether we need to copy
* to the cache or not, and that may differ between adjacent
* subreqs.
*/
if (notes & BUFFERED) {
size_t fsize = PAGE_SIZE << rreq->front_folio_order;
/* Clear the tail of a short read. */
if (!(notes & HIT_PENDING) &&
front->error == 0 &&
transferred < front->len &&
(test_bit(NETFS_SREQ_HIT_EOF, &front->flags) ||
test_bit(NETFS_SREQ_CLEAR_TAIL, &front->flags))) {
netfs_clear_unread(front);
transferred = front->transferred = front->len;
trace_netfs_sreq(front, netfs_sreq_trace_clear);
}
stream->collected_to = front->start + transferred;
rreq->collected_to = stream->collected_to;
if (test_bit(NETFS_SREQ_COPY_TO_CACHE, &front->flags))
notes |= COPY_TO_CACHE;
if (test_bit(NETFS_SREQ_FAILED, &front->flags)) {
rreq->abandon_to = front->start + front->len;
front->transferred = front->len;
transferred = front->len;
trace_netfs_rreq(rreq, netfs_rreq_trace_set_abandon);
}
if (front->start + transferred >= rreq->cleaned_to + fsize ||
test_bit(NETFS_SREQ_HIT_EOF, &front->flags))
netfs_read_unlock_folios(rreq, &notes);
} else {
stream->collected_to = front->start + transferred;
rreq->collected_to = stream->collected_to;
}
/* Stall if the front is still undergoing I/O. */
if (notes & HIT_PENDING)
break;
if (test_bit(NETFS_SREQ_FAILED, &front->flags)) {
if (!stream->failed) {
stream->error = front->error;
rreq->error = front->error;
set_bit(NETFS_RREQ_FAILED, &rreq->flags);
stream->failed = true;
}
notes |= MADE_PROGRESS | ABANDON_SREQ;
} else if (test_bit(NETFS_SREQ_NEED_RETRY, &front->flags)) {
stream->need_retry = true;
notes |= NEED_RETRY | MADE_PROGRESS;
break;
} else {
if (!stream->failed)
stream->transferred = stream->collected_to - rreq->start;
notes |= MADE_PROGRESS;
}
/* Remove if completely consumed. */
stream->source = front->source;
spin_lock(&rreq->lock);
remove = front;
trace_netfs_sreq(front, netfs_sreq_trace_discard);
list_del_init(&front->rreq_link);
front = list_first_entry_or_null(&stream->subrequests,
struct netfs_io_subrequest, rreq_link);
stream->front = front;
spin_unlock(&rreq->lock); spin_unlock(&rreq->lock);
cond_resched(); netfs_put_subrequest(remove, false,
goto donation_changed; notes & ABANDON_SREQ ?
netfs_sreq_trace_put_cancel :
netfs_sreq_trace_put_done);
} }
/* Deal with the trickiest case: that this subreq is in the middle of a trace_netfs_collect_stream(rreq, stream);
* folio, not touching either edge, but finishes first. In such a trace_netfs_collect_state(rreq, rreq->collected_to, notes);
* case, we donate to the previous subreq, if there is one, so that the
* donation is only handled when that completes - and remove this if (!(notes & BUFFERED))
* subreq from the list. rreq->cleaned_to = rreq->collected_to;
*
* If the previous subreq finished first, we will have acquired their if (notes & NEED_RETRY)
* donation and should be able to unlock folios and/or donate nextwards. goto need_retry;
if ((notes & MADE_PROGRESS) && test_bit(NETFS_RREQ_PAUSE, &rreq->flags)) {
trace_netfs_rreq(rreq, netfs_rreq_trace_unpause);
clear_bit_unlock(NETFS_RREQ_PAUSE, &rreq->flags);
wake_up_bit(&rreq->flags, NETFS_RREQ_PAUSE);
}
if (notes & MADE_PROGRESS) {
//cond_resched();
goto reassess;
}
out:
_leave(" = %x", notes);
return;
need_retry:
/* Okay... We're going to have to retry parts of the stream. Note
* that any partially completed op will have had any wholly transferred
* folios removed from it.
*/ */
if (!subreq->consumed && _debug("retry");
!prev_donated && netfs_retry_reads(rreq);
!list_is_first(&subreq->rreq_link, &rreq->subrequests)) { goto out;
prev = list_prev_entry(subreq, rreq_link);
WRITE_ONCE(prev->next_donated, prev->next_donated + subreq->len);
subreq->start += subreq->len;
subreq->len = 0;
subreq->transferred = 0;
trace_netfs_donate(rreq, subreq, prev, subreq->len,
netfs_trace_donate_to_prev);
trace_netfs_sreq(subreq, netfs_sreq_trace_donate_to_prev);
goto remove_subreq_locked;
}
/* If we can't donate down the chain, donate up the chain instead. */
excess = subreq->len - subreq->consumed + next_donated;
if (!subreq->consumed)
excess += prev_donated;
if (list_is_last(&subreq->rreq_link, &rreq->subrequests)) {
rreq->prev_donated = excess;
trace_netfs_donate(rreq, subreq, NULL, excess,
netfs_trace_donate_to_deferred_next);
} else {
next = list_next_entry(subreq, rreq_link);
WRITE_ONCE(next->prev_donated, excess);
trace_netfs_donate(rreq, subreq, next, excess,
netfs_trace_donate_to_next);
}
trace_netfs_sreq(subreq, netfs_sreq_trace_donate_to_next);
subreq->len = subreq->consumed;
subreq->transferred = subreq->consumed;
goto remove_subreq_locked;
remove_subreq:
spin_lock(&rreq->lock);
remove_subreq_locked:
subreq->consumed = subreq->len;
list_del(&subreq->rreq_link);
spin_unlock(&rreq->lock);
netfs_put_subrequest(subreq, false, netfs_sreq_trace_put_consumed);
return true;
bad:
/* Errr... prev and next both donated to us, but insufficient to finish
* the folio.
*/
printk("R=%08x[%x] s=%llx-%llx %zx/%zx/%zx\n",
rreq->debug_id, subreq->debug_index,
subreq->start, subreq->start + subreq->transferred - 1,
subreq->consumed, subreq->transferred, subreq->len);
printk("folio: %llx-%llx\n", fpos, fend - 1);
printk("donated: prev=%zx next=%zx\n", prev_donated, next_donated);
printk("s=%llx av=%zx part=%zx\n", start, avail, part);
BUG();
} }
/* /*
@ -321,12 +340,13 @@ bad:
static void netfs_rreq_assess_dio(struct netfs_io_request *rreq) static void netfs_rreq_assess_dio(struct netfs_io_request *rreq)
{ {
struct netfs_io_subrequest *subreq; struct netfs_io_subrequest *subreq;
struct netfs_io_stream *stream = &rreq->io_streams[0];
unsigned int i; unsigned int i;
/* Collect unbuffered reads and direct reads, adding up the transfer /* Collect unbuffered reads and direct reads, adding up the transfer
* sizes until we find the first short or failed subrequest. * sizes until we find the first short or failed subrequest.
*/ */
list_for_each_entry(subreq, &rreq->subrequests, rreq_link) { list_for_each_entry(subreq, &stream->subrequests, rreq_link) {
rreq->transferred += subreq->transferred; rreq->transferred += subreq->transferred;
if (subreq->transferred < subreq->len || if (subreq->transferred < subreq->len ||
@ -363,22 +383,12 @@ static void netfs_rreq_assess_dio(struct netfs_io_request *rreq)
*/ */
static void netfs_rreq_assess_single(struct netfs_io_request *rreq) static void netfs_rreq_assess_single(struct netfs_io_request *rreq)
{ {
struct netfs_io_subrequest *subreq;
struct netfs_io_stream *stream = &rreq->io_streams[0]; struct netfs_io_stream *stream = &rreq->io_streams[0];
subreq = list_first_entry_or_null(&stream->subrequests, if (!rreq->error && stream->source == NETFS_DOWNLOAD_FROM_SERVER &&
struct netfs_io_subrequest, rreq_link); fscache_resources_valid(&rreq->cache_resources)) {
if (subreq) { trace_netfs_rreq(rreq, netfs_rreq_trace_dirty);
if (test_bit(NETFS_SREQ_FAILED, &subreq->flags)) netfs_single_mark_inode_dirty(rreq->inode);
rreq->error = subreq->error;
else
rreq->transferred = subreq->transferred;
if (!rreq->error && subreq->source == NETFS_DOWNLOAD_FROM_SERVER &&
fscache_resources_valid(&rreq->cache_resources)) {
trace_netfs_rreq(rreq, netfs_rreq_trace_dirty);
netfs_single_mark_inode_dirty(rreq->inode);
}
} }
if (rreq->iocb) { if (rreq->iocb) {
@ -392,22 +402,33 @@ static void netfs_rreq_assess_single(struct netfs_io_request *rreq)
} }
/* /*
* Assess the state of a read request and decide what to do next. * Perform the collection of subrequests and folios.
* *
* Note that we're in normal kernel thread context at this point, possibly * Note that we're in normal kernel thread context at this point, possibly
* running on a workqueue. * running on a workqueue.
*/ */
void netfs_rreq_terminated(struct netfs_io_request *rreq) static void netfs_read_collection(struct netfs_io_request *rreq)
{ {
trace_netfs_rreq(rreq, netfs_rreq_trace_assess); struct netfs_io_stream *stream = &rreq->io_streams[0];
netfs_collect_read_results(rreq);
/* We're done when the app thread has finished posting subreqs and the
* queue is empty.
*/
if (!test_bit(NETFS_RREQ_ALL_QUEUED, &rreq->flags))
return;
smp_rmb(); /* Read ALL_QUEUED before subreq lists. */
if (!list_empty(&stream->subrequests))
return;
/* Okay, declare that all I/O is complete. */
rreq->transferred = stream->transferred;
trace_netfs_rreq(rreq, netfs_rreq_trace_complete);
//netfs_rreq_is_still_valid(rreq); //netfs_rreq_is_still_valid(rreq);
if (test_and_clear_bit(NETFS_RREQ_NEED_RETRY, &rreq->flags)) {
netfs_retry_reads(rreq);
return;
}
switch (rreq->origin) { switch (rreq->origin) {
case NETFS_DIO_READ: case NETFS_DIO_READ:
case NETFS_READ_GAPS: case NETFS_READ_GAPS:
@ -432,6 +453,33 @@ void netfs_rreq_terminated(struct netfs_io_request *rreq)
netfs_pgpriv2_write_to_the_cache(rreq); netfs_pgpriv2_write_to_the_cache(rreq);
} }
void netfs_read_collection_worker(struct work_struct *work)
{
struct netfs_io_request *rreq = container_of(work, struct netfs_io_request, work);
netfs_see_request(rreq, netfs_rreq_trace_see_work);
if (test_bit(NETFS_RREQ_IN_PROGRESS, &rreq->flags))
netfs_read_collection(rreq);
netfs_put_request(rreq, false, netfs_rreq_trace_put_work);
}
/*
* Wake the collection work item.
*/
void netfs_wake_read_collector(struct netfs_io_request *rreq)
{
if (test_bit(NETFS_RREQ_OFFLOAD_COLLECTION, &rreq->flags)) {
if (!work_pending(&rreq->work)) {
netfs_get_request(rreq, netfs_rreq_trace_get_work);
if (!queue_work(system_unbound_wq, &rreq->work))
netfs_put_request(rreq, true, netfs_rreq_trace_put_work_nq);
}
} else {
trace_netfs_rreq(rreq, netfs_rreq_trace_wake_queue);
wake_up(&rreq->waitq);
}
}
/** /**
* netfs_read_subreq_progress - Note progress of a read operation. * netfs_read_subreq_progress - Note progress of a read operation.
* @subreq: The read request that has terminated. * @subreq: The read request that has terminated.
@ -445,17 +493,22 @@ void netfs_rreq_terminated(struct netfs_io_request *rreq)
void netfs_read_subreq_progress(struct netfs_io_subrequest *subreq) void netfs_read_subreq_progress(struct netfs_io_subrequest *subreq)
{ {
struct netfs_io_request *rreq = subreq->rreq; struct netfs_io_request *rreq = subreq->rreq;
struct netfs_io_stream *stream = &rreq->io_streams[0];
might_sleep(); size_t fsize = PAGE_SIZE << rreq->front_folio_order;
trace_netfs_sreq(subreq, netfs_sreq_trace_progress); trace_netfs_sreq(subreq, netfs_sreq_trace_progress);
if (subreq->transferred > subreq->consumed && /* If we are at the head of the queue, wake up the collector,
* getting a ref to it if we were the ones to do so.
*/
if (subreq->start + subreq->transferred > rreq->cleaned_to + fsize &&
(rreq->origin == NETFS_READAHEAD || (rreq->origin == NETFS_READAHEAD ||
rreq->origin == NETFS_READPAGE || rreq->origin == NETFS_READPAGE ||
rreq->origin == NETFS_READ_FOR_WRITE)) { rreq->origin == NETFS_READ_FOR_WRITE) &&
netfs_consume_read_data(subreq); list_is_first(&subreq->rreq_link, &stream->subrequests)
__clear_bit(NETFS_SREQ_NO_PROGRESS, &subreq->flags); ) {
__set_bit(NETFS_SREQ_MADE_PROGRESS, &subreq->flags);
netfs_wake_read_collector(rreq);
} }
} }
EXPORT_SYMBOL(netfs_read_subreq_progress); EXPORT_SYMBOL(netfs_read_subreq_progress);
@ -479,8 +532,7 @@ EXPORT_SYMBOL(netfs_read_subreq_progress);
void netfs_read_subreq_terminated(struct netfs_io_subrequest *subreq) void netfs_read_subreq_terminated(struct netfs_io_subrequest *subreq)
{ {
struct netfs_io_request *rreq = subreq->rreq; struct netfs_io_request *rreq = subreq->rreq;
struct netfs_io_stream *stream = &rreq->io_streams[0];
might_sleep();
switch (subreq->source) { switch (subreq->source) {
case NETFS_READ_FROM_CACHE: case NETFS_READ_FROM_CACHE:
@ -493,83 +545,114 @@ void netfs_read_subreq_terminated(struct netfs_io_subrequest *subreq)
break; break;
} }
if (rreq->origin != NETFS_DIO_READ) {
/* Collect buffered reads.
*
* If the read completed validly short, then we can clear the
* tail before going on to unlock the folios.
*/
if (subreq->error == 0 && subreq->transferred < subreq->len &&
(test_bit(NETFS_SREQ_HIT_EOF, &subreq->flags) ||
test_bit(NETFS_SREQ_CLEAR_TAIL, &subreq->flags))) {
netfs_clear_unread(subreq);
subreq->transferred = subreq->len;
trace_netfs_sreq(subreq, netfs_sreq_trace_clear);
}
if (subreq->transferred > subreq->consumed &&
(rreq->origin == NETFS_READAHEAD ||
rreq->origin == NETFS_READPAGE ||
rreq->origin == NETFS_READ_FOR_WRITE)) {
netfs_consume_read_data(subreq);
__clear_bit(NETFS_SREQ_NO_PROGRESS, &subreq->flags);
}
rreq->transferred += subreq->transferred;
}
/* Deal with retry requests, short reads and errors. If we retry /* Deal with retry requests, short reads and errors. If we retry
* but don't make progress, we abandon the attempt. * but don't make progress, we abandon the attempt.
*/ */
if (!subreq->error && subreq->transferred < subreq->len) { if (!subreq->error && subreq->transferred < subreq->len) {
if (test_bit(NETFS_SREQ_HIT_EOF, &subreq->flags)) { if (test_bit(NETFS_SREQ_HIT_EOF, &subreq->flags)) {
trace_netfs_sreq(subreq, netfs_sreq_trace_hit_eof); trace_netfs_sreq(subreq, netfs_sreq_trace_hit_eof);
} else if (test_bit(NETFS_SREQ_NEED_RETRY, &subreq->flags)) {
trace_netfs_sreq(subreq, netfs_sreq_trace_need_retry);
} else if (test_bit(NETFS_SREQ_MADE_PROGRESS, &subreq->flags)) {
__set_bit(NETFS_SREQ_NEED_RETRY, &subreq->flags);
trace_netfs_sreq(subreq, netfs_sreq_trace_partial_read);
} else { } else {
__set_bit(NETFS_SREQ_FAILED, &subreq->flags);
subreq->error = -ENODATA;
trace_netfs_sreq(subreq, netfs_sreq_trace_short); trace_netfs_sreq(subreq, netfs_sreq_trace_short);
if (subreq->transferred > subreq->consumed) {
__set_bit(NETFS_SREQ_NEED_RETRY, &subreq->flags);
__clear_bit(NETFS_SREQ_NO_PROGRESS, &subreq->flags);
set_bit(NETFS_RREQ_NEED_RETRY, &rreq->flags);
} else if (!__test_and_set_bit(NETFS_SREQ_NO_PROGRESS, &subreq->flags)) {
__set_bit(NETFS_SREQ_NEED_RETRY, &subreq->flags);
set_bit(NETFS_RREQ_NEED_RETRY, &rreq->flags);
} else {
__set_bit(NETFS_SREQ_FAILED, &subreq->flags);
subreq->error = -ENODATA;
}
} }
} }
trace_netfs_sreq(subreq, netfs_sreq_trace_terminated);
if (unlikely(subreq->error < 0)) { if (unlikely(subreq->error < 0)) {
trace_netfs_failure(rreq, subreq, subreq->error, netfs_fail_read); trace_netfs_failure(rreq, subreq, subreq->error, netfs_fail_read);
if (subreq->source == NETFS_READ_FROM_CACHE) { if (subreq->source == NETFS_READ_FROM_CACHE) {
netfs_stat(&netfs_n_rh_read_failed); netfs_stat(&netfs_n_rh_read_failed);
__set_bit(NETFS_SREQ_NEED_RETRY, &subreq->flags);
} else { } else {
netfs_stat(&netfs_n_rh_download_failed); netfs_stat(&netfs_n_rh_download_failed);
set_bit(NETFS_RREQ_FAILED, &rreq->flags); __set_bit(NETFS_SREQ_FAILED, &subreq->flags);
rreq->error = subreq->error;
} }
trace_netfs_rreq(rreq, netfs_rreq_trace_set_pause);
set_bit(NETFS_RREQ_PAUSE, &rreq->flags);
} }
if (atomic_dec_and_test(&rreq->nr_outstanding)) trace_netfs_sreq(subreq, netfs_sreq_trace_terminated);
netfs_rreq_terminated(rreq);
netfs_put_subrequest(subreq, false, netfs_sreq_trace_put_terminated); clear_bit_unlock(NETFS_SREQ_IN_PROGRESS, &subreq->flags);
/* If we are at the head of the queue, wake up the collector. */
if (list_is_first(&subreq->rreq_link, &stream->subrequests))
netfs_wake_read_collector(rreq);
netfs_put_subrequest(subreq, true, netfs_sreq_trace_put_terminated);
} }
EXPORT_SYMBOL(netfs_read_subreq_terminated); EXPORT_SYMBOL(netfs_read_subreq_terminated);
/** /*
* netfs_read_subreq_termination_worker - Workqueue helper for read termination * Handle termination of a read from the cache.
* @work: The subreq->work in the I/O request that has been terminated.
*
* Helper function to jump to netfs_read_subreq_terminated() from the
* subrequest work item.
*/ */
void netfs_read_subreq_termination_worker(struct work_struct *work) void netfs_cache_read_terminated(void *priv, ssize_t transferred_or_error, bool was_async)
{ {
struct netfs_io_subrequest *subreq = struct netfs_io_subrequest *subreq = priv;
container_of(work, struct netfs_io_subrequest, work);
if (transferred_or_error > 0) {
subreq->error = 0;
if (transferred_or_error > 0) {
subreq->transferred += transferred_or_error;
__set_bit(NETFS_SREQ_MADE_PROGRESS, &subreq->flags);
}
} else {
subreq->error = transferred_or_error;
}
netfs_read_subreq_terminated(subreq); netfs_read_subreq_terminated(subreq);
} }
EXPORT_SYMBOL(netfs_read_subreq_termination_worker);
/*
* Wait for the read operation to complete, successfully or otherwise.
*/
ssize_t netfs_wait_for_read(struct netfs_io_request *rreq)
{
struct netfs_io_subrequest *subreq;
struct netfs_io_stream *stream = &rreq->io_streams[0];
DEFINE_WAIT(myself);
ssize_t ret;
for (;;) {
trace_netfs_rreq(rreq, netfs_rreq_trace_wait_queue);
prepare_to_wait(&rreq->waitq, &myself, TASK_UNINTERRUPTIBLE);
subreq = list_first_entry_or_null(&stream->subrequests,
struct netfs_io_subrequest, rreq_link);
if (subreq &&
(!test_bit(NETFS_SREQ_IN_PROGRESS, &subreq->flags) ||
test_bit(NETFS_SREQ_MADE_PROGRESS, &subreq->flags)))
netfs_read_collection(rreq);
if (!test_bit(NETFS_RREQ_IN_PROGRESS, &rreq->flags))
break;
schedule();
trace_netfs_rreq(rreq, netfs_rreq_trace_woke_queue);
}
finish_wait(&rreq->waitq, &myself);
ret = rreq->error;
if (ret == 0) {
ret = rreq->transferred;
switch (rreq->origin) {
case NETFS_DIO_READ:
case NETFS_READ_SINGLE:
ret = rreq->transferred;
break;
default:
if (rreq->submitted < rreq->len) {
trace_netfs_failure(rreq, NULL, ret, netfs_fail_short_read);
ret = -EIO;
}
break;
}
}
return ret;
}

View File

@ -18,8 +18,7 @@
* third mark in the folio queue is used to indicate that this folio needs * third mark in the folio queue is used to indicate that this folio needs
* writing. * writing.
*/ */
void netfs_pgpriv2_mark_copy_to_cache(struct netfs_io_subrequest *subreq, void netfs_pgpriv2_mark_copy_to_cache(struct netfs_io_request *rreq,
struct netfs_io_request *rreq,
struct folio_queue *folioq, struct folio_queue *folioq,
int slot) int slot)
{ {

View File

@ -12,15 +12,8 @@
static void netfs_reissue_read(struct netfs_io_request *rreq, static void netfs_reissue_read(struct netfs_io_request *rreq,
struct netfs_io_subrequest *subreq) struct netfs_io_subrequest *subreq)
{ {
struct iov_iter *io_iter = &subreq->io_iter; __clear_bit(NETFS_SREQ_MADE_PROGRESS, &subreq->flags);
__set_bit(NETFS_SREQ_RETRYING, &subreq->flags);
if (iov_iter_is_folioq(io_iter)) {
subreq->curr_folioq = (struct folio_queue *)io_iter->folioq;
subreq->curr_folioq_slot = io_iter->folioq_slot;
subreq->curr_folio_order = subreq->curr_folioq->orders[subreq->curr_folioq_slot];
}
atomic_inc(&rreq->nr_outstanding);
__set_bit(NETFS_SREQ_IN_PROGRESS, &subreq->flags); __set_bit(NETFS_SREQ_IN_PROGRESS, &subreq->flags);
netfs_get_subrequest(subreq, netfs_sreq_trace_get_resubmit); netfs_get_subrequest(subreq, netfs_sreq_trace_get_resubmit);
subreq->rreq->netfs_ops->issue_read(subreq); subreq->rreq->netfs_ops->issue_read(subreq);
@ -33,13 +26,12 @@ static void netfs_reissue_read(struct netfs_io_request *rreq,
static void netfs_retry_read_subrequests(struct netfs_io_request *rreq) static void netfs_retry_read_subrequests(struct netfs_io_request *rreq)
{ {
struct netfs_io_subrequest *subreq; struct netfs_io_subrequest *subreq;
struct netfs_io_stream *stream0 = &rreq->io_streams[0]; struct netfs_io_stream *stream = &rreq->io_streams[0];
LIST_HEAD(sublist); struct list_head *next;
LIST_HEAD(queue);
_enter("R=%x", rreq->debug_id); _enter("R=%x", rreq->debug_id);
if (list_empty(&rreq->subrequests)) if (list_empty(&stream->subrequests))
return; return;
if (rreq->netfs_ops->retry_request) if (rreq->netfs_ops->retry_request)
@ -52,7 +44,7 @@ static void netfs_retry_read_subrequests(struct netfs_io_request *rreq)
!test_bit(NETFS_RREQ_COPY_TO_CACHE, &rreq->flags)) { !test_bit(NETFS_RREQ_COPY_TO_CACHE, &rreq->flags)) {
struct netfs_io_subrequest *subreq; struct netfs_io_subrequest *subreq;
list_for_each_entry(subreq, &rreq->subrequests, rreq_link) { list_for_each_entry(subreq, &stream->subrequests, rreq_link) {
if (test_bit(NETFS_SREQ_FAILED, &subreq->flags)) if (test_bit(NETFS_SREQ_FAILED, &subreq->flags))
break; break;
if (__test_and_clear_bit(NETFS_SREQ_NEED_RETRY, &subreq->flags)) { if (__test_and_clear_bit(NETFS_SREQ_NEED_RETRY, &subreq->flags)) {
@ -73,48 +65,44 @@ static void netfs_retry_read_subrequests(struct netfs_io_request *rreq)
* populating with smaller subrequests. In the event that the subreq * populating with smaller subrequests. In the event that the subreq
* we just launched finishes before we insert the next subreq, it'll * we just launched finishes before we insert the next subreq, it'll
* fill in rreq->prev_donated instead. * fill in rreq->prev_donated instead.
*
* Note: Alternatively, we could split the tail subrequest right before * Note: Alternatively, we could split the tail subrequest right before
* we reissue it and fix up the donations under lock. * we reissue it and fix up the donations under lock.
*/ */
list_splice_init(&rreq->subrequests, &queue); next = stream->subrequests.next;
do { do {
struct netfs_io_subrequest *from; struct netfs_io_subrequest *subreq = NULL, *from, *to, *tmp;
struct iov_iter source; struct iov_iter source;
unsigned long long start, len; unsigned long long start, len;
size_t part, deferred_next_donated = 0; size_t part;
bool boundary = false; bool boundary = false;
/* Go through the subreqs and find the next span of contiguous /* Go through the subreqs and find the next span of contiguous
* buffer that we then rejig (cifs, for example, needs the * buffer that we then rejig (cifs, for example, needs the
* rsize renegotiating) and reissue. * rsize renegotiating) and reissue.
*/ */
from = list_first_entry(&queue, struct netfs_io_subrequest, rreq_link); from = list_entry(next, struct netfs_io_subrequest, rreq_link);
list_move_tail(&from->rreq_link, &sublist); to = from;
start = from->start + from->transferred; start = from->start + from->transferred;
len = from->len - from->transferred; len = from->len - from->transferred;
_debug("from R=%08x[%x] s=%llx ctl=%zx/%zx/%zx", _debug("from R=%08x[%x] s=%llx ctl=%zx/%zx",
rreq->debug_id, from->debug_index, rreq->debug_id, from->debug_index,
from->start, from->consumed, from->transferred, from->len); from->start, from->transferred, from->len);
if (test_bit(NETFS_SREQ_FAILED, &from->flags) || if (test_bit(NETFS_SREQ_FAILED, &from->flags) ||
!test_bit(NETFS_SREQ_NEED_RETRY, &from->flags)) !test_bit(NETFS_SREQ_NEED_RETRY, &from->flags))
goto abandon; goto abandon;
deferred_next_donated = from->next_donated; list_for_each_continue(next, &stream->subrequests) {
while ((subreq = list_first_entry_or_null( subreq = list_entry(next, struct netfs_io_subrequest, rreq_link);
&queue, struct netfs_io_subrequest, rreq_link))) { if (subreq->start + subreq->transferred != start + len ||
if (subreq->start != start + len || test_bit(NETFS_SREQ_BOUNDARY, &subreq->flags) ||
subreq->transferred > 0 ||
!test_bit(NETFS_SREQ_NEED_RETRY, &subreq->flags)) !test_bit(NETFS_SREQ_NEED_RETRY, &subreq->flags))
break; break;
list_move_tail(&subreq->rreq_link, &sublist); to = subreq;
len += subreq->len; len += to->len;
deferred_next_donated = subreq->next_donated;
if (test_bit(NETFS_SREQ_BOUNDARY, &subreq->flags))
break;
} }
_debug(" - range: %llx-%llx %llx", start, start + len - 1, len); _debug(" - range: %llx-%llx %llx", start, start + len - 1, len);
@ -127,36 +115,28 @@ static void netfs_retry_read_subrequests(struct netfs_io_request *rreq)
source.count = len; source.count = len;
/* Work through the sublist. */ /* Work through the sublist. */
while ((subreq = list_first_entry_or_null( subreq = from;
&sublist, struct netfs_io_subrequest, rreq_link))) { list_for_each_entry_from(subreq, &stream->subrequests, rreq_link) {
list_del(&subreq->rreq_link); if (!len)
break;
subreq->source = NETFS_DOWNLOAD_FROM_SERVER; subreq->source = NETFS_DOWNLOAD_FROM_SERVER;
subreq->start = start - subreq->transferred; subreq->start = start - subreq->transferred;
subreq->len = len + subreq->transferred; subreq->len = len + subreq->transferred;
stream0->sreq_max_len = subreq->len;
__clear_bit(NETFS_SREQ_NEED_RETRY, &subreq->flags); __clear_bit(NETFS_SREQ_NEED_RETRY, &subreq->flags);
__set_bit(NETFS_SREQ_RETRYING, &subreq->flags); __set_bit(NETFS_SREQ_RETRYING, &subreq->flags);
spin_lock(&rreq->lock);
list_add_tail(&subreq->rreq_link, &rreq->subrequests);
subreq->prev_donated += rreq->prev_donated;
rreq->prev_donated = 0;
trace_netfs_sreq(subreq, netfs_sreq_trace_retry); trace_netfs_sreq(subreq, netfs_sreq_trace_retry);
spin_unlock(&rreq->lock);
BUG_ON(!len);
/* Renegotiate max_len (rsize) */ /* Renegotiate max_len (rsize) */
stream->sreq_max_len = subreq->len;
if (rreq->netfs_ops->prepare_read(subreq) < 0) { if (rreq->netfs_ops->prepare_read(subreq) < 0) {
trace_netfs_sreq(subreq, netfs_sreq_trace_reprep_failed); trace_netfs_sreq(subreq, netfs_sreq_trace_reprep_failed);
__set_bit(NETFS_SREQ_FAILED, &subreq->flags); __set_bit(NETFS_SREQ_FAILED, &subreq->flags);
goto abandon;
} }
part = umin(len, stream0->sreq_max_len); part = umin(len, stream->sreq_max_len);
if (unlikely(rreq->io_streams[0].sreq_max_segs)) if (unlikely(stream->sreq_max_segs))
part = netfs_limit_iter(&source, 0, part, stream0->sreq_max_segs); part = netfs_limit_iter(&source, 0, part, stream->sreq_max_segs);
subreq->len = subreq->transferred + part; subreq->len = subreq->transferred + part;
subreq->io_iter = source; subreq->io_iter = source;
iov_iter_truncate(&subreq->io_iter, part); iov_iter_truncate(&subreq->io_iter, part);
@ -166,58 +146,106 @@ static void netfs_retry_read_subrequests(struct netfs_io_request *rreq)
if (!len) { if (!len) {
if (boundary) if (boundary)
__set_bit(NETFS_SREQ_BOUNDARY, &subreq->flags); __set_bit(NETFS_SREQ_BOUNDARY, &subreq->flags);
subreq->next_donated = deferred_next_donated;
} else { } else {
__clear_bit(NETFS_SREQ_BOUNDARY, &subreq->flags); __clear_bit(NETFS_SREQ_BOUNDARY, &subreq->flags);
subreq->next_donated = 0;
} }
netfs_get_subrequest(subreq, netfs_sreq_trace_get_resubmit);
netfs_reissue_read(rreq, subreq); netfs_reissue_read(rreq, subreq);
if (!len) if (subreq == to)
break; break;
/* If we ran out of subrequests, allocate another. */
if (list_empty(&sublist)) {
subreq = netfs_alloc_subrequest(rreq);
if (!subreq)
goto abandon;
subreq->source = NETFS_DOWNLOAD_FROM_SERVER;
subreq->start = start;
/* We get two refs, but need just one. */
netfs_put_subrequest(subreq, false, netfs_sreq_trace_new);
trace_netfs_sreq(subreq, netfs_sreq_trace_split);
list_add_tail(&subreq->rreq_link, &sublist);
}
} }
/* If we managed to use fewer subreqs, we can discard the /* If we managed to use fewer subreqs, we can discard the
* excess. * excess; if we used the same number, then we're done.
*/ */
while ((subreq = list_first_entry_or_null( if (!len) {
&sublist, struct netfs_io_subrequest, rreq_link))) { if (subreq == to)
trace_netfs_sreq(subreq, netfs_sreq_trace_discard); continue;
list_del(&subreq->rreq_link); list_for_each_entry_safe_from(subreq, tmp,
netfs_put_subrequest(subreq, false, netfs_sreq_trace_put_done); &stream->subrequests, rreq_link) {
trace_netfs_sreq(subreq, netfs_sreq_trace_discard);
list_del(&subreq->rreq_link);
netfs_put_subrequest(subreq, false, netfs_sreq_trace_put_done);
if (subreq == to)
break;
}
continue;
} }
} while (!list_empty(&queue)); /* We ran out of subrequests, so we need to allocate some more
* and insert them after.
*/
do {
subreq = netfs_alloc_subrequest(rreq);
if (!subreq) {
subreq = to;
goto abandon_after;
}
subreq->source = NETFS_DOWNLOAD_FROM_SERVER;
subreq->start = start;
subreq->len = len;
subreq->debug_index = atomic_inc_return(&rreq->subreq_counter);
subreq->stream_nr = stream->stream_nr;
__set_bit(NETFS_SREQ_RETRYING, &subreq->flags);
trace_netfs_sreq_ref(rreq->debug_id, subreq->debug_index,
refcount_read(&subreq->ref),
netfs_sreq_trace_new);
netfs_get_subrequest(subreq, netfs_sreq_trace_get_resubmit);
list_add(&subreq->rreq_link, &to->rreq_link);
to = list_next_entry(to, rreq_link);
trace_netfs_sreq(subreq, netfs_sreq_trace_retry);
stream->sreq_max_len = umin(len, rreq->rsize);
stream->sreq_max_segs = 0;
if (unlikely(stream->sreq_max_segs))
part = netfs_limit_iter(&source, 0, part, stream->sreq_max_segs);
netfs_stat(&netfs_n_rh_download);
if (rreq->netfs_ops->prepare_read(subreq) < 0) {
trace_netfs_sreq(subreq, netfs_sreq_trace_reprep_failed);
__set_bit(NETFS_SREQ_FAILED, &subreq->flags);
goto abandon;
}
part = umin(len, stream->sreq_max_len);
subreq->len = subreq->transferred + part;
subreq->io_iter = source;
iov_iter_truncate(&subreq->io_iter, part);
iov_iter_advance(&source, part);
len -= part;
start += part;
if (!len && boundary) {
__set_bit(NETFS_SREQ_BOUNDARY, &to->flags);
boundary = false;
}
netfs_reissue_read(rreq, subreq);
} while (len);
} while (!list_is_head(next, &stream->subrequests));
return; return;
/* If we hit ENOMEM, fail all remaining subrequests */ /* If we hit an error, fail all remaining incomplete subrequests */
abandon_after:
if (list_is_last(&subreq->rreq_link, &stream->subrequests))
return;
subreq = list_next_entry(subreq, rreq_link);
abandon: abandon:
list_splice_init(&sublist, &queue); list_for_each_entry_from(subreq, &stream->subrequests, rreq_link) {
list_for_each_entry(subreq, &queue, rreq_link) { if (!subreq->error &&
if (!subreq->error) !test_bit(NETFS_SREQ_FAILED, &subreq->flags) &&
subreq->error = -ENOMEM; !test_bit(NETFS_SREQ_NEED_RETRY, &subreq->flags))
__clear_bit(NETFS_SREQ_FAILED, &subreq->flags); continue;
subreq->error = -ENOMEM;
__set_bit(NETFS_SREQ_FAILED, &subreq->flags);
__clear_bit(NETFS_SREQ_NEED_RETRY, &subreq->flags); __clear_bit(NETFS_SREQ_NEED_RETRY, &subreq->flags);
__clear_bit(NETFS_SREQ_RETRYING, &subreq->flags); __clear_bit(NETFS_SREQ_RETRYING, &subreq->flags);
} }
spin_lock(&rreq->lock);
list_splice_tail_init(&queue, &rreq->subrequests);
spin_unlock(&rreq->lock);
} }
/* /*
@ -225,14 +253,19 @@ abandon:
*/ */
void netfs_retry_reads(struct netfs_io_request *rreq) void netfs_retry_reads(struct netfs_io_request *rreq)
{ {
struct netfs_io_subrequest *subreq;
struct netfs_io_stream *stream = &rreq->io_streams[0];
/* Wait for all outstanding I/O to quiesce before performing retries as
* we may need to renegotiate the I/O sizes.
*/
list_for_each_entry(subreq, &stream->subrequests, rreq_link) {
wait_on_bit(&subreq->flags, NETFS_SREQ_IN_PROGRESS,
TASK_UNINTERRUPTIBLE);
}
trace_netfs_rreq(rreq, netfs_rreq_trace_resubmit); trace_netfs_rreq(rreq, netfs_rreq_trace_resubmit);
atomic_inc(&rreq->nr_outstanding);
netfs_retry_read_subrequests(rreq); netfs_retry_read_subrequests(rreq);
if (atomic_dec_and_test(&rreq->nr_outstanding))
netfs_rreq_terminated(rreq);
} }
/* /*

View File

@ -77,6 +77,7 @@ static void netfs_single_read_cache(struct netfs_io_request *rreq,
{ {
struct netfs_cache_resources *cres = &rreq->cache_resources; struct netfs_cache_resources *cres = &rreq->cache_resources;
_enter("R=%08x[%x]", rreq->debug_id, subreq->debug_index);
netfs_stat(&netfs_n_rh_read); netfs_stat(&netfs_n_rh_read);
cres->ops->read(cres, subreq->start, &subreq->io_iter, NETFS_READ_HOLE_FAIL, cres->ops->read(cres, subreq->start, &subreq->io_iter, NETFS_READ_HOLE_FAIL,
netfs_cache_read_terminated, subreq); netfs_cache_read_terminated, subreq);
@ -88,28 +89,28 @@ static void netfs_single_read_cache(struct netfs_io_request *rreq,
*/ */
static int netfs_single_dispatch_read(struct netfs_io_request *rreq) static int netfs_single_dispatch_read(struct netfs_io_request *rreq)
{ {
struct netfs_io_stream *stream = &rreq->io_streams[0];
struct netfs_io_subrequest *subreq; struct netfs_io_subrequest *subreq;
int ret = 0; int ret = 0;
atomic_set(&rreq->nr_outstanding, 1);
subreq = netfs_alloc_subrequest(rreq); subreq = netfs_alloc_subrequest(rreq);
if (!subreq) { if (!subreq)
ret = -ENOMEM; return -ENOMEM;
goto out;
}
subreq->source = NETFS_DOWNLOAD_FROM_SERVER; subreq->source = NETFS_DOWNLOAD_FROM_SERVER;
subreq->start = 0; subreq->start = 0;
subreq->len = rreq->len; subreq->len = rreq->len;
subreq->io_iter = rreq->buffer.iter; subreq->io_iter = rreq->buffer.iter;
atomic_inc(&rreq->nr_outstanding); __set_bit(NETFS_SREQ_IN_PROGRESS, &subreq->flags);
spin_lock_bh(&rreq->lock); spin_lock(&rreq->lock);
list_add_tail(&subreq->rreq_link, &rreq->subrequests); list_add_tail(&subreq->rreq_link, &stream->subrequests);
trace_netfs_sreq(subreq, netfs_sreq_trace_added); trace_netfs_sreq(subreq, netfs_sreq_trace_added);
spin_unlock_bh(&rreq->lock); stream->front = subreq;
/* Store list pointers before active flag */
smp_store_release(&stream->active, true);
spin_unlock(&rreq->lock);
netfs_single_cache_prepare_read(rreq, subreq); netfs_single_cache_prepare_read(rreq, subreq);
switch (subreq->source) { switch (subreq->source) {
@ -137,14 +138,12 @@ static int netfs_single_dispatch_read(struct netfs_io_request *rreq)
break; break;
} }
out: smp_wmb(); /* Write lists before ALL_QUEUED. */
if (atomic_dec_and_test(&rreq->nr_outstanding)) set_bit(NETFS_RREQ_ALL_QUEUED, &rreq->flags);
netfs_rreq_terminated(rreq);
return ret; return ret;
cancel: cancel:
atomic_dec(&rreq->nr_outstanding);
netfs_put_subrequest(subreq, false, netfs_sreq_trace_put_cancel); netfs_put_subrequest(subreq, false, netfs_sreq_trace_put_cancel);
goto out; return ret;
} }
/** /**
@ -185,13 +184,7 @@ ssize_t netfs_read_single(struct inode *inode, struct file *file, struct iov_ite
rreq->buffer.iter = *iter; rreq->buffer.iter = *iter;
netfs_single_dispatch_read(rreq); netfs_single_dispatch_read(rreq);
trace_netfs_rreq(rreq, netfs_rreq_trace_wait_ip); ret = netfs_wait_for_read(rreq);
wait_on_bit(&rreq->flags, NETFS_RREQ_IN_PROGRESS,
TASK_UNINTERRUPTIBLE);
ret = rreq->error;
if (ret == 0)
ret = rreq->transferred;
netfs_put_request(rreq, true, netfs_rreq_trace_put_return); netfs_put_request(rreq, true, netfs_rreq_trace_put_return);
return ret; return ret;

View File

@ -94,15 +94,21 @@ static void netfs_retry_write_stream(struct netfs_io_request *wreq,
list_for_each_entry_from(subreq, &stream->subrequests, rreq_link) { list_for_each_entry_from(subreq, &stream->subrequests, rreq_link) {
if (!len) if (!len)
break; break;
/* Renegotiate max_len (wsize) */
trace_netfs_sreq(subreq, netfs_sreq_trace_retry); subreq->start = start;
subreq->len = len;
__clear_bit(NETFS_SREQ_NEED_RETRY, &subreq->flags); __clear_bit(NETFS_SREQ_NEED_RETRY, &subreq->flags);
__set_bit(NETFS_SREQ_RETRYING, &subreq->flags); __set_bit(NETFS_SREQ_RETRYING, &subreq->flags);
trace_netfs_sreq(subreq, netfs_sreq_trace_retry);
/* Renegotiate max_len (wsize) */
stream->sreq_max_len = len;
stream->prepare_write(subreq); stream->prepare_write(subreq);
part = min(len, stream->sreq_max_len); part = umin(len, stream->sreq_max_len);
if (unlikely(stream->sreq_max_segs))
part = netfs_limit_iter(&source, 0, part, stream->sreq_max_segs);
subreq->len = part; subreq->len = part;
subreq->start = start;
subreq->transferred = 0; subreq->transferred = 0;
len -= part; len -= part;
start += part; start += part;

View File

@ -1320,6 +1320,8 @@ cifs_readv_callback(struct mid_q_entry *mid)
__set_bit(NETFS_SREQ_HIT_EOF, &rdata->subreq.flags); __set_bit(NETFS_SREQ_HIT_EOF, &rdata->subreq.flags);
rdata->result = 0; rdata->result = 0;
} }
if (rdata->got_bytes)
__set_bit(NETFS_SREQ_MADE_PROGRESS, &rdata->subreq.flags);
} }
rdata->credits.value = 0; rdata->credits.value = 0;

View File

@ -4607,6 +4607,8 @@ smb2_readv_callback(struct mid_q_entry *mid)
__set_bit(NETFS_SREQ_HIT_EOF, &rdata->subreq.flags); __set_bit(NETFS_SREQ_HIT_EOF, &rdata->subreq.flags);
rdata->result = 0; rdata->result = 0;
} }
if (rdata->got_bytes)
__set_bit(NETFS_SREQ_MADE_PROGRESS, &rdata->subreq.flags);
} }
trace_smb3_rw_credits(rreq_debug_id, subreq_debug_index, rdata->credits.value, trace_smb3_rw_credits(rreq_debug_id, subreq_debug_index, rdata->credits.value,
server->credits, server->in_flight, server->credits, server->in_flight,
@ -4615,7 +4617,7 @@ smb2_readv_callback(struct mid_q_entry *mid)
rdata->subreq.error = rdata->result; rdata->subreq.error = rdata->result;
rdata->subreq.transferred += rdata->got_bytes; rdata->subreq.transferred += rdata->got_bytes;
trace_netfs_sreq(&rdata->subreq, netfs_sreq_trace_io_progress); trace_netfs_sreq(&rdata->subreq, netfs_sreq_trace_io_progress);
queue_work(cifsiod_wq, &rdata->subreq.work); netfs_read_subreq_terminated(&rdata->subreq);
release_mid(mid); release_mid(mid);
trace_smb3_rw_credits(rreq_debug_id, subreq_debug_index, 0, trace_smb3_rw_credits(rreq_debug_id, subreq_debug_index, 0,
server->credits, server->in_flight, server->credits, server->in_flight,

View File

@ -181,23 +181,17 @@ struct netfs_io_subrequest {
unsigned long long start; /* Where to start the I/O */ unsigned long long start; /* Where to start the I/O */
size_t len; /* Size of the I/O */ size_t len; /* Size of the I/O */
size_t transferred; /* Amount of data transferred */ size_t transferred; /* Amount of data transferred */
size_t consumed; /* Amount of read data consumed */
size_t prev_donated; /* Amount of data donated from previous subreq */
size_t next_donated; /* Amount of data donated from next subreq */
refcount_t ref; refcount_t ref;
short error; /* 0 or error that occurred */ short error; /* 0 or error that occurred */
unsigned short debug_index; /* Index in list (for debugging output) */ unsigned short debug_index; /* Index in list (for debugging output) */
unsigned int nr_segs; /* Number of segs in io_iter */ unsigned int nr_segs; /* Number of segs in io_iter */
enum netfs_io_source source; /* Where to read from/write to */ enum netfs_io_source source; /* Where to read from/write to */
unsigned char stream_nr; /* I/O stream this belongs to */ unsigned char stream_nr; /* I/O stream this belongs to */
unsigned char curr_folioq_slot; /* Folio currently being read */
unsigned char curr_folio_order; /* Order of folio */
struct folio_queue *curr_folioq; /* Queue segment in which current folio resides */
unsigned long flags; unsigned long flags;
#define NETFS_SREQ_COPY_TO_CACHE 0 /* Set if should copy the data to the cache */ #define NETFS_SREQ_COPY_TO_CACHE 0 /* Set if should copy the data to the cache */
#define NETFS_SREQ_CLEAR_TAIL 1 /* Set if the rest of the read should be cleared */ #define NETFS_SREQ_CLEAR_TAIL 1 /* Set if the rest of the read should be cleared */
#define NETFS_SREQ_SEEK_DATA_READ 3 /* Set if ->read() should SEEK_DATA first */ #define NETFS_SREQ_SEEK_DATA_READ 3 /* Set if ->read() should SEEK_DATA first */
#define NETFS_SREQ_NO_PROGRESS 4 /* Set if we didn't manage to read any data */ #define NETFS_SREQ_MADE_PROGRESS 4 /* Set if we managed to read more data */
#define NETFS_SREQ_ONDEMAND 5 /* Set if it's from on-demand read mode */ #define NETFS_SREQ_ONDEMAND 5 /* Set if it's from on-demand read mode */
#define NETFS_SREQ_BOUNDARY 6 /* Set if ends on hard boundary (eg. ceph object) */ #define NETFS_SREQ_BOUNDARY 6 /* Set if ends on hard boundary (eg. ceph object) */
#define NETFS_SREQ_HIT_EOF 7 /* Set if short due to EOF */ #define NETFS_SREQ_HIT_EOF 7 /* Set if short due to EOF */
@ -238,13 +232,13 @@ struct netfs_io_request {
struct netfs_cache_resources cache_resources; struct netfs_cache_resources cache_resources;
struct readahead_control *ractl; /* Readahead descriptor */ struct readahead_control *ractl; /* Readahead descriptor */
struct list_head proc_link; /* Link in netfs_iorequests */ struct list_head proc_link; /* Link in netfs_iorequests */
struct list_head subrequests; /* Contributory I/O operations */
struct netfs_io_stream io_streams[2]; /* Streams of parallel I/O operations */ struct netfs_io_stream io_streams[2]; /* Streams of parallel I/O operations */
#define NR_IO_STREAMS 2 //wreq->nr_io_streams #define NR_IO_STREAMS 2 //wreq->nr_io_streams
struct netfs_group *group; /* Writeback group being written back */ struct netfs_group *group; /* Writeback group being written back */
struct rolling_buffer buffer; /* Unencrypted buffer */ struct rolling_buffer buffer; /* Unencrypted buffer */
#define NETFS_ROLLBUF_PUT_MARK ROLLBUF_MARK_1 #define NETFS_ROLLBUF_PUT_MARK ROLLBUF_MARK_1
#define NETFS_ROLLBUF_PAGECACHE_MARK ROLLBUF_MARK_2 #define NETFS_ROLLBUF_PAGECACHE_MARK ROLLBUF_MARK_2
wait_queue_head_t waitq; /* Processor waiter */
void *netfs_priv; /* Private data for the netfs */ void *netfs_priv; /* Private data for the netfs */
void *netfs_priv2; /* Private data for the netfs */ void *netfs_priv2; /* Private data for the netfs */
struct bio_vec *direct_bv; /* DIO buffer list (when handling iovec-iter) */ struct bio_vec *direct_bv; /* DIO buffer list (when handling iovec-iter) */
@ -255,7 +249,6 @@ struct netfs_io_request {
atomic_t subreq_counter; /* Next subreq->debug_index */ atomic_t subreq_counter; /* Next subreq->debug_index */
unsigned int nr_group_rel; /* Number of refs to release on ->group */ unsigned int nr_group_rel; /* Number of refs to release on ->group */
spinlock_t lock; /* Lock for queuing subreqs */ spinlock_t lock; /* Lock for queuing subreqs */
atomic_t nr_outstanding; /* Number of ops in progress */
unsigned long long submitted; /* Amount submitted for I/O so far */ unsigned long long submitted; /* Amount submitted for I/O so far */
unsigned long long len; /* Length of the request */ unsigned long long len; /* Length of the request */
size_t transferred; /* Amount to be indicated as transferred */ size_t transferred; /* Amount to be indicated as transferred */
@ -267,15 +260,18 @@ struct netfs_io_request {
atomic64_t issued_to; /* Write issuer folio cursor */ atomic64_t issued_to; /* Write issuer folio cursor */
unsigned long long collected_to; /* Point we've collected to */ unsigned long long collected_to; /* Point we've collected to */
unsigned long long cleaned_to; /* Position we've cleaned folios to */ unsigned long long cleaned_to; /* Position we've cleaned folios to */
unsigned long long abandon_to; /* Position to abandon folios to */
pgoff_t no_unlock_folio; /* Don't unlock this folio after read */ pgoff_t no_unlock_folio; /* Don't unlock this folio after read */
size_t prev_donated; /* Fallback for subreq->prev_donated */ unsigned char front_folio_order; /* Order (size) of front folio */
refcount_t ref; refcount_t ref;
unsigned long flags; unsigned long flags;
#define NETFS_RREQ_OFFLOAD_COLLECTION 0 /* Offload collection to workqueue */
#define NETFS_RREQ_COPY_TO_CACHE 1 /* Need to write to the cache */ #define NETFS_RREQ_COPY_TO_CACHE 1 /* Need to write to the cache */
#define NETFS_RREQ_NO_UNLOCK_FOLIO 2 /* Don't unlock no_unlock_folio on completion */ #define NETFS_RREQ_NO_UNLOCK_FOLIO 2 /* Don't unlock no_unlock_folio on completion */
#define NETFS_RREQ_DONT_UNLOCK_FOLIOS 3 /* Don't unlock the folios on completion */ #define NETFS_RREQ_DONT_UNLOCK_FOLIOS 3 /* Don't unlock the folios on completion */
#define NETFS_RREQ_FAILED 4 /* The request failed */ #define NETFS_RREQ_FAILED 4 /* The request failed */
#define NETFS_RREQ_IN_PROGRESS 5 /* Unlocked when the request completes */ #define NETFS_RREQ_IN_PROGRESS 5 /* Unlocked when the request completes */
#define NETFS_RREQ_FOLIO_COPY_TO_CACHE 6 /* Copy current folio to cache from read */
#define NETFS_RREQ_UPLOAD_TO_SERVER 8 /* Need to write to the server */ #define NETFS_RREQ_UPLOAD_TO_SERVER 8 /* Need to write to the server */
#define NETFS_RREQ_NONBLOCK 9 /* Don't block if possible (O_NONBLOCK) */ #define NETFS_RREQ_NONBLOCK 9 /* Don't block if possible (O_NONBLOCK) */
#define NETFS_RREQ_BLOCKED 10 /* We blocked */ #define NETFS_RREQ_BLOCKED 10 /* We blocked */
@ -440,7 +436,6 @@ vm_fault_t netfs_page_mkwrite(struct vm_fault *vmf, struct netfs_group *netfs_gr
/* (Sub)request management API. */ /* (Sub)request management API. */
void netfs_read_subreq_progress(struct netfs_io_subrequest *subreq); void netfs_read_subreq_progress(struct netfs_io_subrequest *subreq);
void netfs_read_subreq_terminated(struct netfs_io_subrequest *subreq); void netfs_read_subreq_terminated(struct netfs_io_subrequest *subreq);
void netfs_read_subreq_termination_worker(struct work_struct *work);
void netfs_get_subrequest(struct netfs_io_subrequest *subreq, void netfs_get_subrequest(struct netfs_io_subrequest *subreq,
enum netfs_sreq_ref_trace what); enum netfs_sreq_ref_trace what);
void netfs_put_subrequest(struct netfs_io_subrequest *subreq, void netfs_put_subrequest(struct netfs_io_subrequest *subreq,

View File

@ -50,18 +50,23 @@
EM(netfs_rreq_trace_assess, "ASSESS ") \ EM(netfs_rreq_trace_assess, "ASSESS ") \
EM(netfs_rreq_trace_copy, "COPY ") \ EM(netfs_rreq_trace_copy, "COPY ") \
EM(netfs_rreq_trace_collect, "COLLECT") \ EM(netfs_rreq_trace_collect, "COLLECT") \
EM(netfs_rreq_trace_complete, "COMPLET") \
EM(netfs_rreq_trace_dirty, "DIRTY ") \ EM(netfs_rreq_trace_dirty, "DIRTY ") \
EM(netfs_rreq_trace_done, "DONE ") \ EM(netfs_rreq_trace_done, "DONE ") \
EM(netfs_rreq_trace_free, "FREE ") \ EM(netfs_rreq_trace_free, "FREE ") \
EM(netfs_rreq_trace_redirty, "REDIRTY") \ EM(netfs_rreq_trace_redirty, "REDIRTY") \
EM(netfs_rreq_trace_resubmit, "RESUBMT") \ EM(netfs_rreq_trace_resubmit, "RESUBMT") \
EM(netfs_rreq_trace_set_abandon, "S-ABNDN") \
EM(netfs_rreq_trace_set_pause, "PAUSE ") \ EM(netfs_rreq_trace_set_pause, "PAUSE ") \
EM(netfs_rreq_trace_unlock, "UNLOCK ") \ EM(netfs_rreq_trace_unlock, "UNLOCK ") \
EM(netfs_rreq_trace_unlock_pgpriv2, "UNLCK-2") \ EM(netfs_rreq_trace_unlock_pgpriv2, "UNLCK-2") \
EM(netfs_rreq_trace_unmark, "UNMARK ") \ EM(netfs_rreq_trace_unmark, "UNMARK ") \
EM(netfs_rreq_trace_wait_ip, "WAIT-IP") \ EM(netfs_rreq_trace_wait_ip, "WAIT-IP") \
EM(netfs_rreq_trace_wait_pause, "WT-PAUS") \ EM(netfs_rreq_trace_wait_pause, "WT-PAUS") \
EM(netfs_rreq_trace_wait_queue, "WAIT-Q ") \
EM(netfs_rreq_trace_wake_ip, "WAKE-IP") \ EM(netfs_rreq_trace_wake_ip, "WAKE-IP") \
EM(netfs_rreq_trace_wake_queue, "WAKE-Q ") \
EM(netfs_rreq_trace_woke_queue, "WOKE-Q ") \
EM(netfs_rreq_trace_unpause, "UNPAUSE") \ EM(netfs_rreq_trace_unpause, "UNPAUSE") \
E_(netfs_rreq_trace_write_done, "WR-DONE") E_(netfs_rreq_trace_write_done, "WR-DONE")
@ -91,6 +96,8 @@
EM(netfs_sreq_trace_hit_eof, "EOF ") \ EM(netfs_sreq_trace_hit_eof, "EOF ") \
EM(netfs_sreq_trace_io_progress, "IO ") \ EM(netfs_sreq_trace_io_progress, "IO ") \
EM(netfs_sreq_trace_limited, "LIMIT") \ EM(netfs_sreq_trace_limited, "LIMIT") \
EM(netfs_sreq_trace_partial_read, "PARTR") \
EM(netfs_sreq_trace_need_retry, "NRTRY") \
EM(netfs_sreq_trace_prepare, "PREP ") \ EM(netfs_sreq_trace_prepare, "PREP ") \
EM(netfs_sreq_trace_prep_failed, "PRPFL") \ EM(netfs_sreq_trace_prep_failed, "PRPFL") \
EM(netfs_sreq_trace_progress, "PRGRS") \ EM(netfs_sreq_trace_progress, "PRGRS") \
@ -176,6 +183,7 @@
EM(netfs_folio_trace_mkwrite, "mkwrite") \ EM(netfs_folio_trace_mkwrite, "mkwrite") \
EM(netfs_folio_trace_mkwrite_plus, "mkwrite+") \ EM(netfs_folio_trace_mkwrite_plus, "mkwrite+") \
EM(netfs_folio_trace_not_under_wback, "!wback") \ EM(netfs_folio_trace_not_under_wback, "!wback") \
EM(netfs_folio_trace_not_locked, "!locked") \
EM(netfs_folio_trace_put, "put") \ EM(netfs_folio_trace_put, "put") \
EM(netfs_folio_trace_read, "read") \ EM(netfs_folio_trace_read, "read") \
EM(netfs_folio_trace_read_done, "read-done") \ EM(netfs_folio_trace_read_done, "read-done") \
@ -204,7 +212,6 @@
EM(netfs_trace_folioq_clear, "clear") \ EM(netfs_trace_folioq_clear, "clear") \
EM(netfs_trace_folioq_delete, "delete") \ EM(netfs_trace_folioq_delete, "delete") \
EM(netfs_trace_folioq_make_space, "make-space") \ EM(netfs_trace_folioq_make_space, "make-space") \
EM(netfs_trace_folioq_prep_write, "prep-wr") \
EM(netfs_trace_folioq_rollbuf_init, "roll-init") \ EM(netfs_trace_folioq_rollbuf_init, "roll-init") \
E_(netfs_trace_folioq_read_progress, "r-progress") E_(netfs_trace_folioq_read_progress, "r-progress")
@ -352,7 +359,7 @@ TRACE_EVENT(netfs_sreq,
__entry->len = sreq->len; __entry->len = sreq->len;
__entry->transferred = sreq->transferred; __entry->transferred = sreq->transferred;
__entry->start = sreq->start; __entry->start = sreq->start;
__entry->slot = sreq->curr_folioq_slot; __entry->slot = sreq->io_iter.folioq_slot;
), ),
TP_printk("R=%08x[%x] %s %s f=%02x s=%llx %zx/%zx s=%u e=%d", TP_printk("R=%08x[%x] %s %s f=%02x s=%llx %zx/%zx s=%u e=%d",
@ -701,71 +708,6 @@ TRACE_EVENT(netfs_collect_stream,
__entry->collected_to, __entry->front) __entry->collected_to, __entry->front)
); );
TRACE_EVENT(netfs_progress,
TP_PROTO(const struct netfs_io_subrequest *subreq,
unsigned long long start, size_t avail, size_t part),
TP_ARGS(subreq, start, avail, part),
TP_STRUCT__entry(
__field(unsigned int, rreq)
__field(unsigned int, subreq)
__field(unsigned int, consumed)
__field(unsigned int, transferred)
__field(unsigned long long, f_start)
__field(unsigned int, f_avail)
__field(unsigned int, f_part)
__field(unsigned char, slot)
),
TP_fast_assign(
__entry->rreq = subreq->rreq->debug_id;
__entry->subreq = subreq->debug_index;
__entry->consumed = subreq->consumed;
__entry->transferred = subreq->transferred;
__entry->f_start = start;
__entry->f_avail = avail;
__entry->f_part = part;
__entry->slot = subreq->curr_folioq_slot;
),
TP_printk("R=%08x[%02x] s=%llx ct=%x/%x pa=%x/%x sl=%x",
__entry->rreq, __entry->subreq, __entry->f_start,
__entry->consumed, __entry->transferred,
__entry->f_part, __entry->f_avail, __entry->slot)
);
TRACE_EVENT(netfs_donate,
TP_PROTO(const struct netfs_io_request *rreq,
const struct netfs_io_subrequest *from,
const struct netfs_io_subrequest *to,
size_t amount,
enum netfs_donate_trace trace),
TP_ARGS(rreq, from, to, amount, trace),
TP_STRUCT__entry(
__field(unsigned int, rreq)
__field(unsigned int, from)
__field(unsigned int, to)
__field(unsigned int, amount)
__field(enum netfs_donate_trace, trace)
),
TP_fast_assign(
__entry->rreq = rreq->debug_id;
__entry->from = from->debug_index;
__entry->to = to ? to->debug_index : -1;
__entry->amount = amount;
__entry->trace = trace;
),
TP_printk("R=%08x[%02x] -> [%02x] %s am=%x",
__entry->rreq, __entry->from, __entry->to,
__print_symbolic(__entry->trace, netfs_donate_traces),
__entry->amount)
);
TRACE_EVENT(netfs_folioq, TRACE_EVENT(netfs_folioq,
TP_PROTO(const struct folio_queue *fq, TP_PROTO(const struct folio_queue *fq,
enum netfs_folioq_trace trace), enum netfs_folioq_trace trace),