aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorHuang Rui <vowstar@gmail.com>2020-07-25 10:16:41 +0800
committerHuang Rui <vowstar@gmail.com>2020-07-25 10:16:41 +0800
commit293c4b73dfa408aab7c1d72683b3bd8964fc4858 (patch)
treebbd2b3a7663a7b9787103036cd12f81ade8ca9e7 /sys-libs/libnvidia-container/files/libnvidia-container-1.1.1-add-sysfs-c.patch
parentsci-electronics/verilator: bump to 4.038 (diff)
downloadguru-293c4b73dfa408aab7c1d72683b3bd8964fc4858.tar.gz
guru-293c4b73dfa408aab7c1d72683b3bd8964fc4858.tar.bz2
guru-293c4b73dfa408aab7c1d72683b3bd8964fc4858.zip
profiles: add updates/3Q-2020 and move libnvidia-container
move app-emulation/libnvidia-container to sys-libs/libnvidia-container Package-Manager: Portage-3.0.0, Repoman-2.3.23 Signed-off-by: Huang Rui <vowstar@gmail.com>
Diffstat (limited to 'sys-libs/libnvidia-container/files/libnvidia-container-1.1.1-add-sysfs-c.patch')
-rw-r--r--sys-libs/libnvidia-container/files/libnvidia-container-1.1.1-add-sysfs-c.patch536
1 files changed, 536 insertions, 0 deletions
diff --git a/sys-libs/libnvidia-container/files/libnvidia-container-1.1.1-add-sysfs-c.patch b/sys-libs/libnvidia-container/files/libnvidia-container-1.1.1-add-sysfs-c.patch
new file mode 100644
index 000000000..5291dd9d5
--- /dev/null
+++ b/sys-libs/libnvidia-container/files/libnvidia-container-1.1.1-add-sysfs-c.patch
@@ -0,0 +1,536 @@
+diff --git a/src/pci-sysfs.c b/src/pci-sysfs.c
+new file mode 100644
+index 0000000..210bf40
+--- /dev/null
++++ b/src/pci-sysfs.c
+@@ -0,0 +1,529 @@
++/*
++ * (C) Copyright IBM Corporation 2006
++ *
++ * Copyright (c) 2014-2018 NVIDIA Corporation
++ *
++ * All Rights Reserved.
++ *
++ * Permission is hereby granted, free of charge, to any person obtaining a copy
++ * of this software and associated documentation files (the "Software"), to deal
++ * in the Software without restriction, including without limitation the rights
++ * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
++ * copies of the Software, and to permit persons to whom the Software is
++ * furnished to do so, subject to the following conditions:
++ *
++ * The above copyright notice and this permission notice shall be included in
++ * all copies or substantial portions of the Software.
++ *
++ * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
++ * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
++ * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
++ * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
++ * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
++ * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
++ * THE SOFTWARE.
++ */
++
++/**
++ * pcienum-sysfs.c
++ *
++ * Based on libpciaccess/src/linux_sysfs.c from libpciaccess-0.12.1, which was
++ * found here:
++ *
++ * http://cgit.freedesktop.org/xorg/lib/libpciaccess
++ *
++ * Access PCI subsystem using Linux's sysfs interface. This interface is
++ * available starting somewhere in the late 2.5.x kernel phase, and is the
++ * preferred method on all 2.6.x kernels.
++ *
++ * Original author: Ian Romanick <idr@us.ibm.com>
++ */
++
++#if defined(NV_LINUX)
++
++#include <stdlib.h>
++#include <string.h>
++#include <stdio.h>
++#include <unistd.h>
++#include <sys/types.h>
++#include <sys/stat.h>
++#include <fcntl.h>
++#include <sys/mman.h>
++#include <dirent.h>
++#include <errno.h>
++#include <sys/time.h>
++#include <time.h>
++#include <limits.h>
++
++#include "pci-enum.h"
++#include "pci-sysfs.h"
++
++#define SYS_BUS_PCI "/sys/bus/pci/"
++#define SYS_BUS_PCI_DEVICES SYS_BUS_PCI "devices"
++#define SYS_BUS_PCI_RESCAN SYS_BUS_PCI "rescan"
++#define PCI_DBDF_FORMAT "%04x:%02x:%02x.%1u"
++#define SYSFS_PCI_BRIDGE_RESCAN_FMT SYS_BUS_PCI_DEVICES "/" PCI_DBDF_FORMAT "/rescan"
++#define SYSFS_RESCAN_STRING "1\n"
++#define SYSFS_RESCAN_STRING_SIZE 2
++#define PCI_CAP_TTL_MAX 20
++#define SYSFS_PATH_SIZE 256
++
++#define BAIL_ON_IO_ERR(buf, err, cnt, action) \
++do { \
++ if (((err) != 0) || ((cnt) < sizeof(buf))) \
++ { \
++ (err) = ((err) == 0) ? EIO : (err); \
++ action; \
++ } \
++} while (0)
++
++static int pci_sysfs_read_cfg(uint16_t, uint16_t, uint16_t, uint16_t, uint16_t, void *,
++ uint16_t size, uint16_t *);
++
++static int find_matches(struct pci_id_match *match);
++
++/**
++ * Attempt to access PCI subsystem using Linux's sysfs interface to enumerate
++ * the matched devices.
++ */
++int
++pci_enum_match_id(struct pci_id_match *match)
++{
++ int err = 0;
++ struct stat st;
++
++
++ /*
++ * If the directory "/sys/bus/pci/devices" exists, then the PCI subsystem
++ * can be accessed using this interface.
++ */
++ match->num_matches = 0;
++ if (stat(SYS_BUS_PCI_DEVICES, &st) == 0)
++ {
++ err = find_matches(match);
++ }
++ else
++ {
++ err = errno;
++ }
++
++ return err;
++}
++
++
++/**
++ * The sysfs lookup method uses the directory entries in /sys/bus/pci/devices
++ * to enumerate all PCI devices, and then uses a file in each that is mapped to
++ * the device's PCI config space to extract the data to match against.
++ */
++static int
++find_matches(struct pci_id_match *match)
++{
++ struct dirent *d;
++ DIR *sysfs_pci_dir;
++ int err = 0;
++
++ sysfs_pci_dir = opendir(SYS_BUS_PCI_DEVICES);
++ if (sysfs_pci_dir == NULL)
++ {
++ return errno;
++ }
++
++ while ((d = readdir(sysfs_pci_dir)) != NULL)
++ {
++ uint8_t config[48];
++ uint16_t bytes;
++ unsigned dom, bus, dev, func;
++ uint16_t vendor_id, device_id, subvendor_id, subdevice_id;
++ uint16_t device_class;
++
++ /* Ignore the . and .. dirents */
++ if ((strcmp(d->d_name, ".") == 0) || (strcmp(d->d_name, "..") == 0))
++ {
++ continue;
++ }
++
++ sscanf(d->d_name, PCI_DBDF_FORMAT,
++ & dom, & bus, & dev, & func);
++
++ err = pci_sysfs_read_cfg(dom, bus, dev, func, 0, config, 48, & bytes);
++ if ((bytes == 48) && !err)
++ {
++ vendor_id = (uint16_t)config[0] + ((uint16_t)config[1] << 8);
++ device_id = (uint16_t)config[2] + ((uint16_t)config[3] << 8);
++ device_class = (uint16_t)config[10] +
++ ((uint16_t)config[11] << 8);
++ subvendor_id = (uint16_t)config[44] +
++ ((uint16_t)config[45] << 8);
++ subdevice_id = (uint16_t)config[46] +
++ ((uint16_t)config[47] << 8);
++
++ /*
++ * This logic, originally in common_iterator.c, will tell if
++ * this device is a match for the search criteria.
++ */
++ if (PCI_ID_COMPARE(match->vendor_id, vendor_id) &&
++ PCI_ID_COMPARE(match->device_id, device_id) &&
++ PCI_ID_COMPARE(match->subvendor_id, subvendor_id) &&
++ PCI_ID_COMPARE(match->subdevice_id, subdevice_id) &&
++ ((device_class & match->device_class_mask) ==
++ match->device_class))
++ {
++ match->num_matches++;
++ }
++ }
++
++ if (err)
++ {
++ break;
++ }
++ }
++
++ closedir(sysfs_pci_dir);
++ return err;
++}
++
++static int
++pci_sysfs_read_cfg(uint16_t domain, uint16_t bus, uint16_t device,
++ uint16_t function, uint16_t off, void *data,
++ uint16_t size, uint16_t *bytes_read)
++{
++ char name[SYSFS_PATH_SIZE];
++ uint16_t temp_size = size;
++ int err = 0;
++ int fd;
++ char *data_bytes = data;
++
++ if (bytes_read != NULL)
++ {
++ *bytes_read = 0;
++ }
++
++ /*
++ * Each device has a directory under sysfs. Within that directory there
++ * is a file named "config". This file used to access the PCI config
++ * space. It is used here to obtain most of the information about the
++ * device.
++ */
++ snprintf(name, SYSFS_PATH_SIZE - 1, "%s/" PCI_DBDF_FORMAT "/config",
++ SYS_BUS_PCI_DEVICES, domain, bus, device, function);
++
++ fd = open(name, O_RDONLY);
++ if (fd < 0)
++ {
++ return errno;
++ }
++
++ if (off != 0)
++ {
++ if (lseek(fd, (off_t) off, SEEK_SET) < 0)
++ {
++ close(fd);
++ return errno;
++ }
++ }
++
++ while (temp_size > 0)
++ {
++ const ssize_t bytes = read(fd, data_bytes, temp_size);
++
++ /*
++ * If zero bytes were read, then we assume it's the end of the
++ * config file.
++ */
++ if (bytes <= 0)
++ {
++ err = errno;
++ break;
++ }
++
++ temp_size -= bytes;
++ data_bytes += bytes;
++ }
++
++ if (bytes_read != NULL)
++ {
++ *bytes_read = size - temp_size;
++ }
++
++ close(fd);
++ return err;
++}
++
++static int
++pci_sysfs_write_cfg(uint16_t domain, uint16_t bus, uint16_t device,
++ uint16_t function, uint16_t off, void *data,
++ uint16_t size, uint16_t *bytes_written)
++{
++ char name[SYSFS_PATH_SIZE];
++ uint16_t temp_size = size;
++ int err = 0;
++ int fd;
++ char *data_bytes = data;
++
++ if (bytes_written != NULL)
++ {
++ *bytes_written = 0;
++ }
++
++ /*
++ * Each device has a directory under sysfs. Within that directory there
++ * is a file named "config". This file used to access the PCI config
++ * space.
++ */
++ snprintf(name, SYSFS_PATH_SIZE - 1, "%s/" PCI_DBDF_FORMAT "/config",
++ SYS_BUS_PCI_DEVICES, domain, bus, device, function);
++
++ fd = open(name, O_WRONLY);
++ if (fd < 0)
++ {
++ return errno;
++ }
++
++ if (off != 0)
++ {
++ if (lseek(fd, (off_t) off, SEEK_SET) < 0)
++ {
++ close(fd);
++ return errno;
++ }
++ }
++
++ while (temp_size > 0)
++ {
++ const ssize_t bytes = write(fd, data_bytes, temp_size);
++
++ if (bytes < 0)
++ {
++ err = errno;
++ break;
++ }
++ /*
++ * If zero bytes were written, then we assume it's the end of the
++ * config file.
++ */
++ if (bytes == 0)
++ {
++ break;
++ }
++
++ temp_size -= bytes;
++ data_bytes += bytes;
++ }
++
++ if (bytes_written != NULL)
++ {
++ *bytes_written = size - temp_size;
++ }
++
++ close(fd);
++ return err;
++}
++
++int
++pci_rescan(uint16_t domain, uint8_t bus, uint8_t slot, uint8_t function)
++{
++ char const *node;
++ char node_buf[SYSFS_PATH_SIZE];
++ int node_fd;
++ ssize_t cnt;
++
++ if ((domain | bus | slot | function) == 0)
++ {
++ /* rescan the entire PCI tree */
++ node = SYS_BUS_PCI_RESCAN;
++ }
++ else
++ {
++ snprintf(node_buf, sizeof(node_buf) - 1, SYSFS_PCI_BRIDGE_RESCAN_FMT,
++ domain, bus, slot, function);
++ node = node_buf;
++ }
++
++ node_fd = open(node, O_WRONLY);
++
++ if (node_fd < 0)
++ {
++ return errno;
++ }
++
++ cnt = write(node_fd, SYSFS_RESCAN_STRING, SYSFS_RESCAN_STRING_SIZE);
++
++ close(node_fd);
++
++ return cnt == SYSFS_RESCAN_STRING_SIZE ? 0 : EIO;
++}
++
++int
++pci_find_parent_bridge(pci_info_t *p_gpu_info, pci_info_t *p_bridge_info)
++{
++ char gpu_path[SYSFS_PATH_SIZE];
++ char bridge_path[PATH_MAX];
++ char *p_node;
++
++ snprintf(gpu_path, SYSFS_PATH_SIZE - 1, "%s/" PCI_DBDF_FORMAT "/..", SYS_BUS_PCI_DEVICES,
++ p_gpu_info->domain, p_gpu_info->bus,
++ p_gpu_info->dev, p_gpu_info->ftn);
++
++ if (realpath(gpu_path, bridge_path) == NULL)
++ {
++ return errno;
++ }
++
++ p_node = strrchr(bridge_path, '/');
++
++ if (p_node == NULL)
++ {
++ return ENOENT;
++ }
++
++ ++p_node;
++
++ if (sscanf(p_node, PCI_DBDF_FORMAT,
++ &p_bridge_info->domain, &p_bridge_info->bus,
++ &p_bridge_info->dev, &p_bridge_info->ftn) != 4)
++ {
++ return ENOENT;
++ }
++
++ return 0;
++}
++
++static int
++pci_find_pcie_caps(uint16_t domain, uint8_t bus, uint8_t device, uint8_t ftn, uint8_t *p_caps)
++{
++ unsigned ttl;
++ uint8_t off;
++ uint8_t cap_id;
++ int err = ENXIO;
++ uint16_t cnt;
++
++ for (off = PCI_CAPABILITY_LIST, ttl = PCI_CAP_TTL_MAX; ttl; --ttl)
++ {
++ err = pci_sysfs_read_cfg(domain, bus, device, ftn, off,
++ &off, sizeof(off), &cnt);
++ BAIL_ON_IO_ERR(off, err, cnt, break);
++
++ /* Capabilities must reside above the std config header */
++ if ((off < PCI_STD_HEADER_SIZEOF) || (off == 0xff))
++ {
++ break;
++ }
++
++ /* Clear the reserved bits */
++ off &= ~3;
++
++ err = pci_sysfs_read_cfg(domain, bus, device, ftn, off + PCI_CAP_LIST_ID,
++ &cap_id, sizeof(cap_id), &cnt);
++ BAIL_ON_IO_ERR(cap_id, err, cnt, break);
++
++ if (cap_id == PCI_CAP_ID_EXP)
++ {
++ goto found;
++ }
++
++ if (cap_id == 0xff)
++ {
++ break;
++ }
++
++ off += PCI_CAP_LIST_NEXT;
++ }
++ return err;
++found:
++ *p_caps = off;
++ return 0;
++}
++
++int
++pci_bridge_link_set_enable(uint16_t domain, uint8_t bus, uint8_t device, uint8_t ftn, int enable)
++{
++ uint8_t pcie_caps = 0;
++ uint16_t reg;
++ uint32_t cap_reg;
++ uint16_t cnt;
++ int err;
++ struct timeval start;
++ struct timeval curr;
++ struct timeval diff;
++ struct timespec delay = {0, PCI_LINK_DELAY_NS};
++ struct timespec dlllar_disable_delay = {0, PCI_LINK_DLLLAR_DISABLE_DELAY_NS};
++
++ err = pci_find_pcie_caps(domain, bus, device, ftn, &pcie_caps);
++
++ if (err != 0)
++ {
++ return err;
++ }
++
++ err = pci_sysfs_read_cfg(domain, bus, device, ftn, pcie_caps + PCI_EXP_LNKCTL,
++ &reg, sizeof(reg), &cnt);
++ BAIL_ON_IO_ERR(reg, err, cnt, return err);
++
++ if (enable)
++ {
++ reg &= ~PCI_EXP_LNKCTL_LD;
++ }
++ else
++ {
++ reg |= PCI_EXP_LNKCTL_LD;
++ }
++
++ err = pci_sysfs_write_cfg(domain, bus, device, ftn, pcie_caps + PCI_EXP_LNKCTL,
++ &reg, sizeof(reg), &cnt);
++ BAIL_ON_IO_ERR(reg, err, cnt, return err);
++
++ if (enable)
++ {
++ /*
++ * Data Link Layer Link Active Reporting must be capable for
++ * zero power capable downstream port. But old controller might
++ * not implement it. In this case, we wait for 30 ms.
++ */
++ err = pci_sysfs_read_cfg(domain, bus, device, ftn, pcie_caps + PCI_EXP_LNKCAP,
++ &cap_reg, sizeof(cap_reg), &cnt);
++ BAIL_ON_IO_ERR(cap_reg, err, cnt, return err);
++
++ if (cap_reg & PCI_EXP_LNKCAP_DLLLARC)
++ {
++ /* wait for the link to go up and then sleep for 100 ms */
++
++ gettimeofday(&start, NULL);
++
++ for (;;)
++ {
++ err = pci_sysfs_read_cfg(domain, bus, device, ftn, pcie_caps + PCI_EXP_LNKSTA,
++ &reg, sizeof(reg), &cnt);
++ BAIL_ON_IO_ERR(reg, err, cnt, return err);
++
++ if ((reg & PCI_EXP_LNKSTA_DLLLA) != 0)
++ {
++ break;
++ }
++
++ gettimeofday(&curr, NULL);
++ timersub(&curr, &start, &diff);
++
++ if ((diff.tv_sec > 0) || (diff.tv_usec >= PCI_LINK_WAIT_US))
++ {
++ return ETIME;
++ }
++ }
++ }
++ else
++ {
++ /*
++ * Measured the time on DGX1 for link to become established in a bridge,
++ * where the DLLLA reporting is supported and its approximately ~9ms,
++ * so wait for 30ms where DLLLA reporting is not supported.
++ */
++ PCI_NANOSLEEP(&dlllar_disable_delay, NULL);
++ }
++
++ PCI_NANOSLEEP(&delay, NULL);
++ }
++
++ return err;
++}
++
++#endif /* defined(NV_LINUX) */
+\ No newline at end of file