mirror of https://gitee.com/openkylin/linux.git
1264 lines
37 KiB
C
1264 lines
37 KiB
C
// SPDX-License-Identifier: GPL-2.0+
|
|
/*
|
|
* SSH request transport layer.
|
|
*
|
|
* Copyright (C) 2019-2020 Maximilian Luz <luzmaximilian@gmail.com>
|
|
*/
|
|
|
|
#include <asm/unaligned.h>
|
|
#include <linux/atomic.h>
|
|
#include <linux/completion.h>
|
|
#include <linux/error-injection.h>
|
|
#include <linux/ktime.h>
|
|
#include <linux/limits.h>
|
|
#include <linux/list.h>
|
|
#include <linux/slab.h>
|
|
#include <linux/spinlock.h>
|
|
#include <linux/types.h>
|
|
#include <linux/workqueue.h>
|
|
|
|
#include <linux/surface_aggregator/serial_hub.h>
|
|
#include <linux/surface_aggregator/controller.h>
|
|
|
|
#include "ssh_packet_layer.h"
|
|
#include "ssh_request_layer.h"
|
|
|
|
#include "trace.h"
|
|
|
|
/*
|
|
* SSH_RTL_REQUEST_TIMEOUT - Request timeout.
|
|
*
|
|
* Timeout as ktime_t delta for request responses. If we have not received a
|
|
* response in this time-frame after finishing the underlying packet
|
|
* transmission, the request will be completed with %-ETIMEDOUT as status
|
|
* code.
|
|
*/
|
|
#define SSH_RTL_REQUEST_TIMEOUT ms_to_ktime(3000)
|
|
|
|
/*
|
|
* SSH_RTL_REQUEST_TIMEOUT_RESOLUTION - Request timeout granularity.
|
|
*
|
|
* Time-resolution for timeouts. Should be larger than one jiffy to avoid
|
|
* direct re-scheduling of reaper work_struct.
|
|
*/
|
|
#define SSH_RTL_REQUEST_TIMEOUT_RESOLUTION ms_to_ktime(max(2000 / HZ, 50))
|
|
|
|
/*
|
|
* SSH_RTL_MAX_PENDING - Maximum number of pending requests.
|
|
*
|
|
* Maximum number of requests concurrently waiting to be completed (i.e.
|
|
* waiting for the corresponding packet transmission to finish if they don't
|
|
* have a response or waiting for a response if they have one).
|
|
*/
|
|
#define SSH_RTL_MAX_PENDING 3
|
|
|
|
/*
|
|
* SSH_RTL_TX_BATCH - Maximum number of requests processed per work execution.
|
|
* Used to prevent livelocking of the workqueue. Value chosen via educated
|
|
* guess, may be adjusted.
|
|
*/
|
|
#define SSH_RTL_TX_BATCH 10
|
|
|
|
#ifdef CONFIG_SURFACE_AGGREGATOR_ERROR_INJECTION
|
|
|
|
/**
|
|
* ssh_rtl_should_drop_response() - Error injection hook to drop request
|
|
* responses.
|
|
*
|
|
* Useful to cause request transmission timeouts in the driver by dropping the
|
|
* response to a request.
|
|
*/
|
|
static noinline bool ssh_rtl_should_drop_response(void)
|
|
{
|
|
return false;
|
|
}
|
|
ALLOW_ERROR_INJECTION(ssh_rtl_should_drop_response, TRUE);
|
|
|
|
#else
|
|
|
|
static inline bool ssh_rtl_should_drop_response(void)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
#endif
|
|
|
|
static u16 ssh_request_get_rqid(struct ssh_request *rqst)
|
|
{
|
|
return get_unaligned_le16(rqst->packet.data.ptr
|
|
+ SSH_MSGOFFSET_COMMAND(rqid));
|
|
}
|
|
|
|
static u32 ssh_request_get_rqid_safe(struct ssh_request *rqst)
|
|
{
|
|
if (!rqst->packet.data.ptr)
|
|
return U32_MAX;
|
|
|
|
return ssh_request_get_rqid(rqst);
|
|
}
|
|
|
|
static void ssh_rtl_queue_remove(struct ssh_request *rqst)
|
|
{
|
|
struct ssh_rtl *rtl = ssh_request_rtl(rqst);
|
|
|
|
spin_lock(&rtl->queue.lock);
|
|
|
|
if (!test_and_clear_bit(SSH_REQUEST_SF_QUEUED_BIT, &rqst->state)) {
|
|
spin_unlock(&rtl->queue.lock);
|
|
return;
|
|
}
|
|
|
|
list_del(&rqst->node);
|
|
|
|
spin_unlock(&rtl->queue.lock);
|
|
ssh_request_put(rqst);
|
|
}
|
|
|
|
static bool ssh_rtl_queue_empty(struct ssh_rtl *rtl)
|
|
{
|
|
bool empty;
|
|
|
|
spin_lock(&rtl->queue.lock);
|
|
empty = list_empty(&rtl->queue.head);
|
|
spin_unlock(&rtl->queue.lock);
|
|
|
|
return empty;
|
|
}
|
|
|
|
static void ssh_rtl_pending_remove(struct ssh_request *rqst)
|
|
{
|
|
struct ssh_rtl *rtl = ssh_request_rtl(rqst);
|
|
|
|
spin_lock(&rtl->pending.lock);
|
|
|
|
if (!test_and_clear_bit(SSH_REQUEST_SF_PENDING_BIT, &rqst->state)) {
|
|
spin_unlock(&rtl->pending.lock);
|
|
return;
|
|
}
|
|
|
|
atomic_dec(&rtl->pending.count);
|
|
list_del(&rqst->node);
|
|
|
|
spin_unlock(&rtl->pending.lock);
|
|
|
|
ssh_request_put(rqst);
|
|
}
|
|
|
|
static int ssh_rtl_tx_pending_push(struct ssh_request *rqst)
|
|
{
|
|
struct ssh_rtl *rtl = ssh_request_rtl(rqst);
|
|
|
|
spin_lock(&rtl->pending.lock);
|
|
|
|
if (test_bit(SSH_REQUEST_SF_LOCKED_BIT, &rqst->state)) {
|
|
spin_unlock(&rtl->pending.lock);
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (test_and_set_bit(SSH_REQUEST_SF_PENDING_BIT, &rqst->state)) {
|
|
spin_unlock(&rtl->pending.lock);
|
|
return -EALREADY;
|
|
}
|
|
|
|
atomic_inc(&rtl->pending.count);
|
|
list_add_tail(&ssh_request_get(rqst)->node, &rtl->pending.head);
|
|
|
|
spin_unlock(&rtl->pending.lock);
|
|
return 0;
|
|
}
|
|
|
|
static void ssh_rtl_complete_with_status(struct ssh_request *rqst, int status)
|
|
{
|
|
struct ssh_rtl *rtl = ssh_request_rtl(rqst);
|
|
|
|
trace_ssam_request_complete(rqst, status);
|
|
|
|
/* rtl/ptl may not be set if we're canceling before submitting. */
|
|
rtl_dbg_cond(rtl, "rtl: completing request (rqid: %#06x, status: %d)\n",
|
|
ssh_request_get_rqid_safe(rqst), status);
|
|
|
|
rqst->ops->complete(rqst, NULL, NULL, status);
|
|
}
|
|
|
|
static void ssh_rtl_complete_with_rsp(struct ssh_request *rqst,
|
|
const struct ssh_command *cmd,
|
|
const struct ssam_span *data)
|
|
{
|
|
struct ssh_rtl *rtl = ssh_request_rtl(rqst);
|
|
|
|
trace_ssam_request_complete(rqst, 0);
|
|
|
|
rtl_dbg(rtl, "rtl: completing request with response (rqid: %#06x)\n",
|
|
ssh_request_get_rqid(rqst));
|
|
|
|
rqst->ops->complete(rqst, cmd, data, 0);
|
|
}
|
|
|
|
static bool ssh_rtl_tx_can_process(struct ssh_request *rqst)
|
|
{
|
|
struct ssh_rtl *rtl = ssh_request_rtl(rqst);
|
|
|
|
if (test_bit(SSH_REQUEST_TY_FLUSH_BIT, &rqst->state))
|
|
return !atomic_read(&rtl->pending.count);
|
|
|
|
return atomic_read(&rtl->pending.count) < SSH_RTL_MAX_PENDING;
|
|
}
|
|
|
|
static struct ssh_request *ssh_rtl_tx_next(struct ssh_rtl *rtl)
|
|
{
|
|
struct ssh_request *rqst = ERR_PTR(-ENOENT);
|
|
struct ssh_request *p, *n;
|
|
|
|
spin_lock(&rtl->queue.lock);
|
|
|
|
/* Find first non-locked request and remove it. */
|
|
list_for_each_entry_safe(p, n, &rtl->queue.head, node) {
|
|
if (unlikely(test_bit(SSH_REQUEST_SF_LOCKED_BIT, &p->state)))
|
|
continue;
|
|
|
|
if (!ssh_rtl_tx_can_process(p)) {
|
|
rqst = ERR_PTR(-EBUSY);
|
|
break;
|
|
}
|
|
|
|
/* Remove from queue and mark as transmitting. */
|
|
set_bit(SSH_REQUEST_SF_TRANSMITTING_BIT, &p->state);
|
|
/* Ensure state never gets zero. */
|
|
smp_mb__before_atomic();
|
|
clear_bit(SSH_REQUEST_SF_QUEUED_BIT, &p->state);
|
|
|
|
list_del(&p->node);
|
|
|
|
rqst = p;
|
|
break;
|
|
}
|
|
|
|
spin_unlock(&rtl->queue.lock);
|
|
return rqst;
|
|
}
|
|
|
|
static int ssh_rtl_tx_try_process_one(struct ssh_rtl *rtl)
|
|
{
|
|
struct ssh_request *rqst;
|
|
int status;
|
|
|
|
/* Get and prepare next request for transmit. */
|
|
rqst = ssh_rtl_tx_next(rtl);
|
|
if (IS_ERR(rqst))
|
|
return PTR_ERR(rqst);
|
|
|
|
/* Add it to/mark it as pending. */
|
|
status = ssh_rtl_tx_pending_push(rqst);
|
|
if (status) {
|
|
ssh_request_put(rqst);
|
|
return -EAGAIN;
|
|
}
|
|
|
|
/* Submit packet. */
|
|
status = ssh_ptl_submit(&rtl->ptl, &rqst->packet);
|
|
if (status == -ESHUTDOWN) {
|
|
/*
|
|
* Packet has been refused due to the packet layer shutting
|
|
* down. Complete it here.
|
|
*/
|
|
set_bit(SSH_REQUEST_SF_LOCKED_BIT, &rqst->state);
|
|
/*
|
|
* Note: A barrier is not required here, as there are only two
|
|
* references in the system at this point: The one that we have,
|
|
* and the other one that belongs to the pending set. Due to the
|
|
* request being marked as "transmitting", our process is the
|
|
* only one allowed to remove the pending node and change the
|
|
* state. Normally, the task would fall to the packet callback,
|
|
* but as this is a path where submission failed, this callback
|
|
* will never be executed.
|
|
*/
|
|
|
|
ssh_rtl_pending_remove(rqst);
|
|
ssh_rtl_complete_with_status(rqst, -ESHUTDOWN);
|
|
|
|
ssh_request_put(rqst);
|
|
return -ESHUTDOWN;
|
|
|
|
} else if (status) {
|
|
/*
|
|
* If submitting the packet failed and the packet layer isn't
|
|
* shutting down, the packet has either been submitted/queued
|
|
* before (-EALREADY, which cannot happen as we have
|
|
* guaranteed that requests cannot be re-submitted), or the
|
|
* packet was marked as locked (-EINVAL). To mark the packet
|
|
* locked at this stage, the request, and thus the packets
|
|
* itself, had to have been canceled. Simply drop the
|
|
* reference. Cancellation itself will remove it from the set
|
|
* of pending requests.
|
|
*/
|
|
|
|
WARN_ON(status != -EINVAL);
|
|
|
|
ssh_request_put(rqst);
|
|
return -EAGAIN;
|
|
}
|
|
|
|
ssh_request_put(rqst);
|
|
return 0;
|
|
}
|
|
|
|
static bool ssh_rtl_tx_schedule(struct ssh_rtl *rtl)
|
|
{
|
|
if (atomic_read(&rtl->pending.count) >= SSH_RTL_MAX_PENDING)
|
|
return false;
|
|
|
|
if (ssh_rtl_queue_empty(rtl))
|
|
return false;
|
|
|
|
return schedule_work(&rtl->tx.work);
|
|
}
|
|
|
|
static void ssh_rtl_tx_work_fn(struct work_struct *work)
|
|
{
|
|
struct ssh_rtl *rtl = to_ssh_rtl(work, tx.work);
|
|
unsigned int iterations = SSH_RTL_TX_BATCH;
|
|
int status;
|
|
|
|
/*
|
|
* Try to be nice and not block/live-lock the workqueue: Run a maximum
|
|
* of 10 tries, then re-submit if necessary. This should not be
|
|
* necessary for normal execution, but guarantee it anyway.
|
|
*/
|
|
do {
|
|
status = ssh_rtl_tx_try_process_one(rtl);
|
|
if (status == -ENOENT || status == -EBUSY)
|
|
return; /* No more requests to process. */
|
|
|
|
if (status == -ESHUTDOWN) {
|
|
/*
|
|
* Packet system shutting down. No new packets can be
|
|
* transmitted. Return silently, the party initiating
|
|
* the shutdown should handle the rest.
|
|
*/
|
|
return;
|
|
}
|
|
|
|
WARN_ON(status != 0 && status != -EAGAIN);
|
|
} while (--iterations);
|
|
|
|
/* Out of tries, reschedule. */
|
|
ssh_rtl_tx_schedule(rtl);
|
|
}
|
|
|
|
/**
|
|
* ssh_rtl_submit() - Submit a request to the transport layer.
|
|
* @rtl: The request transport layer.
|
|
* @rqst: The request to submit.
|
|
*
|
|
* Submits a request to the transport layer. A single request may not be
|
|
* submitted multiple times without reinitializing it.
|
|
*
|
|
* Return: Returns zero on success, %-EINVAL if the request type is invalid or
|
|
* the request has been canceled prior to submission, %-EALREADY if the
|
|
* request has already been submitted, or %-ESHUTDOWN in case the request
|
|
* transport layer has been shut down.
|
|
*/
|
|
int ssh_rtl_submit(struct ssh_rtl *rtl, struct ssh_request *rqst)
|
|
{
|
|
trace_ssam_request_submit(rqst);
|
|
|
|
/*
|
|
* Ensure that requests expecting a response are sequenced. If this
|
|
* invariant ever changes, see the comment in ssh_rtl_complete() on what
|
|
* is required to be changed in the code.
|
|
*/
|
|
if (test_bit(SSH_REQUEST_TY_HAS_RESPONSE_BIT, &rqst->state))
|
|
if (!test_bit(SSH_PACKET_TY_SEQUENCED_BIT, &rqst->packet.state))
|
|
return -EINVAL;
|
|
|
|
spin_lock(&rtl->queue.lock);
|
|
|
|
/*
|
|
* Try to set ptl and check if this request has already been submitted.
|
|
*
|
|
* Must be inside lock as we might run into a lost update problem
|
|
* otherwise: If this were outside of the lock, cancellation in
|
|
* ssh_rtl_cancel_nonpending() may run after we've set the ptl
|
|
* reference but before we enter the lock. In that case, we'd detect
|
|
* that the request is being added to the queue and would try to remove
|
|
* it from that, but removal might fail because it hasn't actually been
|
|
* added yet. By putting this cmpxchg in the critical section, we
|
|
* ensure that the queuing detection only triggers when we are already
|
|
* in the critical section and the remove process will wait until the
|
|
* push operation has been completed (via lock) due to that. Only then,
|
|
* we can safely try to remove it.
|
|
*/
|
|
if (cmpxchg(&rqst->packet.ptl, NULL, &rtl->ptl)) {
|
|
spin_unlock(&rtl->queue.lock);
|
|
return -EALREADY;
|
|
}
|
|
|
|
/*
|
|
* Ensure that we set ptl reference before we continue modifying state.
|
|
* This is required for non-pending cancellation. This barrier is paired
|
|
* with the one in ssh_rtl_cancel_nonpending().
|
|
*
|
|
* By setting the ptl reference before we test for "locked", we can
|
|
* check if the "locked" test may have already run. See comments in
|
|
* ssh_rtl_cancel_nonpending() for more detail.
|
|
*/
|
|
smp_mb__after_atomic();
|
|
|
|
if (test_bit(SSH_RTL_SF_SHUTDOWN_BIT, &rtl->state)) {
|
|
spin_unlock(&rtl->queue.lock);
|
|
return -ESHUTDOWN;
|
|
}
|
|
|
|
if (test_bit(SSH_REQUEST_SF_LOCKED_BIT, &rqst->state)) {
|
|
spin_unlock(&rtl->queue.lock);
|
|
return -EINVAL;
|
|
}
|
|
|
|
set_bit(SSH_REQUEST_SF_QUEUED_BIT, &rqst->state);
|
|
list_add_tail(&ssh_request_get(rqst)->node, &rtl->queue.head);
|
|
|
|
spin_unlock(&rtl->queue.lock);
|
|
|
|
ssh_rtl_tx_schedule(rtl);
|
|
return 0;
|
|
}
|
|
|
|
static void ssh_rtl_timeout_reaper_mod(struct ssh_rtl *rtl, ktime_t now,
|
|
ktime_t expires)
|
|
{
|
|
unsigned long delta = msecs_to_jiffies(ktime_ms_delta(expires, now));
|
|
ktime_t aexp = ktime_add(expires, SSH_RTL_REQUEST_TIMEOUT_RESOLUTION);
|
|
|
|
spin_lock(&rtl->rtx_timeout.lock);
|
|
|
|
/* Re-adjust / schedule reaper only if it is above resolution delta. */
|
|
if (ktime_before(aexp, rtl->rtx_timeout.expires)) {
|
|
rtl->rtx_timeout.expires = expires;
|
|
mod_delayed_work(system_wq, &rtl->rtx_timeout.reaper, delta);
|
|
}
|
|
|
|
spin_unlock(&rtl->rtx_timeout.lock);
|
|
}
|
|
|
|
static void ssh_rtl_timeout_start(struct ssh_request *rqst)
|
|
{
|
|
struct ssh_rtl *rtl = ssh_request_rtl(rqst);
|
|
ktime_t timestamp = ktime_get_coarse_boottime();
|
|
ktime_t timeout = rtl->rtx_timeout.timeout;
|
|
|
|
if (test_bit(SSH_REQUEST_SF_LOCKED_BIT, &rqst->state))
|
|
return;
|
|
|
|
/*
|
|
* Note: The timestamp gets set only once. This happens on the packet
|
|
* callback. All other access to it is read-only.
|
|
*/
|
|
WRITE_ONCE(rqst->timestamp, timestamp);
|
|
/*
|
|
* Ensure timestamp is set before starting the reaper. Paired with
|
|
* implicit barrier following check on ssh_request_get_expiration() in
|
|
* ssh_rtl_timeout_reap.
|
|
*/
|
|
smp_mb__after_atomic();
|
|
|
|
ssh_rtl_timeout_reaper_mod(rtl, timestamp, timestamp + timeout);
|
|
}
|
|
|
|
static void ssh_rtl_complete(struct ssh_rtl *rtl,
|
|
const struct ssh_command *command,
|
|
const struct ssam_span *command_data)
|
|
{
|
|
struct ssh_request *r = NULL;
|
|
struct ssh_request *p, *n;
|
|
u16 rqid = get_unaligned_le16(&command->rqid);
|
|
|
|
trace_ssam_rx_response_received(command, command_data->len);
|
|
|
|
/*
|
|
* Get request from pending based on request ID and mark it as response
|
|
* received and locked.
|
|
*/
|
|
spin_lock(&rtl->pending.lock);
|
|
list_for_each_entry_safe(p, n, &rtl->pending.head, node) {
|
|
/* We generally expect requests to be processed in order. */
|
|
if (unlikely(ssh_request_get_rqid(p) != rqid))
|
|
continue;
|
|
|
|
/* Simulate response timeout. */
|
|
if (ssh_rtl_should_drop_response()) {
|
|
spin_unlock(&rtl->pending.lock);
|
|
|
|
trace_ssam_ei_rx_drop_response(p);
|
|
rtl_info(rtl, "request error injection: dropping response for request %p\n",
|
|
&p->packet);
|
|
return;
|
|
}
|
|
|
|
/*
|
|
* Mark as "response received" and "locked" as we're going to
|
|
* complete it.
|
|
*/
|
|
set_bit(SSH_REQUEST_SF_LOCKED_BIT, &p->state);
|
|
set_bit(SSH_REQUEST_SF_RSPRCVD_BIT, &p->state);
|
|
/* Ensure state never gets zero. */
|
|
smp_mb__before_atomic();
|
|
clear_bit(SSH_REQUEST_SF_PENDING_BIT, &p->state);
|
|
|
|
atomic_dec(&rtl->pending.count);
|
|
list_del(&p->node);
|
|
|
|
r = p;
|
|
break;
|
|
}
|
|
spin_unlock(&rtl->pending.lock);
|
|
|
|
if (!r) {
|
|
rtl_warn(rtl, "rtl: dropping unexpected command message (rqid = %#06x)\n",
|
|
rqid);
|
|
return;
|
|
}
|
|
|
|
/* If the request hasn't been completed yet, we will do this now. */
|
|
if (test_and_set_bit(SSH_REQUEST_SF_COMPLETED_BIT, &r->state)) {
|
|
ssh_request_put(r);
|
|
ssh_rtl_tx_schedule(rtl);
|
|
return;
|
|
}
|
|
|
|
/*
|
|
* Make sure the request has been transmitted. In case of a sequenced
|
|
* request, we are guaranteed that the completion callback will run on
|
|
* the receiver thread directly when the ACK for the packet has been
|
|
* received. Similarly, this function is guaranteed to run on the
|
|
* receiver thread. Thus we are guaranteed that if the packet has been
|
|
* successfully transmitted and received an ACK, the transmitted flag
|
|
* has been set and is visible here.
|
|
*
|
|
* We are currently not handling unsequenced packets here, as those
|
|
* should never expect a response as ensured in ssh_rtl_submit. If this
|
|
* ever changes, one would have to test for
|
|
*
|
|
* (r->state & (transmitting | transmitted))
|
|
*
|
|
* on unsequenced packets to determine if they could have been
|
|
* transmitted. There are no synchronization guarantees as in the
|
|
* sequenced case, since, in this case, the callback function will not
|
|
* run on the same thread. Thus an exact determination is impossible.
|
|
*/
|
|
if (!test_bit(SSH_REQUEST_SF_TRANSMITTED_BIT, &r->state)) {
|
|
rtl_err(rtl, "rtl: received response before ACK for request (rqid = %#06x)\n",
|
|
rqid);
|
|
|
|
/*
|
|
* NB: Timeout has already been canceled, request already been
|
|
* removed from pending and marked as locked and completed. As
|
|
* we receive a "false" response, the packet might still be
|
|
* queued though.
|
|
*/
|
|
ssh_rtl_queue_remove(r);
|
|
|
|
ssh_rtl_complete_with_status(r, -EREMOTEIO);
|
|
ssh_request_put(r);
|
|
|
|
ssh_rtl_tx_schedule(rtl);
|
|
return;
|
|
}
|
|
|
|
/*
|
|
* NB: Timeout has already been canceled, request already been
|
|
* removed from pending and marked as locked and completed. The request
|
|
* can also not be queued any more, as it has been marked as
|
|
* transmitting and later transmitted. Thus no need to remove it from
|
|
* anywhere.
|
|
*/
|
|
|
|
ssh_rtl_complete_with_rsp(r, command, command_data);
|
|
ssh_request_put(r);
|
|
|
|
ssh_rtl_tx_schedule(rtl);
|
|
}
|
|
|
|
static bool ssh_rtl_cancel_nonpending(struct ssh_request *r)
|
|
{
|
|
struct ssh_rtl *rtl;
|
|
unsigned long flags, fixed;
|
|
bool remove;
|
|
|
|
/*
|
|
* Handle unsubmitted request: Try to mark the packet as locked,
|
|
* expecting the state to be zero (i.e. unsubmitted). Note that, if
|
|
* setting the state worked, we might still be adding the packet to the
|
|
* queue in a currently executing submit call. In that case, however,
|
|
* ptl reference must have been set previously, as locked is checked
|
|
* after setting ptl. Furthermore, when the ptl reference is set, the
|
|
* submission process is guaranteed to have entered the critical
|
|
* section. Thus only if we successfully locked this request and ptl is
|
|
* NULL, we have successfully removed the request, i.e. we are
|
|
* guaranteed that, due to the "locked" check in ssh_rtl_submit(), the
|
|
* packet will never be added. Otherwise, we need to try and grab it
|
|
* from the queue, where we are now guaranteed that the packet is or has
|
|
* been due to the critical section.
|
|
*
|
|
* Note that if the cmpxchg() fails, we are guaranteed that ptl has
|
|
* been set and is non-NULL, as states can only be nonzero after this
|
|
* has been set. Also note that we need to fetch the static (type)
|
|
* flags to ensure that they don't cause the cmpxchg() to fail.
|
|
*/
|
|
fixed = READ_ONCE(r->state) & SSH_REQUEST_FLAGS_TY_MASK;
|
|
flags = cmpxchg(&r->state, fixed, SSH_REQUEST_SF_LOCKED_BIT);
|
|
|
|
/*
|
|
* Force correct ordering with regards to state and ptl reference access
|
|
* to safe-guard cancellation to concurrent submission against a
|
|
* lost-update problem. First try to exchange state, then also check
|
|
* ptl if that worked. This barrier is paired with the
|
|
* one in ssh_rtl_submit().
|
|
*/
|
|
smp_mb__after_atomic();
|
|
|
|
if (flags == fixed && !READ_ONCE(r->packet.ptl)) {
|
|
if (test_and_set_bit(SSH_REQUEST_SF_COMPLETED_BIT, &r->state))
|
|
return true;
|
|
|
|
ssh_rtl_complete_with_status(r, -ECANCELED);
|
|
return true;
|
|
}
|
|
|
|
rtl = ssh_request_rtl(r);
|
|
spin_lock(&rtl->queue.lock);
|
|
|
|
/*
|
|
* Note: 1) Requests cannot be re-submitted. 2) If a request is
|
|
* queued, it cannot be "transmitting"/"pending" yet. Thus, if we
|
|
* successfully remove the request here, we have removed all its
|
|
* occurrences in the system.
|
|
*/
|
|
|
|
remove = test_and_clear_bit(SSH_REQUEST_SF_QUEUED_BIT, &r->state);
|
|
if (!remove) {
|
|
spin_unlock(&rtl->queue.lock);
|
|
return false;
|
|
}
|
|
|
|
set_bit(SSH_REQUEST_SF_LOCKED_BIT, &r->state);
|
|
list_del(&r->node);
|
|
|
|
spin_unlock(&rtl->queue.lock);
|
|
|
|
ssh_request_put(r); /* Drop reference obtained from queue. */
|
|
|
|
if (test_and_set_bit(SSH_REQUEST_SF_COMPLETED_BIT, &r->state))
|
|
return true;
|
|
|
|
ssh_rtl_complete_with_status(r, -ECANCELED);
|
|
return true;
|
|
}
|
|
|
|
static bool ssh_rtl_cancel_pending(struct ssh_request *r)
|
|
{
|
|
/* If the packet is already locked, it's going to be removed shortly. */
|
|
if (test_and_set_bit(SSH_REQUEST_SF_LOCKED_BIT, &r->state))
|
|
return true;
|
|
|
|
/*
|
|
* Now that we have locked the packet, we have guaranteed that it can't
|
|
* be added to the system any more. If ptl is NULL, the locked
|
|
* check in ssh_rtl_submit() has not been run and any submission,
|
|
* currently in progress or called later, won't add the packet. Thus we
|
|
* can directly complete it.
|
|
*
|
|
* The implicit memory barrier of test_and_set_bit() should be enough
|
|
* to ensure that the correct order (first lock, then check ptl) is
|
|
* ensured. This is paired with the barrier in ssh_rtl_submit().
|
|
*/
|
|
if (!READ_ONCE(r->packet.ptl)) {
|
|
if (test_and_set_bit(SSH_REQUEST_SF_COMPLETED_BIT, &r->state))
|
|
return true;
|
|
|
|
ssh_rtl_complete_with_status(r, -ECANCELED);
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
* Try to cancel the packet. If the packet has not been completed yet,
|
|
* this will subsequently (and synchronously) call the completion
|
|
* callback of the packet, which will complete the request.
|
|
*/
|
|
ssh_ptl_cancel(&r->packet);
|
|
|
|
/*
|
|
* If the packet has been completed with success, i.e. has not been
|
|
* canceled by the above call, the request may not have been completed
|
|
* yet (may be waiting for a response). Check if we need to do this
|
|
* here.
|
|
*/
|
|
if (test_and_set_bit(SSH_REQUEST_SF_COMPLETED_BIT, &r->state))
|
|
return true;
|
|
|
|
ssh_rtl_queue_remove(r);
|
|
ssh_rtl_pending_remove(r);
|
|
ssh_rtl_complete_with_status(r, -ECANCELED);
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* ssh_rtl_cancel() - Cancel request.
|
|
* @rqst: The request to cancel.
|
|
* @pending: Whether to also cancel pending requests.
|
|
*
|
|
* Cancels the given request. If @pending is %false, this will not cancel
|
|
* pending requests, i.e. requests that have already been submitted to the
|
|
* packet layer but not been completed yet. If @pending is %true, this will
|
|
* cancel the given request regardless of the state it is in.
|
|
*
|
|
* If the request has been canceled by calling this function, both completion
|
|
* and release callbacks of the request will be executed in a reasonable
|
|
* time-frame. This may happen during execution of this function, however,
|
|
* there is no guarantee for this. For example, a request currently
|
|
* transmitting will be canceled/completed only after transmission has
|
|
* completed, and the respective callbacks will be executed on the transmitter
|
|
* thread, which may happen during, but also some time after execution of the
|
|
* cancel function.
|
|
*
|
|
* Return: Returns %true if the given request has been canceled or completed,
|
|
* either by this function or prior to calling this function, %false
|
|
* otherwise. If @pending is %true, this function will always return %true.
|
|
*/
|
|
bool ssh_rtl_cancel(struct ssh_request *rqst, bool pending)
|
|
{
|
|
struct ssh_rtl *rtl;
|
|
bool canceled;
|
|
|
|
if (test_and_set_bit(SSH_REQUEST_SF_CANCELED_BIT, &rqst->state))
|
|
return true;
|
|
|
|
trace_ssam_request_cancel(rqst);
|
|
|
|
if (pending)
|
|
canceled = ssh_rtl_cancel_pending(rqst);
|
|
else
|
|
canceled = ssh_rtl_cancel_nonpending(rqst);
|
|
|
|
/* Note: rtl may be NULL if request has not been submitted yet. */
|
|
rtl = ssh_request_rtl(rqst);
|
|
if (canceled && rtl)
|
|
ssh_rtl_tx_schedule(rtl);
|
|
|
|
return canceled;
|
|
}
|
|
|
|
static void ssh_rtl_packet_callback(struct ssh_packet *p, int status)
|
|
{
|
|
struct ssh_request *r = to_ssh_request(p);
|
|
|
|
if (unlikely(status)) {
|
|
set_bit(SSH_REQUEST_SF_LOCKED_BIT, &r->state);
|
|
|
|
if (test_and_set_bit(SSH_REQUEST_SF_COMPLETED_BIT, &r->state))
|
|
return;
|
|
|
|
/*
|
|
* The packet may get canceled even though it has not been
|
|
* submitted yet. The request may still be queued. Check the
|
|
* queue and remove it if necessary. As the timeout would have
|
|
* been started in this function on success, there's no need
|
|
* to cancel it here.
|
|
*/
|
|
ssh_rtl_queue_remove(r);
|
|
ssh_rtl_pending_remove(r);
|
|
ssh_rtl_complete_with_status(r, status);
|
|
|
|
ssh_rtl_tx_schedule(ssh_request_rtl(r));
|
|
return;
|
|
}
|
|
|
|
/* Update state: Mark as transmitted and clear transmitting. */
|
|
set_bit(SSH_REQUEST_SF_TRANSMITTED_BIT, &r->state);
|
|
/* Ensure state never gets zero. */
|
|
smp_mb__before_atomic();
|
|
clear_bit(SSH_REQUEST_SF_TRANSMITTING_BIT, &r->state);
|
|
|
|
/* If we expect a response, we just need to start the timeout. */
|
|
if (test_bit(SSH_REQUEST_TY_HAS_RESPONSE_BIT, &r->state)) {
|
|
/*
|
|
* Note: This is the only place where the timestamp gets set,
|
|
* all other access to it is read-only.
|
|
*/
|
|
ssh_rtl_timeout_start(r);
|
|
return;
|
|
}
|
|
|
|
/*
|
|
* If we don't expect a response, lock, remove, and complete the
|
|
* request. Note that, at this point, the request is guaranteed to have
|
|
* left the queue and no timeout has been started. Thus we only need to
|
|
* remove it from pending. If the request has already been completed (it
|
|
* may have been canceled) return.
|
|
*/
|
|
|
|
set_bit(SSH_REQUEST_SF_LOCKED_BIT, &r->state);
|
|
if (test_and_set_bit(SSH_REQUEST_SF_COMPLETED_BIT, &r->state))
|
|
return;
|
|
|
|
ssh_rtl_pending_remove(r);
|
|
ssh_rtl_complete_with_status(r, 0);
|
|
|
|
ssh_rtl_tx_schedule(ssh_request_rtl(r));
|
|
}
|
|
|
|
static ktime_t ssh_request_get_expiration(struct ssh_request *r, ktime_t timeout)
|
|
{
|
|
ktime_t timestamp = READ_ONCE(r->timestamp);
|
|
|
|
if (timestamp != KTIME_MAX)
|
|
return ktime_add(timestamp, timeout);
|
|
else
|
|
return KTIME_MAX;
|
|
}
|
|
|
|
static void ssh_rtl_timeout_reap(struct work_struct *work)
|
|
{
|
|
struct ssh_rtl *rtl = to_ssh_rtl(work, rtx_timeout.reaper.work);
|
|
struct ssh_request *r, *n;
|
|
LIST_HEAD(claimed);
|
|
ktime_t now = ktime_get_coarse_boottime();
|
|
ktime_t timeout = rtl->rtx_timeout.timeout;
|
|
ktime_t next = KTIME_MAX;
|
|
|
|
trace_ssam_rtl_timeout_reap(atomic_read(&rtl->pending.count));
|
|
|
|
/*
|
|
* Mark reaper as "not pending". This is done before checking any
|
|
* requests to avoid lost-update type problems.
|
|
*/
|
|
spin_lock(&rtl->rtx_timeout.lock);
|
|
rtl->rtx_timeout.expires = KTIME_MAX;
|
|
spin_unlock(&rtl->rtx_timeout.lock);
|
|
|
|
spin_lock(&rtl->pending.lock);
|
|
list_for_each_entry_safe(r, n, &rtl->pending.head, node) {
|
|
ktime_t expires = ssh_request_get_expiration(r, timeout);
|
|
|
|
/*
|
|
* Check if the timeout hasn't expired yet. Find out next
|
|
* expiration date to be handled after this run.
|
|
*/
|
|
if (ktime_after(expires, now)) {
|
|
next = ktime_before(expires, next) ? expires : next;
|
|
continue;
|
|
}
|
|
|
|
/* Avoid further transitions if locked. */
|
|
if (test_and_set_bit(SSH_REQUEST_SF_LOCKED_BIT, &r->state))
|
|
continue;
|
|
|
|
/*
|
|
* We have now marked the packet as locked. Thus it cannot be
|
|
* added to the pending or queued lists again after we've
|
|
* removed it here. We can therefore re-use the node of this
|
|
* packet temporarily.
|
|
*/
|
|
|
|
clear_bit(SSH_REQUEST_SF_PENDING_BIT, &r->state);
|
|
|
|
atomic_dec(&rtl->pending.count);
|
|
list_del(&r->node);
|
|
|
|
list_add_tail(&r->node, &claimed);
|
|
}
|
|
spin_unlock(&rtl->pending.lock);
|
|
|
|
/* Cancel and complete the request. */
|
|
list_for_each_entry_safe(r, n, &claimed, node) {
|
|
trace_ssam_request_timeout(r);
|
|
|
|
/*
|
|
* At this point we've removed the packet from pending. This
|
|
* means that we've obtained the last (only) reference of the
|
|
* system to it. Thus we can just complete it.
|
|
*/
|
|
if (!test_and_set_bit(SSH_REQUEST_SF_COMPLETED_BIT, &r->state))
|
|
ssh_rtl_complete_with_status(r, -ETIMEDOUT);
|
|
|
|
/*
|
|
* Drop the reference we've obtained by removing it from the
|
|
* pending set.
|
|
*/
|
|
list_del(&r->node);
|
|
ssh_request_put(r);
|
|
}
|
|
|
|
/* Ensure that the reaper doesn't run again immediately. */
|
|
next = max(next, ktime_add(now, SSH_RTL_REQUEST_TIMEOUT_RESOLUTION));
|
|
if (next != KTIME_MAX)
|
|
ssh_rtl_timeout_reaper_mod(rtl, now, next);
|
|
|
|
ssh_rtl_tx_schedule(rtl);
|
|
}
|
|
|
|
static void ssh_rtl_rx_event(struct ssh_rtl *rtl, const struct ssh_command *cmd,
|
|
const struct ssam_span *data)
|
|
{
|
|
trace_ssam_rx_event_received(cmd, data->len);
|
|
|
|
rtl_dbg(rtl, "rtl: handling event (rqid: %#06x)\n",
|
|
get_unaligned_le16(&cmd->rqid));
|
|
|
|
rtl->ops.handle_event(rtl, cmd, data);
|
|
}
|
|
|
|
static void ssh_rtl_rx_command(struct ssh_ptl *p, const struct ssam_span *data)
|
|
{
|
|
struct ssh_rtl *rtl = to_ssh_rtl(p, ptl);
|
|
struct device *dev = &p->serdev->dev;
|
|
struct ssh_command *command;
|
|
struct ssam_span command_data;
|
|
|
|
if (sshp_parse_command(dev, data, &command, &command_data))
|
|
return;
|
|
|
|
if (ssh_rqid_is_event(get_unaligned_le16(&command->rqid)))
|
|
ssh_rtl_rx_event(rtl, command, &command_data);
|
|
else
|
|
ssh_rtl_complete(rtl, command, &command_data);
|
|
}
|
|
|
|
static void ssh_rtl_rx_data(struct ssh_ptl *p, const struct ssam_span *data)
|
|
{
|
|
if (!data->len) {
|
|
ptl_err(p, "rtl: rx: no data frame payload\n");
|
|
return;
|
|
}
|
|
|
|
switch (data->ptr[0]) {
|
|
case SSH_PLD_TYPE_CMD:
|
|
ssh_rtl_rx_command(p, data);
|
|
break;
|
|
|
|
default:
|
|
ptl_err(p, "rtl: rx: unknown frame payload type (type: %#04x)\n",
|
|
data->ptr[0]);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void ssh_rtl_packet_release(struct ssh_packet *p)
|
|
{
|
|
struct ssh_request *rqst;
|
|
|
|
rqst = to_ssh_request(p);
|
|
rqst->ops->release(rqst);
|
|
}
|
|
|
|
static const struct ssh_packet_ops ssh_rtl_packet_ops = {
|
|
.complete = ssh_rtl_packet_callback,
|
|
.release = ssh_rtl_packet_release,
|
|
};
|
|
|
|
/**
|
|
* ssh_request_init() - Initialize SSH request.
|
|
* @rqst: The request to initialize.
|
|
* @flags: Request flags, determining the type of the request.
|
|
* @ops: Request operations.
|
|
*
|
|
* Initializes the given SSH request and underlying packet. Sets the message
|
|
* buffer pointer to %NULL and the message buffer length to zero. This buffer
|
|
* has to be set separately via ssh_request_set_data() before submission and
|
|
* must contain a valid SSH request message.
|
|
*
|
|
* Return: Returns zero on success or %-EINVAL if the given flags are invalid.
|
|
*/
|
|
int ssh_request_init(struct ssh_request *rqst, enum ssam_request_flags flags,
|
|
const struct ssh_request_ops *ops)
|
|
{
|
|
unsigned long type = BIT(SSH_PACKET_TY_BLOCKING_BIT);
|
|
|
|
/* Unsequenced requests cannot have a response. */
|
|
if (flags & SSAM_REQUEST_UNSEQUENCED && flags & SSAM_REQUEST_HAS_RESPONSE)
|
|
return -EINVAL;
|
|
|
|
if (!(flags & SSAM_REQUEST_UNSEQUENCED))
|
|
type |= BIT(SSH_PACKET_TY_SEQUENCED_BIT);
|
|
|
|
ssh_packet_init(&rqst->packet, type, SSH_PACKET_PRIORITY(DATA, 0),
|
|
&ssh_rtl_packet_ops);
|
|
|
|
INIT_LIST_HEAD(&rqst->node);
|
|
|
|
rqst->state = 0;
|
|
if (flags & SSAM_REQUEST_HAS_RESPONSE)
|
|
rqst->state |= BIT(SSH_REQUEST_TY_HAS_RESPONSE_BIT);
|
|
|
|
rqst->timestamp = KTIME_MAX;
|
|
rqst->ops = ops;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* ssh_rtl_init() - Initialize request transport layer.
|
|
* @rtl: The request transport layer to initialize.
|
|
* @serdev: The underlying serial device, i.e. the lower-level transport.
|
|
* @ops: Request transport layer operations.
|
|
*
|
|
* Initializes the given request transport layer and associated packet
|
|
* transport layer. Transmitter and receiver threads must be started
|
|
* separately via ssh_rtl_start(), after the request-layer has been
|
|
* initialized and the lower-level serial device layer has been set up.
|
|
*
|
|
* Return: Returns zero on success and a nonzero error code on failure.
|
|
*/
|
|
int ssh_rtl_init(struct ssh_rtl *rtl, struct serdev_device *serdev,
|
|
const struct ssh_rtl_ops *ops)
|
|
{
|
|
struct ssh_ptl_ops ptl_ops;
|
|
int status;
|
|
|
|
ptl_ops.data_received = ssh_rtl_rx_data;
|
|
|
|
status = ssh_ptl_init(&rtl->ptl, serdev, &ptl_ops);
|
|
if (status)
|
|
return status;
|
|
|
|
spin_lock_init(&rtl->queue.lock);
|
|
INIT_LIST_HEAD(&rtl->queue.head);
|
|
|
|
spin_lock_init(&rtl->pending.lock);
|
|
INIT_LIST_HEAD(&rtl->pending.head);
|
|
atomic_set_release(&rtl->pending.count, 0);
|
|
|
|
INIT_WORK(&rtl->tx.work, ssh_rtl_tx_work_fn);
|
|
|
|
spin_lock_init(&rtl->rtx_timeout.lock);
|
|
rtl->rtx_timeout.timeout = SSH_RTL_REQUEST_TIMEOUT;
|
|
rtl->rtx_timeout.expires = KTIME_MAX;
|
|
INIT_DELAYED_WORK(&rtl->rtx_timeout.reaper, ssh_rtl_timeout_reap);
|
|
|
|
rtl->ops = *ops;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* ssh_rtl_destroy() - Deinitialize request transport layer.
|
|
* @rtl: The request transport layer to deinitialize.
|
|
*
|
|
* Deinitializes the given request transport layer and frees resources
|
|
* associated with it. If receiver and/or transmitter threads have been
|
|
* started, the layer must first be shut down via ssh_rtl_shutdown() before
|
|
* this function can be called.
|
|
*/
|
|
void ssh_rtl_destroy(struct ssh_rtl *rtl)
|
|
{
|
|
ssh_ptl_destroy(&rtl->ptl);
|
|
}
|
|
|
|
/**
|
|
* ssh_rtl_start() - Start request transmitter and receiver.
|
|
* @rtl: The request transport layer.
|
|
*
|
|
* Return: Returns zero on success, a negative error code on failure.
|
|
*/
|
|
int ssh_rtl_start(struct ssh_rtl *rtl)
|
|
{
|
|
int status;
|
|
|
|
status = ssh_ptl_tx_start(&rtl->ptl);
|
|
if (status)
|
|
return status;
|
|
|
|
ssh_rtl_tx_schedule(rtl);
|
|
|
|
status = ssh_ptl_rx_start(&rtl->ptl);
|
|
if (status) {
|
|
ssh_rtl_flush(rtl, msecs_to_jiffies(5000));
|
|
ssh_ptl_tx_stop(&rtl->ptl);
|
|
return status;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
struct ssh_flush_request {
|
|
struct ssh_request base;
|
|
struct completion completion;
|
|
int status;
|
|
};
|
|
|
|
static void ssh_rtl_flush_request_complete(struct ssh_request *r,
|
|
const struct ssh_command *cmd,
|
|
const struct ssam_span *data,
|
|
int status)
|
|
{
|
|
struct ssh_flush_request *rqst;
|
|
|
|
rqst = container_of(r, struct ssh_flush_request, base);
|
|
rqst->status = status;
|
|
}
|
|
|
|
static void ssh_rtl_flush_request_release(struct ssh_request *r)
|
|
{
|
|
struct ssh_flush_request *rqst;
|
|
|
|
rqst = container_of(r, struct ssh_flush_request, base);
|
|
complete_all(&rqst->completion);
|
|
}
|
|
|
|
static const struct ssh_request_ops ssh_rtl_flush_request_ops = {
|
|
.complete = ssh_rtl_flush_request_complete,
|
|
.release = ssh_rtl_flush_request_release,
|
|
};
|
|
|
|
/**
|
|
* ssh_rtl_flush() - Flush the request transport layer.
|
|
* @rtl: request transport layer
|
|
* @timeout: timeout for the flush operation in jiffies
|
|
*
|
|
* Queue a special flush request and wait for its completion. This request
|
|
* will be completed after all other currently queued and pending requests
|
|
* have been completed. Instead of a normal data packet, this request submits
|
|
* a special flush packet, meaning that upon completion, also the underlying
|
|
* packet transport layer has been flushed.
|
|
*
|
|
* Flushing the request layer guarantees that all previously submitted
|
|
* requests have been fully completed before this call returns. Additionally,
|
|
* flushing blocks execution of all later submitted requests until the flush
|
|
* has been completed.
|
|
*
|
|
* If the caller ensures that no new requests are submitted after a call to
|
|
* this function, the request transport layer is guaranteed to have no
|
|
* remaining requests when this call returns. The same guarantee does not hold
|
|
* for the packet layer, on which control packets may still be queued after
|
|
* this call.
|
|
*
|
|
* Return: Returns zero on success, %-ETIMEDOUT if the flush timed out and has
|
|
* been canceled as a result of the timeout, or %-ESHUTDOWN if the packet
|
|
* and/or request transport layer has been shut down before this call. May
|
|
* also return %-EINTR if the underlying packet transmission has been
|
|
* interrupted.
|
|
*/
|
|
int ssh_rtl_flush(struct ssh_rtl *rtl, unsigned long timeout)
|
|
{
|
|
const unsigned int init_flags = SSAM_REQUEST_UNSEQUENCED;
|
|
struct ssh_flush_request rqst;
|
|
int status;
|
|
|
|
ssh_request_init(&rqst.base, init_flags, &ssh_rtl_flush_request_ops);
|
|
rqst.base.packet.state |= BIT(SSH_PACKET_TY_FLUSH_BIT);
|
|
rqst.base.packet.priority = SSH_PACKET_PRIORITY(FLUSH, 0);
|
|
rqst.base.state |= BIT(SSH_REQUEST_TY_FLUSH_BIT);
|
|
|
|
init_completion(&rqst.completion);
|
|
|
|
status = ssh_rtl_submit(rtl, &rqst.base);
|
|
if (status)
|
|
return status;
|
|
|
|
ssh_request_put(&rqst.base);
|
|
|
|
if (!wait_for_completion_timeout(&rqst.completion, timeout)) {
|
|
ssh_rtl_cancel(&rqst.base, true);
|
|
wait_for_completion(&rqst.completion);
|
|
}
|
|
|
|
WARN_ON(rqst.status != 0 && rqst.status != -ECANCELED &&
|
|
rqst.status != -ESHUTDOWN && rqst.status != -EINTR);
|
|
|
|
return rqst.status == -ECANCELED ? -ETIMEDOUT : rqst.status;
|
|
}
|
|
|
|
/**
|
|
* ssh_rtl_shutdown() - Shut down request transport layer.
|
|
* @rtl: The request transport layer.
|
|
*
|
|
* Shuts down the request transport layer, removing and canceling all queued
|
|
* and pending requests. Requests canceled by this operation will be completed
|
|
* with %-ESHUTDOWN as status. Receiver and transmitter threads will be
|
|
* stopped, the lower-level packet layer will be shutdown.
|
|
*
|
|
* As a result of this function, the transport layer will be marked as shut
|
|
* down. Submission of requests after the transport layer has been shut down
|
|
* will fail with %-ESHUTDOWN.
|
|
*/
|
|
void ssh_rtl_shutdown(struct ssh_rtl *rtl)
|
|
{
|
|
struct ssh_request *r, *n;
|
|
LIST_HEAD(claimed);
|
|
int pending;
|
|
|
|
set_bit(SSH_RTL_SF_SHUTDOWN_BIT, &rtl->state);
|
|
/*
|
|
* Ensure that the layer gets marked as shut-down before actually
|
|
* stopping it. In combination with the check in ssh_rtl_submit(),
|
|
* this guarantees that no new requests can be added and all already
|
|
* queued requests are properly canceled.
|
|
*/
|
|
smp_mb__after_atomic();
|
|
|
|
/* Remove requests from queue. */
|
|
spin_lock(&rtl->queue.lock);
|
|
list_for_each_entry_safe(r, n, &rtl->queue.head, node) {
|
|
set_bit(SSH_REQUEST_SF_LOCKED_BIT, &r->state);
|
|
/* Ensure state never gets zero. */
|
|
smp_mb__before_atomic();
|
|
clear_bit(SSH_REQUEST_SF_QUEUED_BIT, &r->state);
|
|
|
|
list_del(&r->node);
|
|
list_add_tail(&r->node, &claimed);
|
|
}
|
|
spin_unlock(&rtl->queue.lock);
|
|
|
|
/*
|
|
* We have now guaranteed that the queue is empty and no more new
|
|
* requests can be submitted (i.e. it will stay empty). This means that
|
|
* calling ssh_rtl_tx_schedule() will not schedule tx.work any more. So
|
|
* we can simply call cancel_work_sync() on tx.work here and when that
|
|
* returns, we've locked it down. This also means that after this call,
|
|
* we don't submit any more packets to the underlying packet layer, so
|
|
* we can also shut that down.
|
|
*/
|
|
|
|
cancel_work_sync(&rtl->tx.work);
|
|
ssh_ptl_shutdown(&rtl->ptl);
|
|
cancel_delayed_work_sync(&rtl->rtx_timeout.reaper);
|
|
|
|
/*
|
|
* Shutting down the packet layer should also have canceled all
|
|
* requests. Thus the pending set should be empty. Attempt to handle
|
|
* this gracefully anyways, even though this should be dead code.
|
|
*/
|
|
|
|
pending = atomic_read(&rtl->pending.count);
|
|
if (WARN_ON(pending)) {
|
|
spin_lock(&rtl->pending.lock);
|
|
list_for_each_entry_safe(r, n, &rtl->pending.head, node) {
|
|
set_bit(SSH_REQUEST_SF_LOCKED_BIT, &r->state);
|
|
/* Ensure state never gets zero. */
|
|
smp_mb__before_atomic();
|
|
clear_bit(SSH_REQUEST_SF_PENDING_BIT, &r->state);
|
|
|
|
list_del(&r->node);
|
|
list_add_tail(&r->node, &claimed);
|
|
}
|
|
spin_unlock(&rtl->pending.lock);
|
|
}
|
|
|
|
/* Finally, cancel and complete the requests we claimed before. */
|
|
list_for_each_entry_safe(r, n, &claimed, node) {
|
|
/*
|
|
* We need test_and_set() because we still might compete with
|
|
* cancellation.
|
|
*/
|
|
if (!test_and_set_bit(SSH_REQUEST_SF_COMPLETED_BIT, &r->state))
|
|
ssh_rtl_complete_with_status(r, -ESHUTDOWN);
|
|
|
|
/*
|
|
* Drop the reference we've obtained by removing it from the
|
|
* lists.
|
|
*/
|
|
list_del(&r->node);
|
|
ssh_request_put(r);
|
|
}
|
|
}
|