mirror of
https://github.com/torvalds/linux.git
synced 2024-12-26 21:02:19 +00:00
721c551d31
Signed-off-by: Alastair D'Silva <alastair@d-silva.org> Acked-by: Frederic Barrat <fbarrat@linux.vnet.ibm.com> Acked-by: Andrew Donnellan <andrew.donnellan@au1.ibm.com> Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
177 lines
5.4 KiB
ReStructuredText
177 lines
5.4 KiB
ReStructuredText
========================================================
|
|
OpenCAPI (Open Coherent Accelerator Processor Interface)
|
|
========================================================
|
|
|
|
OpenCAPI is an interface between processors and accelerators. It aims
|
|
at being low-latency and high-bandwidth. The specification is
|
|
developed by the `OpenCAPI Consortium <http://opencapi.org/>`_.
|
|
|
|
It allows an accelerator (which could be a FPGA, ASICs, ...) to access
|
|
the host memory coherently, using virtual addresses. An OpenCAPI
|
|
device can also host its own memory, that can be accessed from the
|
|
host.
|
|
|
|
OpenCAPI is known in linux as 'ocxl', as the open, processor-agnostic
|
|
evolution of 'cxl' (the driver for the IBM CAPI interface for
|
|
powerpc), which was named that way to avoid confusion with the ISDN
|
|
CAPI subsystem.
|
|
|
|
|
|
High-level view
|
|
===============
|
|
|
|
OpenCAPI defines a Data Link Layer (DL) and Transaction Layer (TL), to
|
|
be implemented on top of a physical link. Any processor or device
|
|
implementing the DL and TL can start sharing memory.
|
|
|
|
::
|
|
|
|
+-----------+ +-------------+
|
|
| | | |
|
|
| | | Accelerated |
|
|
| Processor | | Function |
|
|
| | +--------+ | Unit | +--------+
|
|
| |--| Memory | | (AFU) |--| Memory |
|
|
| | +--------+ | | +--------+
|
|
+-----------+ +-------------+
|
|
| |
|
|
+-----------+ +-------------+
|
|
| TL | | TLX |
|
|
+-----------+ +-------------+
|
|
| |
|
|
+-----------+ +-------------+
|
|
| DL | | DLX |
|
|
+-----------+ +-------------+
|
|
| |
|
|
| PHY |
|
|
+---------------------------------------+
|
|
|
|
|
|
|
|
Device discovery
|
|
================
|
|
|
|
OpenCAPI relies on a PCI-like configuration space, implemented on the
|
|
device. So the host can discover AFUs by querying the config space.
|
|
|
|
OpenCAPI devices in Linux are treated like PCI devices (with a few
|
|
caveats). The firmware is expected to abstract the hardware as if it
|
|
was a PCI link. A lot of the existing PCI infrastructure is reused:
|
|
devices are scanned and BARs are assigned during the standard PCI
|
|
enumeration. Commands like 'lspci' can therefore be used to see what
|
|
devices are available.
|
|
|
|
The configuration space defines the AFU(s) that can be found on the
|
|
physical adapter, such as its name, how many memory contexts it can
|
|
work with, the size of its MMIO areas, ...
|
|
|
|
|
|
|
|
MMIO
|
|
====
|
|
|
|
OpenCAPI defines two MMIO areas for each AFU:
|
|
|
|
* the global MMIO area, with registers pertinent to the whole AFU.
|
|
* a per-process MMIO area, which has a fixed size for each context.
|
|
|
|
|
|
|
|
AFU interrupts
|
|
==============
|
|
|
|
OpenCAPI includes the possibility for an AFU to send an interrupt to a
|
|
host process. It is done through a 'intrp_req' defined in the
|
|
Transaction Layer, specifying a 64-bit object handle which defines the
|
|
interrupt.
|
|
|
|
The driver allows a process to allocate an interrupt and obtain its
|
|
64-bit object handle, that can be passed to the AFU.
|
|
|
|
|
|
|
|
char devices
|
|
============
|
|
|
|
The driver creates one char device per AFU found on the physical
|
|
device. A physical device may have multiple functions and each
|
|
function can have multiple AFUs. At the time of this writing though,
|
|
it has only been tested with devices exporting only one AFU.
|
|
|
|
Char devices can be found in /dev/ocxl/ and are named as:
|
|
/dev/ocxl/<AFU name>.<location>.<index>
|
|
|
|
where <AFU name> is a max 20-character long name, as found in the
|
|
config space of the AFU.
|
|
<location> is added by the driver and can help distinguish devices
|
|
when a system has more than one instance of the same OpenCAPI device.
|
|
<index> is also to help distinguish AFUs in the unlikely case where a
|
|
device carries multiple copies of the same AFU.
|
|
|
|
|
|
|
|
Sysfs class
|
|
===========
|
|
|
|
An ocxl class is added for the devices representing the AFUs. See
|
|
/sys/class/ocxl. The layout is described in
|
|
Documentation/ABI/testing/sysfs-class-ocxl
|
|
|
|
|
|
|
|
User API
|
|
========
|
|
|
|
open
|
|
----
|
|
|
|
Based on the AFU definition found in the config space, an AFU may
|
|
support working with more than one memory context, in which case the
|
|
associated char device may be opened multiple times by different
|
|
processes.
|
|
|
|
|
|
ioctl
|
|
-----
|
|
|
|
OCXL_IOCTL_ATTACH:
|
|
|
|
Attach the memory context of the calling process to the AFU so that
|
|
the AFU can access its memory.
|
|
|
|
OCXL_IOCTL_IRQ_ALLOC:
|
|
|
|
Allocate an AFU interrupt and return an identifier.
|
|
|
|
OCXL_IOCTL_IRQ_FREE:
|
|
|
|
Free a previously allocated AFU interrupt.
|
|
|
|
OCXL_IOCTL_IRQ_SET_FD:
|
|
|
|
Associate an event fd to an AFU interrupt so that the user process
|
|
can be notified when the AFU sends an interrupt.
|
|
|
|
OCXL_IOCTL_GET_METADATA:
|
|
|
|
Obtains configuration information from the card, such at the size of
|
|
MMIO areas, the AFU version, and the PASID for the current context.
|
|
|
|
OCXL_IOCTL_ENABLE_P9_WAIT:
|
|
|
|
Allows the AFU to wake a userspace thread executing 'wait'. Returns
|
|
information to userspace to allow it to configure the AFU. Note that
|
|
this is only available on POWER9.
|
|
|
|
OCXL_IOCTL_GET_FEATURES:
|
|
|
|
Reports on which CPU features that affect OpenCAPI are usable from
|
|
userspace.
|
|
|
|
|
|
mmap
|
|
----
|
|
|
|
A process can mmap the per-process MMIO area for interactions with the
|
|
AFU.
|