

# Method for sharing a (PCI) device between multiple PMDs

Fiona Trahe DPDK Summit Userspace - Dublin- 2017

### Problem Statement



- Hardware accelerators can provide multiple functions via the same PCI device
  - e.g. Intel's QuickAssist devices provide symmetric crypto, asymmetric crypto and compression functions under the same BAR.
- EAL PCI presumes one and only one driver per device. This can lead to artificially bundling multiple functions under one API though it may be desirable to present these to applications using separate APIs.
- This presentation proposes a mechanism to share a pci device between multiple PMDs.
- It may also be extendable to non-pci devices.

### Example problem



- Using Symmetric and Asymmetric crypto as examples of independent functions in the following diagrams.
- Just to illustrate the generic problem.
- Doesn't imply that these APIs shouldn't be grouped under the same API, as currently proposed in Asymmetric RFC.

### Limitation: 1 PCI device – 1 driver



#### Option1 – combine APIs



### Option2 – allow PMDs to share devices





- Up to PMDs to decide how they can share. Not prescriptive, it will vary depending on HW device design.
- For example it could contain
  - ▶ a set of resources. Each driver marks which resources it's using.
  - Iocks to control access to shared CSRs and shared data.
  - state, e.g. if some part of device setup should only be done once, then could be done by first PMD and marked as done.

### Status of development



- POC working with 2 QAT PMDs, the sym crypto driver and another dummy driver.
- Open issue re how ethdev drivers can accommodate the changes.

Some of the tricky bits:

- Cryptodev used BDF string as name, needed to be unique, e.g. name = "0000:02:01.1" Fixed by adding function to device name e.g. "0000:02.01.1.sym" and "0000:02.01.1.asym".
- Shareability added as driver attribute rather than device attribute. I.e. indicates driver understands how to share devices. It's not a promise that it will share them. It could mark in the shared data area that it has taken all the resources and won't share. If a second driver needs resources already taken by the first it will only be probed if it's capable of sharing and should fail to start due to lack of resources.
- rte\_pci\_devices are in global table. Each contains rte\_driver and rte\_pci\_driver ptrs. Written by first probe, overwritten by second. Fix: Replaced rte\_pci\_driver with an array.
  Repercussions in ethdev need investigation. Still investigating rte\_driver usage.

#### What it's not



- Not a mechanism for sharing a device across processes. Shares a device across PMDs within a process NOT across processes.
- Not a mechanism for making all devices on a bus shareable. The driver decides whether it's capable of sharing a specific device-type with another driver.
- Not a framework for handling how the drivers share the device. The details of how they share it are controlled by the PMDs, not part of this mechanism. Supporting functions, e.g. like a regmap library for managing access to shared CSRs could be added later.

### Code snippet

```
@@ -131,12 +128,10 @@ struct rte_pci_device {
       TAILQ_ENTRY(rte_pci_device) next; /**< Next probed PCI device. */</pre>
       struct rte device device;
                                      /**< Inherit core device */
       struct rte_pci_addr addr;
                                              /**< PCI location. */
       struct rte_pci_id id;
                                               /**< PCI ID. */
       struct rte mem resource mem resource[PCI MAX RESOURCE];
                                               /**< PCI Memory Resource */
       struct rte_intr_handle intr_handle;
                                              /**< Interrupt handle */
       struct rte_pci_driver *driver[PCI_MAX_DRIVERS_SHARING]; /**< Associated drivers */</pre>
       struct rte pci driver *driver;
                                       /**< Associated driver */
       uint16 t max vfs;
                                             /**< sriov enable if not zero */
       enum rte_kernel_driver kdrv;
                                         /**< Kernel driver passthrough */
                                      /**< PCI location (ASCII) */
       char name[PCI PRI STR SIZE+1];
                                             /**< For shared devices, number of drivers sharing */</pre>
       uint8 t driver count;
       void * priv_shared_data;
                                              /**< private data for shared devices */</pre>
 };
```

#### Code snippets



```
static struct rte_pci_driver rte_qat_pmd = {
       .id table = pci id qat map,
       .drv_flags = RTE_PCI_DRV_NEED_MAPPING | RTE_PCI_DRV_CAN_SHARE,
        .drv flags = RTE PCI DRV NEED MAPPING,
       .probe = crypto qat pci probe,
       .remove = crypto_qat_pci_remove
};
                          FOREACH DRIVER ON PCIBUS(dr) {
                                  rc = rte pci probe one driver(dr, dev);
                                  if (rc < 0)
                                         /* negative value is an error */
                                          return -1;
                                  if (rc > 0)
                                          /* positive value means driver doesn't support it */
                                         continue;
                                  /* driver has claimed this device.
                                   * Check if it can share it with other PMDs, if so continue searching.*/
                                  if (dr->drv flags & RTE PCI DRV CAN SHARE)
                                         continue:
                                  return 0;
```

### Questions?

### Feedback?

Fiona Trahe fiona.trahe@intel.com