Many open source projects have moved to using SPDX identifiers to specify license information, reducing the amount of boilerplate code in every source file. This patch replaces the bulk of SPDK .c, .cpp and Makefiles with the BSD-3-Clause identifier. Almost all of these files share the exact same license text, and this patch only modifies the files that contain the most common license text. There can be slight variations because the third clause contains company names - most say "Intel Corporation", but there are instances for Nvidia, Samsung, Eideticom and even "the copyright holder". Used a bash script to automate replacement of the license text with SPDX identifier which is checked into scripts/spdx.sh. Signed-off-by: Jim Harris <james.r.harris@intel.com> Change-Id: Iaa88ab5e92ea471691dc298cfe41ebfb5d169780 Reviewed-on: https://review.spdk.io/gerrit/c/spdk/spdk/+/12904 Community-CI: Broadcom CI <spdk-ci.pdl@broadcom.com> Community-CI: Mellanox Build Bot Tested-by: SPDK CI Jenkins <sys_sgci@intel.com> Reviewed-by: Aleksey Marchuk <alexeymar@nvidia.com> Reviewed-by: Changpeng Liu <changpeng.liu@intel.com> Reviewed-by: Dong Yi <dongx.yi@intel.com> Reviewed-by: Konrad Sztyber <konrad.sztyber@intel.com> Reviewed-by: Paul Luse <paul.e.luse@intel.com> Reviewed-by: <qun.wan@intel.com>
239 lines
5.9 KiB
C
239 lines
5.9 KiB
C
/* SPDX-License-Identifier: BSD-3-Clause
|
|
* Copyright (c) Intel Corporation.
|
|
* All rights reserved.
|
|
*/
|
|
|
|
#include "spdk/stdinc.h"
|
|
#include "spdk/string.h"
|
|
|
|
#include "spdk/log.h"
|
|
#include "spdk/env.h"
|
|
|
|
#ifdef __linux__
|
|
|
|
#include <linux/netlink.h>
|
|
|
|
#define SPDK_UEVENT_MSG_LEN 4096
|
|
#define SPDK_UEVENT_RECVBUF_SIZE 1024 * 1024
|
|
|
|
int
|
|
spdk_pci_event_listen(void)
|
|
{
|
|
struct sockaddr_nl addr;
|
|
int netlink_fd;
|
|
int size = SPDK_UEVENT_RECVBUF_SIZE;
|
|
int buf_size;
|
|
socklen_t opt_size;
|
|
int flag, rc;
|
|
|
|
memset(&addr, 0, sizeof(addr));
|
|
addr.nl_family = AF_NETLINK;
|
|
addr.nl_pid = 0;
|
|
addr.nl_groups = 0xffffffff;
|
|
|
|
netlink_fd = socket(PF_NETLINK, SOCK_DGRAM, NETLINK_KOBJECT_UEVENT);
|
|
if (netlink_fd < 0) {
|
|
SPDK_ERRLOG("Failed to create netlink socket\n");
|
|
return netlink_fd;
|
|
}
|
|
|
|
if (setsockopt(netlink_fd, SOL_SOCKET, SO_RCVBUFFORCE, &size, sizeof(size)) < 0) {
|
|
if (setsockopt(netlink_fd, SOL_SOCKET, SO_RCVBUF, &size, sizeof(size)) < 0) {
|
|
rc = errno;
|
|
SPDK_ERRLOG("Failed to set socket option SO_RCVBUF\n");
|
|
goto error;
|
|
}
|
|
opt_size = sizeof(buf_size);
|
|
if (getsockopt(netlink_fd, SOL_SOCKET, SO_RCVBUF, &buf_size, &opt_size) < 0) {
|
|
rc = errno;
|
|
SPDK_ERRLOG("Failed to get socket option SO_RCVBUF\n");
|
|
goto error;
|
|
}
|
|
if (buf_size < SPDK_UEVENT_RECVBUF_SIZE) {
|
|
SPDK_ERRLOG("Socket recv buffer is too small (< %d), see SO_RCVBUF "
|
|
"section in socket(7) man page for specifics on how to "
|
|
"adjust the system setting.", SPDK_UEVENT_RECVBUF_SIZE);
|
|
rc = ENOSPC;
|
|
goto error;
|
|
}
|
|
}
|
|
|
|
flag = fcntl(netlink_fd, F_GETFL);
|
|
if (flag < 0) {
|
|
rc = errno;
|
|
SPDK_ERRLOG("Failed to get socket flag, fd: %d\n", netlink_fd);
|
|
goto error;
|
|
}
|
|
|
|
if (fcntl(netlink_fd, F_SETFL, flag | O_NONBLOCK) < 0) {
|
|
rc = errno;
|
|
SPDK_ERRLOG("Fcntl can't set nonblocking mode for socket, fd: %d\n", netlink_fd);
|
|
goto error;
|
|
}
|
|
|
|
if (bind(netlink_fd, (struct sockaddr *) &addr, sizeof(addr)) < 0) {
|
|
rc = errno;
|
|
SPDK_ERRLOG("Failed to bind the netlink\n");
|
|
goto error;
|
|
}
|
|
|
|
return netlink_fd;
|
|
error:
|
|
close(netlink_fd);
|
|
return -rc;
|
|
}
|
|
|
|
/* Note: We parse the event from uio and vfio subsystem and will ignore
|
|
* all the event from other subsystem. the event from uio subsystem
|
|
* as below:
|
|
* action: "add" or "remove"
|
|
* subsystem: "uio"
|
|
* dev_path: "/devices/pci0000:80/0000:80:01.0/0000:81:00.0/uio/uio0"
|
|
* VFIO subsystem add event:
|
|
* ACTION=bind
|
|
* DRIVER=vfio-pci
|
|
* PCI_SLOT_NAME=0000:d8:00.0
|
|
*/
|
|
static int
|
|
parse_subsystem_event(const char *buf, struct spdk_pci_event *event)
|
|
{
|
|
char subsystem[SPDK_UEVENT_MSG_LEN];
|
|
char action[SPDK_UEVENT_MSG_LEN];
|
|
char dev_path[SPDK_UEVENT_MSG_LEN];
|
|
char driver[SPDK_UEVENT_MSG_LEN];
|
|
char vfio_pci_addr[SPDK_UEVENT_MSG_LEN];
|
|
char *pci_address, *tmp;
|
|
int rc;
|
|
|
|
memset(subsystem, 0, SPDK_UEVENT_MSG_LEN);
|
|
memset(action, 0, SPDK_UEVENT_MSG_LEN);
|
|
memset(dev_path, 0, SPDK_UEVENT_MSG_LEN);
|
|
memset(driver, 0, SPDK_UEVENT_MSG_LEN);
|
|
memset(vfio_pci_addr, 0, SPDK_UEVENT_MSG_LEN);
|
|
|
|
while (*buf) {
|
|
if (!strncmp(buf, "SUBSYSTEM=", 10)) {
|
|
buf += 10;
|
|
snprintf(subsystem, sizeof(subsystem), "%s", buf);
|
|
} else if (!strncmp(buf, "ACTION=", 7)) {
|
|
buf += 7;
|
|
snprintf(action, sizeof(action), "%s", buf);
|
|
} else if (!strncmp(buf, "DEVPATH=", 8)) {
|
|
buf += 8;
|
|
snprintf(dev_path, sizeof(dev_path), "%s", buf);
|
|
} else if (!strncmp(buf, "DRIVER=", 7)) {
|
|
buf += 7;
|
|
snprintf(driver, sizeof(driver), "%s", buf);
|
|
} else if (!strncmp(buf, "PCI_SLOT_NAME=", 14)) {
|
|
buf += 14;
|
|
snprintf(vfio_pci_addr, sizeof(vfio_pci_addr), "%s", buf);
|
|
}
|
|
|
|
while (*buf++)
|
|
;
|
|
}
|
|
|
|
if (!strncmp(subsystem, "uio", 3)) {
|
|
if (!strncmp(action, "remove", 6)) {
|
|
event->action = SPDK_UEVENT_REMOVE;
|
|
} else if (!strncmp(action, "add", 3)) {
|
|
/* Support the ADD UEVENT for the device allow */
|
|
event->action = SPDK_UEVENT_ADD;
|
|
} else {
|
|
return 0;
|
|
}
|
|
|
|
tmp = strstr(dev_path, "/uio/");
|
|
if (!tmp) {
|
|
SPDK_ERRLOG("Invalid format of uevent: %s\n", dev_path);
|
|
return -EBADMSG;
|
|
}
|
|
memset(tmp, 0, SPDK_UEVENT_MSG_LEN - (tmp - dev_path));
|
|
|
|
pci_address = strrchr(dev_path, '/');
|
|
if (!pci_address) {
|
|
SPDK_ERRLOG("Not found PCI device BDF in uevent: %s\n", dev_path);
|
|
return -EBADMSG;
|
|
}
|
|
pci_address++;
|
|
|
|
rc = spdk_pci_addr_parse(&event->traddr, pci_address);
|
|
if (rc != 0) {
|
|
SPDK_ERRLOG("Invalid format for PCI device BDF: %s\n", pci_address);
|
|
return rc;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
if (!strncmp(driver, "vfio-pci", 8)) {
|
|
if (!strncmp(action, "bind", 4)) {
|
|
/* Support the ADD UEVENT for the device allow */
|
|
event->action = SPDK_UEVENT_ADD;
|
|
} else {
|
|
/* Only need to support add event.
|
|
* VFIO hotplug interface is "pci.c:pci_device_rte_dev_event".
|
|
* VFIO informs the userspace hotplug through vfio req notifier interrupt.
|
|
* The app needs to free the device userspace driver resource first then
|
|
* the OS remove the device VFIO driver and broadcast the VFIO uevent.
|
|
*/
|
|
return 0;
|
|
}
|
|
|
|
rc = spdk_pci_addr_parse(&event->traddr, vfio_pci_addr);
|
|
if (rc != 0) {
|
|
SPDK_ERRLOG("Invalid format for PCI device BDF: %s\n", vfio_pci_addr);
|
|
return rc;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
spdk_pci_get_event(int fd, struct spdk_pci_event *event)
|
|
{
|
|
int ret;
|
|
char buf[SPDK_UEVENT_MSG_LEN];
|
|
|
|
memset(buf, 0, SPDK_UEVENT_MSG_LEN);
|
|
memset(event, 0, sizeof(*event));
|
|
|
|
ret = recv(fd, buf, SPDK_UEVENT_MSG_LEN - 1, MSG_DONTWAIT);
|
|
if (ret > 0) {
|
|
return parse_subsystem_event(buf, event);
|
|
} else if (ret < 0) {
|
|
if (errno == EAGAIN || errno == EWOULDBLOCK) {
|
|
return 0;
|
|
} else {
|
|
ret = errno;
|
|
SPDK_ERRLOG("Socket read error %d\n", errno);
|
|
return -ret;
|
|
}
|
|
} else {
|
|
/* connection closed */
|
|
return -ENOTCONN;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
#else /* Not Linux */
|
|
|
|
int
|
|
spdk_pci_event_listen(void)
|
|
{
|
|
SPDK_ERRLOG("Non-Linux does not support this operation\n");
|
|
return -ENOTSUP;
|
|
}
|
|
|
|
int
|
|
spdk_pci_get_event(int fd, struct spdk_pci_event *event)
|
|
{
|
|
SPDK_ERRLOG("Non-Linux does not support this operation\n");
|
|
return -ENOTSUP;
|
|
}
|
|
#endif
|