144 lines
4.4 KiB
C
144 lines
4.4 KiB
C
/* SPDX-License-Identifier: GPL-2.0+ */
|
|
/*
|
|
* SSH request transport layer.
|
|
*
|
|
* Copyright (C) 2019-2021 Maximilian Luz <luzmaximilian@gmail.com>
|
|
*/
|
|
|
|
#ifndef _SURFACE_AGGREGATOR_SSH_REQUEST_LAYER_H
|
|
#define _SURFACE_AGGREGATOR_SSH_REQUEST_LAYER_H
|
|
|
|
#include <linux/atomic.h>
|
|
#include <linux/ktime.h>
|
|
#include <linux/list.h>
|
|
#include <linux/spinlock.h>
|
|
#include <linux/workqueue.h>
|
|
|
|
#include <linux/surface_aggregator/serial_hub.h>
|
|
#include <linux/surface_aggregator/controller.h>
|
|
|
|
#include "ssh_packet_layer.h"
|
|
|
|
/**
|
|
* enum ssh_rtl_state_flags - State-flags for &struct ssh_rtl.
|
|
*
|
|
* @SSH_RTL_SF_SHUTDOWN_BIT:
|
|
* Indicates that the request transport layer has been shut down or is
|
|
* being shut down and should not accept any new requests.
|
|
*/
|
|
enum ssh_rtl_state_flags {
|
|
SSH_RTL_SF_SHUTDOWN_BIT,
|
|
};
|
|
|
|
/**
|
|
* struct ssh_rtl_ops - Callback operations for request transport layer.
|
|
* @handle_event: Function called when a SSH event has been received. The
|
|
* specified function takes the request layer, received command
|
|
* struct, and corresponding payload as arguments. If the event
|
|
* has no payload, the payload span is empty (not %NULL).
|
|
*/
|
|
struct ssh_rtl_ops {
|
|
void (*handle_event)(struct ssh_rtl *rtl, const struct ssh_command *cmd,
|
|
const struct ssam_span *data);
|
|
};
|
|
|
|
/**
|
|
* struct ssh_rtl - SSH request transport layer.
|
|
* @ptl: Underlying packet transport layer.
|
|
* @state: State(-flags) of the transport layer.
|
|
* @queue: Request submission queue.
|
|
* @queue.lock: Lock for modifying the request submission queue.
|
|
* @queue.head: List-head of the request submission queue.
|
|
* @pending: Set/list of pending requests.
|
|
* @pending.lock: Lock for modifying the request set.
|
|
* @pending.head: List-head of the pending set/list.
|
|
* @pending.count: Number of currently pending requests.
|
|
* @tx: Transmitter subsystem.
|
|
* @tx.work: Transmitter work item.
|
|
* @rtx_timeout: Retransmission timeout subsystem.
|
|
* @rtx_timeout.lock: Lock for modifying the retransmission timeout reaper.
|
|
* @rtx_timeout.timeout: Timeout interval for retransmission.
|
|
* @rtx_timeout.expires: Time specifying when the reaper work is next scheduled.
|
|
* @rtx_timeout.reaper: Work performing timeout checks and subsequent actions.
|
|
* @ops: Request layer operations.
|
|
*/
|
|
struct ssh_rtl {
|
|
struct ssh_ptl ptl;
|
|
unsigned long state;
|
|
|
|
struct {
|
|
spinlock_t lock;
|
|
struct list_head head;
|
|
} queue;
|
|
|
|
struct {
|
|
spinlock_t lock;
|
|
struct list_head head;
|
|
atomic_t count;
|
|
} pending;
|
|
|
|
struct {
|
|
struct work_struct work;
|
|
} tx;
|
|
|
|
struct {
|
|
spinlock_t lock;
|
|
ktime_t timeout;
|
|
ktime_t expires;
|
|
struct delayed_work reaper;
|
|
} rtx_timeout;
|
|
|
|
struct ssh_rtl_ops ops;
|
|
};
|
|
|
|
#define rtl_dbg(r, fmt, ...) ptl_dbg(&(r)->ptl, fmt, ##__VA_ARGS__)
|
|
#define rtl_info(p, fmt, ...) ptl_info(&(p)->ptl, fmt, ##__VA_ARGS__)
|
|
#define rtl_warn(r, fmt, ...) ptl_warn(&(r)->ptl, fmt, ##__VA_ARGS__)
|
|
#define rtl_err(r, fmt, ...) ptl_err(&(r)->ptl, fmt, ##__VA_ARGS__)
|
|
#define rtl_dbg_cond(r, fmt, ...) __ssam_prcond(rtl_dbg, r, fmt, ##__VA_ARGS__)
|
|
|
|
#define to_ssh_rtl(ptr, member) \
|
|
container_of(ptr, struct ssh_rtl, member)
|
|
|
|
/**
|
|
* ssh_rtl_get_device() - Get device associated with request transport layer.
|
|
* @rtl: The request transport layer.
|
|
*
|
|
* Return: Returns the device on which the given request transport layer
|
|
* builds upon.
|
|
*/
|
|
static inline struct device *ssh_rtl_get_device(struct ssh_rtl *rtl)
|
|
{
|
|
return ssh_ptl_get_device(&rtl->ptl);
|
|
}
|
|
|
|
/**
|
|
* ssh_request_rtl() - Get request transport layer associated with request.
|
|
* @rqst: The request to get the request transport layer reference for.
|
|
*
|
|
* Return: Returns the &struct ssh_rtl associated with the given SSH request.
|
|
*/
|
|
static inline struct ssh_rtl *ssh_request_rtl(struct ssh_request *rqst)
|
|
{
|
|
struct ssh_ptl *ptl;
|
|
|
|
ptl = READ_ONCE(rqst->packet.ptl);
|
|
return likely(ptl) ? to_ssh_rtl(ptl, ptl) : NULL;
|
|
}
|
|
|
|
int ssh_rtl_submit(struct ssh_rtl *rtl, struct ssh_request *rqst);
|
|
bool ssh_rtl_cancel(struct ssh_request *rqst, bool pending);
|
|
|
|
int ssh_rtl_init(struct ssh_rtl *rtl, struct serdev_device *serdev,
|
|
const struct ssh_rtl_ops *ops);
|
|
|
|
int ssh_rtl_start(struct ssh_rtl *rtl);
|
|
int ssh_rtl_flush(struct ssh_rtl *rtl, unsigned long timeout);
|
|
void ssh_rtl_shutdown(struct ssh_rtl *rtl);
|
|
void ssh_rtl_destroy(struct ssh_rtl *rtl);
|
|
|
|
int ssh_request_init(struct ssh_request *rqst, enum ssam_request_flags flags,
|
|
const struct ssh_request_ops *ops);
|
|
|
|
#endif /* _SURFACE_AGGREGATOR_SSH_REQUEST_LAYER_H */
|