2018-10-01 17:31:21 +08:00
|
|
|
/* SPDX-License-Identifier: GPL-2.0 */
|
2017-10-02 18:38:30 +08:00
|
|
|
/*
|
|
|
|
* Thunderbolt service API
|
|
|
|
*
|
2017-10-02 18:38:31 +08:00
|
|
|
* Copyright (C) 2014 Andreas Noever <andreas.noever@gmail.com>
|
2017-10-02 18:38:30 +08:00
|
|
|
* Copyright (C) 2017, Intel Corporation
|
|
|
|
* Authors: Michael Jamet <michael.jamet@intel.com>
|
|
|
|
* Mika Westerberg <mika.westerberg@linux.intel.com>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef THUNDERBOLT_H_
|
|
|
|
#define THUNDERBOLT_H_
|
|
|
|
|
2017-10-02 18:38:32 +08:00
|
|
|
#include <linux/device.h>
|
2017-10-02 18:38:37 +08:00
|
|
|
#include <linux/idr.h>
|
2017-10-02 18:38:30 +08:00
|
|
|
#include <linux/list.h>
|
2017-10-02 18:38:32 +08:00
|
|
|
#include <linux/mutex.h>
|
2017-10-02 18:38:34 +08:00
|
|
|
#include <linux/mod_devicetable.h>
|
2017-10-02 18:38:42 +08:00
|
|
|
#include <linux/pci.h>
|
2017-10-02 18:38:30 +08:00
|
|
|
#include <linux/uuid.h>
|
2017-10-02 18:38:37 +08:00
|
|
|
#include <linux/workqueue.h>
|
2017-10-02 18:38:30 +08:00
|
|
|
|
2017-10-02 18:38:31 +08:00
|
|
|
enum tb_cfg_pkg_type {
|
|
|
|
TB_CFG_PKG_READ = 1,
|
|
|
|
TB_CFG_PKG_WRITE = 2,
|
|
|
|
TB_CFG_PKG_ERROR = 3,
|
|
|
|
TB_CFG_PKG_NOTIFY_ACK = 4,
|
|
|
|
TB_CFG_PKG_EVENT = 5,
|
|
|
|
TB_CFG_PKG_XDOMAIN_REQ = 6,
|
|
|
|
TB_CFG_PKG_XDOMAIN_RESP = 7,
|
|
|
|
TB_CFG_PKG_OVERRIDE = 8,
|
|
|
|
TB_CFG_PKG_RESET = 9,
|
|
|
|
TB_CFG_PKG_ICM_EVENT = 10,
|
|
|
|
TB_CFG_PKG_ICM_CMD = 11,
|
|
|
|
TB_CFG_PKG_ICM_RESP = 12,
|
|
|
|
TB_CFG_PKG_PREPARE_TO_SLEEP = 13,
|
|
|
|
};
|
|
|
|
|
2017-10-02 18:38:32 +08:00
|
|
|
/**
|
|
|
|
* enum tb_security_level - Thunderbolt security level
|
|
|
|
* @TB_SECURITY_NONE: No security, legacy mode
|
|
|
|
* @TB_SECURITY_USER: User approval required at minimum
|
|
|
|
* @TB_SECURITY_SECURE: One time saved key required at minimum
|
|
|
|
* @TB_SECURITY_DPONLY: Only tunnel Display port (and USB)
|
2017-12-08 19:11:39 +08:00
|
|
|
* @TB_SECURITY_USBONLY: Only tunnel USB controller of the connected
|
|
|
|
* Thunderbolt dock (and Display Port). All PCIe
|
|
|
|
* links downstream of the dock are removed.
|
2020-09-03 18:13:21 +08:00
|
|
|
* @TB_SECURITY_NOPCIE: For USB4 systems this level is used when the
|
|
|
|
* PCIe tunneling is disabled from the BIOS.
|
2017-10-02 18:38:32 +08:00
|
|
|
*/
|
|
|
|
enum tb_security_level {
|
|
|
|
TB_SECURITY_NONE,
|
|
|
|
TB_SECURITY_USER,
|
|
|
|
TB_SECURITY_SECURE,
|
|
|
|
TB_SECURITY_DPONLY,
|
2017-12-08 19:11:39 +08:00
|
|
|
TB_SECURITY_USBONLY,
|
2020-09-03 18:13:21 +08:00
|
|
|
TB_SECURITY_NOPCIE,
|
2017-10-02 18:38:32 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* struct tb - main thunderbolt bus structure
|
|
|
|
* @dev: Domain device
|
|
|
|
* @lock: Big lock. Must be held when accessing any struct
|
|
|
|
* tb_switch / struct tb_port.
|
|
|
|
* @nhi: Pointer to the NHI structure
|
|
|
|
* @ctl: Control channel for this domain
|
|
|
|
* @wq: Ordered workqueue for all domain specific work
|
|
|
|
* @root_switch: Root switch of this domain
|
|
|
|
* @cm_ops: Connection manager specific operations vector
|
|
|
|
* @index: Linux assigned domain number
|
|
|
|
* @security_level: Current security level
|
2018-01-21 18:08:04 +08:00
|
|
|
* @nboot_acl: Number of boot ACLs the domain supports
|
2017-10-02 18:38:32 +08:00
|
|
|
* @privdata: Private connection manager specific data
|
|
|
|
*/
|
|
|
|
struct tb {
|
|
|
|
struct device dev;
|
|
|
|
struct mutex lock;
|
|
|
|
struct tb_nhi *nhi;
|
|
|
|
struct tb_ctl *ctl;
|
|
|
|
struct workqueue_struct *wq;
|
|
|
|
struct tb_switch *root_switch;
|
|
|
|
const struct tb_cm_ops *cm_ops;
|
|
|
|
int index;
|
|
|
|
enum tb_security_level security_level;
|
2018-01-21 18:08:04 +08:00
|
|
|
size_t nboot_acl;
|
2020-05-08 03:25:24 +08:00
|
|
|
unsigned long privdata[];
|
2017-10-02 18:38:32 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
extern struct bus_type tb_bus_type;
|
2017-10-02 18:38:34 +08:00
|
|
|
extern struct device_type tb_service_type;
|
|
|
|
extern struct device_type tb_xdomain_type;
|
2017-10-02 18:38:32 +08:00
|
|
|
|
2017-10-02 18:38:33 +08:00
|
|
|
#define TB_LINKS_PER_PHY_PORT 2
|
|
|
|
|
|
|
|
static inline unsigned int tb_phy_port_from_link(unsigned int link)
|
|
|
|
{
|
|
|
|
return (link - 1) / TB_LINKS_PER_PHY_PORT;
|
|
|
|
}
|
|
|
|
|
2017-10-02 18:38:30 +08:00
|
|
|
/**
|
|
|
|
* struct tb_property_dir - XDomain property directory
|
|
|
|
* @uuid: Directory UUID or %NULL if root directory
|
|
|
|
* @properties: List of properties in this directory
|
|
|
|
*
|
|
|
|
* User needs to provide serialization if needed.
|
|
|
|
*/
|
|
|
|
struct tb_property_dir {
|
|
|
|
const uuid_t *uuid;
|
|
|
|
struct list_head properties;
|
|
|
|
};
|
|
|
|
|
|
|
|
enum tb_property_type {
|
|
|
|
TB_PROPERTY_TYPE_UNKNOWN = 0x00,
|
|
|
|
TB_PROPERTY_TYPE_DIRECTORY = 0x44,
|
|
|
|
TB_PROPERTY_TYPE_DATA = 0x64,
|
|
|
|
TB_PROPERTY_TYPE_TEXT = 0x74,
|
|
|
|
TB_PROPERTY_TYPE_VALUE = 0x76,
|
|
|
|
};
|
|
|
|
|
|
|
|
#define TB_PROPERTY_KEY_SIZE 8
|
|
|
|
|
|
|
|
/**
|
|
|
|
* struct tb_property - XDomain property
|
|
|
|
* @list: Used to link properties together in a directory
|
|
|
|
* @key: Key for the property (always terminated).
|
|
|
|
* @type: Type of the property
|
|
|
|
* @length: Length of the property data in dwords
|
|
|
|
* @value: Property value
|
|
|
|
*
|
|
|
|
* Users use @type to determine which field in @value is filled.
|
|
|
|
*/
|
|
|
|
struct tb_property {
|
|
|
|
struct list_head list;
|
|
|
|
char key[TB_PROPERTY_KEY_SIZE + 1];
|
|
|
|
enum tb_property_type type;
|
|
|
|
size_t length;
|
|
|
|
union {
|
|
|
|
struct tb_property_dir *dir;
|
|
|
|
u8 *data;
|
|
|
|
char *text;
|
|
|
|
u32 immediate;
|
|
|
|
} value;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct tb_property_dir *tb_property_parse_dir(const u32 *block,
|
|
|
|
size_t block_len);
|
|
|
|
ssize_t tb_property_format_dir(const struct tb_property_dir *dir, u32 *block,
|
|
|
|
size_t block_len);
|
2021-01-08 20:38:24 +08:00
|
|
|
struct tb_property_dir *tb_property_copy_dir(const struct tb_property_dir *dir);
|
2017-10-02 18:38:30 +08:00
|
|
|
struct tb_property_dir *tb_property_create_dir(const uuid_t *uuid);
|
|
|
|
void tb_property_free_dir(struct tb_property_dir *dir);
|
|
|
|
int tb_property_add_immediate(struct tb_property_dir *parent, const char *key,
|
|
|
|
u32 value);
|
|
|
|
int tb_property_add_data(struct tb_property_dir *parent, const char *key,
|
|
|
|
const void *buf, size_t buflen);
|
|
|
|
int tb_property_add_text(struct tb_property_dir *parent, const char *key,
|
|
|
|
const char *text);
|
|
|
|
int tb_property_add_dir(struct tb_property_dir *parent, const char *key,
|
|
|
|
struct tb_property_dir *dir);
|
|
|
|
void tb_property_remove(struct tb_property *tb_property);
|
|
|
|
struct tb_property *tb_property_find(struct tb_property_dir *dir,
|
|
|
|
const char *key, enum tb_property_type type);
|
|
|
|
struct tb_property *tb_property_get_next(struct tb_property_dir *dir,
|
|
|
|
struct tb_property *prev);
|
|
|
|
|
|
|
|
#define tb_property_for_each(dir, property) \
|
|
|
|
for (property = tb_property_get_next(dir, NULL); \
|
|
|
|
property; \
|
|
|
|
property = tb_property_get_next(dir, property))
|
|
|
|
|
2017-10-02 18:38:34 +08:00
|
|
|
int tb_register_property_dir(const char *key, struct tb_property_dir *dir);
|
|
|
|
void tb_unregister_property_dir(const char *key, struct tb_property_dir *dir);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* struct tb_xdomain - Cross-domain (XDomain) connection
|
|
|
|
* @dev: XDomain device
|
|
|
|
* @tb: Pointer to the domain
|
|
|
|
* @remote_uuid: UUID of the remote domain (host)
|
|
|
|
* @local_uuid: Cached local UUID
|
|
|
|
* @route: Route string the other domain can be reached
|
|
|
|
* @vendor: Vendor ID of the remote domain
|
|
|
|
* @device: Device ID of the demote domain
|
2021-01-08 20:57:19 +08:00
|
|
|
* @local_max_hopid: Maximum input HopID of this host
|
|
|
|
* @remote_max_hopid: Maximum input HopID of the remote host
|
2017-10-02 18:38:34 +08:00
|
|
|
* @lock: Lock to serialize access to the following fields of this structure
|
|
|
|
* @vendor_name: Name of the vendor (or %NULL if not known)
|
|
|
|
* @device_name: Name of the device (or %NULL if not known)
|
2020-09-24 16:43:58 +08:00
|
|
|
* @link_speed: Speed of the link in Gb/s
|
|
|
|
* @link_width: Width of the link (1 or 2)
|
2022-08-30 23:32:47 +08:00
|
|
|
* @link_usb4: Downstream link is USB4
|
2017-10-02 18:38:34 +08:00
|
|
|
* @is_unplugged: The XDomain is unplugged
|
2018-09-28 21:30:16 +08:00
|
|
|
* @needs_uuid: If the XDomain does not have @remote_uuid it will be
|
|
|
|
* queried first
|
2017-10-02 18:38:34 +08:00
|
|
|
* @service_ids: Used to generate IDs for the services
|
2021-01-08 22:25:39 +08:00
|
|
|
* @in_hopids: Input HopIDs for DMA tunneling
|
|
|
|
* @out_hopids; Output HopIDs for DMA tunneling
|
2021-01-08 20:57:19 +08:00
|
|
|
* @local_property_block: Local block of properties
|
|
|
|
* @local_property_block_gen: Generation of @local_property_block
|
|
|
|
* @local_property_block_len: Length of the @local_property_block in dwords
|
|
|
|
* @remote_properties: Properties exported by the remote domain
|
|
|
|
* @remote_property_block_gen: Generation of @remote_properties
|
2022-02-13 22:44:45 +08:00
|
|
|
* @state: Next XDomain discovery state to run
|
|
|
|
* @state_work: Work used to run the next state
|
|
|
|
* @state_retries: Number of retries remain for the state
|
2017-10-02 18:38:34 +08:00
|
|
|
* @properties_changed_work: Work used to notify the remote domain that
|
|
|
|
* our properties have changed
|
|
|
|
* @properties_changed_retries: Number of times left to send properties
|
|
|
|
* changed notification
|
2022-02-13 22:44:45 +08:00
|
|
|
* @bonding_possible: True if lane bonding is possible on local side
|
|
|
|
* @target_link_width: Target link width from the remote host
|
2017-10-02 18:38:34 +08:00
|
|
|
* @link: Root switch link the remote domain is connected (ICM only)
|
|
|
|
* @depth: Depth in the chain the remote domain is connected (ICM only)
|
|
|
|
*
|
|
|
|
* This structure represents connection across two domains (hosts).
|
|
|
|
* Each XDomain contains zero or more services which are exposed as
|
|
|
|
* &struct tb_service objects.
|
|
|
|
*
|
|
|
|
* Service drivers may access this structure if they need to enumerate
|
|
|
|
* non-standard properties but they need hold @lock when doing so
|
|
|
|
* because properties can be changed asynchronously in response to
|
|
|
|
* changes in the remote domain.
|
|
|
|
*/
|
|
|
|
struct tb_xdomain {
|
|
|
|
struct device dev;
|
|
|
|
struct tb *tb;
|
|
|
|
uuid_t *remote_uuid;
|
|
|
|
const uuid_t *local_uuid;
|
|
|
|
u64 route;
|
|
|
|
u16 vendor;
|
|
|
|
u16 device;
|
2021-01-08 20:57:19 +08:00
|
|
|
unsigned int local_max_hopid;
|
|
|
|
unsigned int remote_max_hopid;
|
2017-10-02 18:38:34 +08:00
|
|
|
struct mutex lock;
|
|
|
|
const char *vendor_name;
|
|
|
|
const char *device_name;
|
2020-09-24 16:43:58 +08:00
|
|
|
unsigned int link_speed;
|
|
|
|
unsigned int link_width;
|
2022-08-30 23:32:47 +08:00
|
|
|
bool link_usb4;
|
2017-10-02 18:38:34 +08:00
|
|
|
bool is_unplugged;
|
2018-09-28 21:30:16 +08:00
|
|
|
bool needs_uuid;
|
2017-10-02 18:38:34 +08:00
|
|
|
struct ida service_ids;
|
2021-01-08 22:25:39 +08:00
|
|
|
struct ida in_hopids;
|
|
|
|
struct ida out_hopids;
|
2021-01-08 20:57:19 +08:00
|
|
|
u32 *local_property_block;
|
|
|
|
u32 local_property_block_gen;
|
|
|
|
u32 local_property_block_len;
|
|
|
|
struct tb_property_dir *remote_properties;
|
|
|
|
u32 remote_property_block_gen;
|
2022-02-13 22:44:45 +08:00
|
|
|
int state;
|
|
|
|
struct delayed_work state_work;
|
|
|
|
int state_retries;
|
2017-10-02 18:38:34 +08:00
|
|
|
struct delayed_work properties_changed_work;
|
|
|
|
int properties_changed_retries;
|
2022-02-13 22:44:45 +08:00
|
|
|
bool bonding_possible;
|
|
|
|
u8 target_link_width;
|
2017-10-02 18:38:34 +08:00
|
|
|
u8 link;
|
|
|
|
u8 depth;
|
|
|
|
};
|
|
|
|
|
2020-09-24 16:44:01 +08:00
|
|
|
int tb_xdomain_lane_bonding_enable(struct tb_xdomain *xd);
|
|
|
|
void tb_xdomain_lane_bonding_disable(struct tb_xdomain *xd);
|
2021-01-08 22:25:39 +08:00
|
|
|
int tb_xdomain_alloc_in_hopid(struct tb_xdomain *xd, int hopid);
|
|
|
|
void tb_xdomain_release_in_hopid(struct tb_xdomain *xd, int hopid);
|
|
|
|
int tb_xdomain_alloc_out_hopid(struct tb_xdomain *xd, int hopid);
|
|
|
|
void tb_xdomain_release_out_hopid(struct tb_xdomain *xd, int hopid);
|
|
|
|
int tb_xdomain_enable_paths(struct tb_xdomain *xd, int transmit_path,
|
|
|
|
int transmit_ring, int receive_path,
|
|
|
|
int receive_ring);
|
|
|
|
int tb_xdomain_disable_paths(struct tb_xdomain *xd, int transmit_path,
|
|
|
|
int transmit_ring, int receive_path,
|
|
|
|
int receive_ring);
|
|
|
|
|
|
|
|
static inline int tb_xdomain_disable_all_paths(struct tb_xdomain *xd)
|
|
|
|
{
|
|
|
|
return tb_xdomain_disable_paths(xd, -1, -1, -1, -1);
|
|
|
|
}
|
|
|
|
|
2017-10-02 18:38:34 +08:00
|
|
|
struct tb_xdomain *tb_xdomain_find_by_uuid(struct tb *tb, const uuid_t *uuid);
|
2017-10-04 19:53:54 +08:00
|
|
|
struct tb_xdomain *tb_xdomain_find_by_route(struct tb *tb, u64 route);
|
2017-10-02 18:38:34 +08:00
|
|
|
|
|
|
|
static inline struct tb_xdomain *
|
|
|
|
tb_xdomain_find_by_uuid_locked(struct tb *tb, const uuid_t *uuid)
|
|
|
|
{
|
|
|
|
struct tb_xdomain *xd;
|
|
|
|
|
|
|
|
mutex_lock(&tb->lock);
|
|
|
|
xd = tb_xdomain_find_by_uuid(tb, uuid);
|
|
|
|
mutex_unlock(&tb->lock);
|
|
|
|
|
|
|
|
return xd;
|
|
|
|
}
|
|
|
|
|
2017-10-04 19:53:54 +08:00
|
|
|
static inline struct tb_xdomain *
|
|
|
|
tb_xdomain_find_by_route_locked(struct tb *tb, u64 route)
|
|
|
|
{
|
|
|
|
struct tb_xdomain *xd;
|
|
|
|
|
|
|
|
mutex_lock(&tb->lock);
|
|
|
|
xd = tb_xdomain_find_by_route(tb, route);
|
|
|
|
mutex_unlock(&tb->lock);
|
|
|
|
|
|
|
|
return xd;
|
|
|
|
}
|
|
|
|
|
2017-10-02 18:38:34 +08:00
|
|
|
static inline struct tb_xdomain *tb_xdomain_get(struct tb_xdomain *xd)
|
|
|
|
{
|
|
|
|
if (xd)
|
|
|
|
get_device(&xd->dev);
|
|
|
|
return xd;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void tb_xdomain_put(struct tb_xdomain *xd)
|
|
|
|
{
|
|
|
|
if (xd)
|
|
|
|
put_device(&xd->dev);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline bool tb_is_xdomain(const struct device *dev)
|
|
|
|
{
|
|
|
|
return dev->type == &tb_xdomain_type;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline struct tb_xdomain *tb_to_xdomain(struct device *dev)
|
|
|
|
{
|
|
|
|
if (tb_is_xdomain(dev))
|
|
|
|
return container_of(dev, struct tb_xdomain, dev);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
int tb_xdomain_response(struct tb_xdomain *xd, const void *response,
|
|
|
|
size_t size, enum tb_cfg_pkg_type type);
|
|
|
|
int tb_xdomain_request(struct tb_xdomain *xd, const void *request,
|
|
|
|
size_t request_size, enum tb_cfg_pkg_type request_type,
|
|
|
|
void *response, size_t response_size,
|
|
|
|
enum tb_cfg_pkg_type response_type,
|
|
|
|
unsigned int timeout_msec);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* tb_protocol_handler - Protocol specific handler
|
|
|
|
* @uuid: XDomain messages with this UUID are dispatched to this handler
|
|
|
|
* @callback: Callback called with the XDomain message. Returning %1
|
|
|
|
* here tells the XDomain core that the message was handled
|
|
|
|
* by this handler and should not be forwared to other
|
|
|
|
* handlers.
|
|
|
|
* @data: Data passed with the callback
|
|
|
|
* @list: Handlers are linked using this
|
|
|
|
*
|
|
|
|
* Thunderbolt services can hook into incoming XDomain requests by
|
|
|
|
* registering protocol handler. Only limitation is that the XDomain
|
|
|
|
* discovery protocol UUID cannot be registered since it is handled by
|
|
|
|
* the core XDomain code.
|
|
|
|
*
|
|
|
|
* The @callback must check that the message is really directed to the
|
|
|
|
* service the driver implements.
|
|
|
|
*/
|
|
|
|
struct tb_protocol_handler {
|
|
|
|
const uuid_t *uuid;
|
|
|
|
int (*callback)(const void *buf, size_t size, void *data);
|
|
|
|
void *data;
|
|
|
|
struct list_head list;
|
|
|
|
};
|
|
|
|
|
|
|
|
int tb_register_protocol_handler(struct tb_protocol_handler *handler);
|
|
|
|
void tb_unregister_protocol_handler(struct tb_protocol_handler *handler);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* struct tb_service - Thunderbolt service
|
|
|
|
* @dev: XDomain device
|
|
|
|
* @id: ID of the service (shown in sysfs)
|
|
|
|
* @key: Protocol key from the properties directory
|
|
|
|
* @prtcid: Protocol ID from the properties directory
|
|
|
|
* @prtcvers: Protocol version from the properties directory
|
|
|
|
* @prtcrevs: Protocol software revision from the properties directory
|
|
|
|
* @prtcstns: Protocol settings mask from the properties directory
|
2020-10-07 22:53:44 +08:00
|
|
|
* @debugfs_dir: Pointer to the service debugfs directory. Always created
|
|
|
|
* when debugfs is enabled. Can be used by service drivers to
|
|
|
|
* add their own entries under the service.
|
2017-10-02 18:38:34 +08:00
|
|
|
*
|
|
|
|
* Each domain exposes set of services it supports as collection of
|
|
|
|
* properties. For each service there will be one corresponding
|
|
|
|
* &struct tb_service. Service drivers are bound to these.
|
|
|
|
*/
|
|
|
|
struct tb_service {
|
|
|
|
struct device dev;
|
|
|
|
int id;
|
|
|
|
const char *key;
|
|
|
|
u32 prtcid;
|
|
|
|
u32 prtcvers;
|
|
|
|
u32 prtcrevs;
|
|
|
|
u32 prtcstns;
|
2020-10-07 22:53:44 +08:00
|
|
|
struct dentry *debugfs_dir;
|
2017-10-02 18:38:34 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
static inline struct tb_service *tb_service_get(struct tb_service *svc)
|
|
|
|
{
|
|
|
|
if (svc)
|
|
|
|
get_device(&svc->dev);
|
|
|
|
return svc;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void tb_service_put(struct tb_service *svc)
|
|
|
|
{
|
|
|
|
if (svc)
|
|
|
|
put_device(&svc->dev);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline bool tb_is_service(const struct device *dev)
|
|
|
|
{
|
|
|
|
return dev->type == &tb_service_type;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline struct tb_service *tb_to_service(struct device *dev)
|
|
|
|
{
|
|
|
|
if (tb_is_service(dev))
|
|
|
|
return container_of(dev, struct tb_service, dev);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* tb_service_driver - Thunderbolt service driver
|
|
|
|
* @driver: Driver structure
|
|
|
|
* @probe: Called when the driver is probed
|
|
|
|
* @remove: Called when the driver is removed (optional)
|
|
|
|
* @shutdown: Called at shutdown time to stop the service (optional)
|
|
|
|
* @id_table: Table of service identifiers the driver supports
|
|
|
|
*/
|
|
|
|
struct tb_service_driver {
|
|
|
|
struct device_driver driver;
|
|
|
|
int (*probe)(struct tb_service *svc, const struct tb_service_id *id);
|
|
|
|
void (*remove)(struct tb_service *svc);
|
|
|
|
void (*shutdown)(struct tb_service *svc);
|
|
|
|
const struct tb_service_id *id_table;
|
|
|
|
};
|
|
|
|
|
|
|
|
#define TB_SERVICE(key, id) \
|
|
|
|
.match_flags = TBSVC_MATCH_PROTOCOL_KEY | \
|
|
|
|
TBSVC_MATCH_PROTOCOL_ID, \
|
|
|
|
.protocol_key = (key), \
|
|
|
|
.protocol_id = (id)
|
|
|
|
|
|
|
|
int tb_register_service_driver(struct tb_service_driver *drv);
|
|
|
|
void tb_unregister_service_driver(struct tb_service_driver *drv);
|
|
|
|
|
|
|
|
static inline void *tb_service_get_drvdata(const struct tb_service *svc)
|
|
|
|
{
|
|
|
|
return dev_get_drvdata(&svc->dev);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void tb_service_set_drvdata(struct tb_service *svc, void *data)
|
|
|
|
{
|
|
|
|
dev_set_drvdata(&svc->dev, data);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline struct tb_xdomain *tb_service_parent(struct tb_service *svc)
|
|
|
|
{
|
|
|
|
return tb_to_xdomain(svc->dev.parent);
|
|
|
|
}
|
|
|
|
|
2017-10-02 18:38:37 +08:00
|
|
|
/**
|
|
|
|
* struct tb_nhi - thunderbolt native host interface
|
|
|
|
* @lock: Must be held during ring creation/destruction. Is acquired by
|
|
|
|
* interrupt_work when dispatching interrupts to individual rings.
|
|
|
|
* @pdev: Pointer to the PCI device
|
2018-01-17 04:19:00 +08:00
|
|
|
* @ops: NHI specific optional ops
|
2017-10-02 18:38:37 +08:00
|
|
|
* @iobase: MMIO space of the NHI
|
|
|
|
* @tx_rings: All Tx rings available on this host controller
|
|
|
|
* @rx_rings: All Rx rings available on this host controller
|
|
|
|
* @msix_ida: Used to allocate MSI-X vectors for rings
|
|
|
|
* @going_away: The host controller device is about to disappear so when
|
|
|
|
* this flag is set, avoid touching the hardware anymore.
|
thunderbolt: Make iommu_dma_protection more accurate
Between me trying to get rid of iommu_present() and Mario wanting to
support the AMD equivalent of DMAR_PLATFORM_OPT_IN, scrutiny has shown
that the iommu_dma_protection attribute is being far too optimistic.
Even if an IOMMU might be present for some PCI segment in the system,
that doesn't necessarily mean it provides translation for the device(s)
we care about. Furthermore, all that DMAR_PLATFORM_OPT_IN really does
is tell us that memory was protected before the kernel was loaded, and
prevent the user from disabling the intel-iommu driver entirely. While
that lets us assume kernel integrity, what matters for actual runtime
DMA protection is whether we trust individual devices, based on the
"external facing" property that we expect firmware to describe for
Thunderbolt ports.
It's proven challenging to determine the appropriate ports accurately
given the variety of possible topologies, so while still not getting a
perfect answer, by putting enough faith in firmware we can at least get
a good bit closer. If we can see that any device near a Thunderbolt NHI
has all the requisites for Kernel DMA Protection, chances are that it
*is* a relevant port, but moreover that implies that firmware is playing
the game overall, so we'll use that to assume that all Thunderbolt ports
should be correctly marked and thus will end up fully protected.
CC: Mario Limonciello <mario.limonciello@amd.com>
Reviewed-by: Christoph Hellwig <hch@lst.de>
Acked-by: Mika Westerberg <mika.westerberg@linux.intel.com>
Signed-off-by: Robin Murphy <robin.murphy@arm.com>
Link: https://lore.kernel.org/r/b153f208bc9eafab5105bad0358b77366509d2d4.1650878781.git.robin.murphy@arm.com
Signed-off-by: Joerg Roedel <jroedel@suse.de>
2022-04-25 20:42:04 +08:00
|
|
|
* @iommu_dma_protection: An IOMMU will isolate external-facing ports.
|
2017-10-02 18:38:37 +08:00
|
|
|
* @interrupt_work: Work scheduled to handle ring interrupt when no
|
|
|
|
* MSI-X is used.
|
|
|
|
* @hop_count: Number of rings (end point hops) supported by NHI.
|
2021-08-07 00:59:05 +08:00
|
|
|
* @quirks: NHI specific quirks if any
|
2017-10-02 18:38:37 +08:00
|
|
|
*/
|
|
|
|
struct tb_nhi {
|
2017-10-02 18:38:40 +08:00
|
|
|
spinlock_t lock;
|
2017-10-02 18:38:37 +08:00
|
|
|
struct pci_dev *pdev;
|
2018-01-17 04:19:00 +08:00
|
|
|
const struct tb_nhi_ops *ops;
|
2017-10-02 18:38:37 +08:00
|
|
|
void __iomem *iobase;
|
|
|
|
struct tb_ring **tx_rings;
|
|
|
|
struct tb_ring **rx_rings;
|
|
|
|
struct ida msix_ida;
|
|
|
|
bool going_away;
|
thunderbolt: Make iommu_dma_protection more accurate
Between me trying to get rid of iommu_present() and Mario wanting to
support the AMD equivalent of DMAR_PLATFORM_OPT_IN, scrutiny has shown
that the iommu_dma_protection attribute is being far too optimistic.
Even if an IOMMU might be present for some PCI segment in the system,
that doesn't necessarily mean it provides translation for the device(s)
we care about. Furthermore, all that DMAR_PLATFORM_OPT_IN really does
is tell us that memory was protected before the kernel was loaded, and
prevent the user from disabling the intel-iommu driver entirely. While
that lets us assume kernel integrity, what matters for actual runtime
DMA protection is whether we trust individual devices, based on the
"external facing" property that we expect firmware to describe for
Thunderbolt ports.
It's proven challenging to determine the appropriate ports accurately
given the variety of possible topologies, so while still not getting a
perfect answer, by putting enough faith in firmware we can at least get
a good bit closer. If we can see that any device near a Thunderbolt NHI
has all the requisites for Kernel DMA Protection, chances are that it
*is* a relevant port, but moreover that implies that firmware is playing
the game overall, so we'll use that to assume that all Thunderbolt ports
should be correctly marked and thus will end up fully protected.
CC: Mario Limonciello <mario.limonciello@amd.com>
Reviewed-by: Christoph Hellwig <hch@lst.de>
Acked-by: Mika Westerberg <mika.westerberg@linux.intel.com>
Signed-off-by: Robin Murphy <robin.murphy@arm.com>
Link: https://lore.kernel.org/r/b153f208bc9eafab5105bad0358b77366509d2d4.1650878781.git.robin.murphy@arm.com
Signed-off-by: Joerg Roedel <jroedel@suse.de>
2022-04-25 20:42:04 +08:00
|
|
|
bool iommu_dma_protection;
|
2017-10-02 18:38:37 +08:00
|
|
|
struct work_struct interrupt_work;
|
|
|
|
u32 hop_count;
|
2021-08-07 00:59:05 +08:00
|
|
|
unsigned long quirks;
|
2017-10-02 18:38:37 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* struct tb_ring - thunderbolt TX or RX ring associated with a NHI
|
|
|
|
* @lock: Lock serializing actions to this ring. Must be acquired after
|
|
|
|
* nhi->lock.
|
|
|
|
* @nhi: Pointer to the native host controller interface
|
|
|
|
* @size: Size of the ring
|
|
|
|
* @hop: Hop (DMA channel) associated with this ring
|
|
|
|
* @head: Head of the ring (write next descriptor here)
|
|
|
|
* @tail: Tail of the ring (complete next descriptor here)
|
|
|
|
* @descriptors: Allocated descriptors for this ring
|
|
|
|
* @queue: Queue holding frames to be transferred over this ring
|
|
|
|
* @in_flight: Queue holding frames that are currently in flight
|
|
|
|
* @work: Interrupt work structure
|
|
|
|
* @is_tx: Is the ring Tx or Rx
|
|
|
|
* @running: Is the ring running
|
|
|
|
* @irq: MSI-X irq number if the ring uses MSI-X. %0 otherwise.
|
|
|
|
* @vector: MSI-X vector number the ring uses (only set if @irq is > 0)
|
|
|
|
* @flags: Ring specific flags
|
2020-10-20 00:15:20 +08:00
|
|
|
* @e2e_tx_hop: Transmit HopID when E2E is enabled. Only applicable to
|
|
|
|
* RX ring. For TX ring this should be set to %0.
|
2017-10-02 18:38:37 +08:00
|
|
|
* @sof_mask: Bit mask used to detect start of frame PDF
|
|
|
|
* @eof_mask: Bit mask used to detect end of frame PDF
|
2017-10-02 18:38:41 +08:00
|
|
|
* @start_poll: Called when ring interrupt is triggered to start
|
|
|
|
* polling. Passing %NULL keeps the ring in interrupt mode.
|
|
|
|
* @poll_data: Data passed to @start_poll
|
2017-10-02 18:38:37 +08:00
|
|
|
*/
|
|
|
|
struct tb_ring {
|
2017-10-02 18:38:39 +08:00
|
|
|
spinlock_t lock;
|
2017-10-02 18:38:37 +08:00
|
|
|
struct tb_nhi *nhi;
|
|
|
|
int size;
|
|
|
|
int hop;
|
|
|
|
int head;
|
|
|
|
int tail;
|
|
|
|
struct ring_desc *descriptors;
|
|
|
|
dma_addr_t descriptors_dma;
|
|
|
|
struct list_head queue;
|
|
|
|
struct list_head in_flight;
|
|
|
|
struct work_struct work;
|
|
|
|
bool is_tx:1;
|
|
|
|
bool running:1;
|
|
|
|
int irq;
|
|
|
|
u8 vector;
|
|
|
|
unsigned int flags;
|
2020-10-20 00:15:20 +08:00
|
|
|
int e2e_tx_hop;
|
2017-10-02 18:38:37 +08:00
|
|
|
u16 sof_mask;
|
|
|
|
u16 eof_mask;
|
2017-10-02 18:38:41 +08:00
|
|
|
void (*start_poll)(void *data);
|
|
|
|
void *poll_data;
|
2017-10-02 18:38:37 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
/* Leave ring interrupt enabled on suspend */
|
|
|
|
#define RING_FLAG_NO_SUSPEND BIT(0)
|
|
|
|
/* Configure the ring to be in frame mode */
|
|
|
|
#define RING_FLAG_FRAME BIT(1)
|
2020-10-20 00:15:20 +08:00
|
|
|
/* Enable end-to-end flow control */
|
|
|
|
#define RING_FLAG_E2E BIT(2)
|
2017-10-02 18:38:37 +08:00
|
|
|
|
|
|
|
struct ring_frame;
|
|
|
|
typedef void (*ring_cb)(struct tb_ring *, struct ring_frame *, bool canceled);
|
|
|
|
|
2017-10-02 18:38:38 +08:00
|
|
|
/**
|
|
|
|
* enum ring_desc_flags - Flags for DMA ring descriptor
|
|
|
|
* %RING_DESC_ISOCH: Enable isonchronous DMA (Tx only)
|
|
|
|
* %RING_DESC_CRC_ERROR: In frame mode CRC check failed for the frame (Rx only)
|
|
|
|
* %RING_DESC_COMPLETED: Descriptor completed (set by NHI)
|
|
|
|
* %RING_DESC_POSTED: Always set this
|
|
|
|
* %RING_DESC_BUFFER_OVERRUN: RX buffer overrun
|
|
|
|
* %RING_DESC_INTERRUPT: Request an interrupt on completion
|
|
|
|
*/
|
|
|
|
enum ring_desc_flags {
|
|
|
|
RING_DESC_ISOCH = 0x1,
|
|
|
|
RING_DESC_CRC_ERROR = 0x1,
|
|
|
|
RING_DESC_COMPLETED = 0x2,
|
|
|
|
RING_DESC_POSTED = 0x4,
|
|
|
|
RING_DESC_BUFFER_OVERRUN = 0x04,
|
|
|
|
RING_DESC_INTERRUPT = 0x8,
|
|
|
|
};
|
|
|
|
|
2017-10-02 18:38:37 +08:00
|
|
|
/**
|
|
|
|
* struct ring_frame - For use with ring_rx/ring_tx
|
|
|
|
* @buffer_phy: DMA mapped address of the frame
|
2017-10-02 18:38:41 +08:00
|
|
|
* @callback: Callback called when the frame is finished (optional)
|
2017-10-02 18:38:37 +08:00
|
|
|
* @list: Frame is linked to a queue using this
|
|
|
|
* @size: Size of the frame in bytes (%0 means %4096)
|
|
|
|
* @flags: Flags for the frame (see &enum ring_desc_flags)
|
|
|
|
* @eof: End of frame protocol defined field
|
|
|
|
* @sof: Start of frame protocol defined field
|
|
|
|
*/
|
|
|
|
struct ring_frame {
|
|
|
|
dma_addr_t buffer_phy;
|
|
|
|
ring_cb callback;
|
|
|
|
struct list_head list;
|
|
|
|
u32 size:12;
|
|
|
|
u32 flags:12;
|
|
|
|
u32 eof:4;
|
|
|
|
u32 sof:4;
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Minimum size for ring_rx */
|
|
|
|
#define TB_FRAME_SIZE 0x100
|
|
|
|
|
|
|
|
struct tb_ring *tb_ring_alloc_tx(struct tb_nhi *nhi, int hop, int size,
|
|
|
|
unsigned int flags);
|
|
|
|
struct tb_ring *tb_ring_alloc_rx(struct tb_nhi *nhi, int hop, int size,
|
2020-10-20 00:15:20 +08:00
|
|
|
unsigned int flags, int e2e_tx_hop,
|
|
|
|
u16 sof_mask, u16 eof_mask,
|
2017-10-02 18:38:41 +08:00
|
|
|
void (*start_poll)(void *), void *poll_data);
|
2017-10-02 18:38:37 +08:00
|
|
|
void tb_ring_start(struct tb_ring *ring);
|
|
|
|
void tb_ring_stop(struct tb_ring *ring);
|
|
|
|
void tb_ring_free(struct tb_ring *ring);
|
|
|
|
|
|
|
|
int __tb_ring_enqueue(struct tb_ring *ring, struct ring_frame *frame);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* tb_ring_rx() - enqueue a frame on an RX ring
|
|
|
|
* @ring: Ring to enqueue the frame
|
|
|
|
* @frame: Frame to enqueue
|
|
|
|
*
|
2017-10-02 18:38:41 +08:00
|
|
|
* @frame->buffer, @frame->buffer_phy have to be set. The buffer must
|
|
|
|
* contain at least %TB_FRAME_SIZE bytes.
|
2017-10-02 18:38:37 +08:00
|
|
|
*
|
|
|
|
* @frame->callback will be invoked with @frame->size, @frame->flags,
|
|
|
|
* @frame->eof, @frame->sof set once the frame has been received.
|
|
|
|
*
|
|
|
|
* If ring_stop() is called after the packet has been enqueued
|
|
|
|
* @frame->callback will be called with canceled set to true.
|
|
|
|
*
|
|
|
|
* Return: Returns %-ESHUTDOWN if ring_stop has been called. Zero otherwise.
|
|
|
|
*/
|
|
|
|
static inline int tb_ring_rx(struct tb_ring *ring, struct ring_frame *frame)
|
|
|
|
{
|
|
|
|
WARN_ON(ring->is_tx);
|
|
|
|
return __tb_ring_enqueue(ring, frame);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* tb_ring_tx() - enqueue a frame on an TX ring
|
|
|
|
* @ring: Ring the enqueue the frame
|
|
|
|
* @frame: Frame to enqueue
|
|
|
|
*
|
2017-10-02 18:38:41 +08:00
|
|
|
* @frame->buffer, @frame->buffer_phy, @frame->size, @frame->eof and
|
|
|
|
* @frame->sof have to be set.
|
2017-10-02 18:38:37 +08:00
|
|
|
*
|
|
|
|
* @frame->callback will be invoked with once the frame has been transmitted.
|
|
|
|
*
|
|
|
|
* If ring_stop() is called after the packet has been enqueued @frame->callback
|
|
|
|
* will be called with canceled set to true.
|
|
|
|
*
|
|
|
|
* Return: Returns %-ESHUTDOWN if ring_stop has been called. Zero otherwise.
|
|
|
|
*/
|
|
|
|
static inline int tb_ring_tx(struct tb_ring *ring, struct ring_frame *frame)
|
|
|
|
{
|
|
|
|
WARN_ON(!ring->is_tx);
|
|
|
|
return __tb_ring_enqueue(ring, frame);
|
|
|
|
}
|
|
|
|
|
2017-10-02 18:38:41 +08:00
|
|
|
/* Used only when the ring is in polling mode */
|
|
|
|
struct ring_frame *tb_ring_poll(struct tb_ring *ring);
|
|
|
|
void tb_ring_poll_complete(struct tb_ring *ring);
|
|
|
|
|
2017-10-02 18:38:42 +08:00
|
|
|
/**
|
|
|
|
* tb_ring_dma_device() - Return device used for DMA mapping
|
|
|
|
* @ring: Ring whose DMA device is retrieved
|
|
|
|
*
|
|
|
|
* Use this function when you are mapping DMA for buffers that are
|
|
|
|
* passed to the ring for sending/receiving.
|
|
|
|
*/
|
|
|
|
static inline struct device *tb_ring_dma_device(struct tb_ring *ring)
|
|
|
|
{
|
|
|
|
return &ring->nhi->pdev->dev;
|
|
|
|
}
|
|
|
|
|
2017-10-02 18:38:30 +08:00
|
|
|
#endif /* THUNDERBOLT_H_ */
|