mirror of
https://github.com/NVIDIA/open-gpu-kernel-modules.git
synced 2025-01-06 01:46:05 +01:00
3317 lines
122 KiB
C
3317 lines
122 KiB
C
/*******************************************************************************
|
|
Copyright (c) 2015-2022 NVIDIA Corporation
|
|
|
|
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.
|
|
|
|
*******************************************************************************/
|
|
|
|
#include "nv_uvm_interface.h"
|
|
#include "uvm_api.h"
|
|
#include "uvm_channel.h"
|
|
#include "uvm_global.h"
|
|
#include "uvm_gpu.h"
|
|
#include "uvm_gpu_semaphore.h"
|
|
#include "uvm_hal.h"
|
|
#include "uvm_procfs.h"
|
|
#include "uvm_pmm_gpu.h"
|
|
#include "uvm_pmm_sysmem.h"
|
|
#include "uvm_va_space.h"
|
|
#include "uvm_user_channel.h"
|
|
#include "uvm_perf_events.h"
|
|
#include "uvm_perf_heuristics.h"
|
|
#include "uvm_common.h"
|
|
#include "ctrl2080mc.h"
|
|
#include "nv-kthread-q.h"
|
|
#include "uvm_gpu_access_counters.h"
|
|
#include "uvm_ats.h"
|
|
#include "uvm_test.h"
|
|
|
|
|
|
|
|
|
|
#include "uvm_linux.h"
|
|
|
|
#define UVM_PROC_GPUS_PEER_DIR_NAME "peers"
|
|
|
|
// The uvm_peer_copy module parameter enables to choose from "phys" or "virt".
|
|
// It determines the addressing mode for P2P copies.
|
|
#define UVM_PARAM_PEER_COPY_VIRTUAL "virt"
|
|
#define UVM_PARAM_PEER_COPY_PHYSICAL "phys"
|
|
static char *uvm_peer_copy = UVM_PARAM_PEER_COPY_PHYSICAL;
|
|
module_param(uvm_peer_copy, charp, S_IRUGO);
|
|
MODULE_PARM_DESC(uvm_peer_copy, "Choose the addressing mode for peer copying, options: "
|
|
UVM_PARAM_PEER_COPY_PHYSICAL " [default] or " UVM_PARAM_PEER_COPY_VIRTUAL ". "
|
|
"Valid for Ampere+ GPUs.");
|
|
|
|
static void remove_gpu(uvm_gpu_t *gpu);
|
|
static void disable_peer_access(uvm_gpu_t *gpu0, uvm_gpu_t *gpu1);
|
|
static NV_STATUS discover_nvlink_peers(uvm_gpu_t *gpu);
|
|
static void destroy_nvlink_peers(uvm_gpu_t *gpu);
|
|
|
|
static uvm_user_channel_t *get_user_channel(uvm_rb_tree_node_t *node)
|
|
{
|
|
return container_of(node, uvm_user_channel_t, instance_ptr.node);
|
|
}
|
|
|
|
static void fill_gpu_info(uvm_parent_gpu_t *parent_gpu, const UvmGpuInfo *gpu_info)
|
|
{
|
|
char uuid_buffer[UVM_GPU_UUID_TEXT_BUFFER_LENGTH];
|
|
|
|
parent_gpu->rm_info = *gpu_info;
|
|
|
|
format_uuid_to_buffer(uuid_buffer, sizeof(uuid_buffer), &parent_gpu->uuid);
|
|
snprintf(parent_gpu->name,
|
|
sizeof(parent_gpu->name),
|
|
"ID %u: %s: %s",
|
|
uvm_id_value(parent_gpu->id),
|
|
parent_gpu->rm_info.name,
|
|
uuid_buffer);
|
|
}
|
|
|
|
static uvm_gpu_link_type_t get_gpu_link_type(UVM_LINK_TYPE link_type)
|
|
{
|
|
switch (link_type) {
|
|
case UVM_LINK_TYPE_PCIE:
|
|
return UVM_GPU_LINK_PCIE;
|
|
case UVM_LINK_TYPE_NVLINK_1:
|
|
return UVM_GPU_LINK_NVLINK_1;
|
|
case UVM_LINK_TYPE_NVLINK_2:
|
|
return UVM_GPU_LINK_NVLINK_2;
|
|
case UVM_LINK_TYPE_NVLINK_3:
|
|
return UVM_GPU_LINK_NVLINK_3;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
default:
|
|
return UVM_GPU_LINK_INVALID;
|
|
}
|
|
}
|
|
|
|
static NV_STATUS get_gpu_caps(uvm_parent_gpu_t *parent_gpu)
|
|
{
|
|
NV_STATUS status;
|
|
UvmGpuCaps gpu_caps;
|
|
|
|
memset(&gpu_caps, 0, sizeof(gpu_caps));
|
|
|
|
status = uvm_rm_locked_call(nvUvmInterfaceQueryCaps(parent_gpu->rm_device, &gpu_caps));
|
|
if (status != NV_OK)
|
|
return status;
|
|
|
|
parent_gpu->sysmem_link = get_gpu_link_type(gpu_caps.sysmemLink);
|
|
UVM_ASSERT(parent_gpu->sysmem_link != UVM_GPU_LINK_INVALID);
|
|
|
|
parent_gpu->sysmem_link_rate_mbyte_per_s = gpu_caps.sysmemLinkRateMBps;
|
|
parent_gpu->nvswitch_info.is_nvswitch_connected = gpu_caps.connectedToSwitch;
|
|
|
|
// nvswitch is routed via physical pages, where the upper 13-bits of the
|
|
// 47-bit address space holds the routing information for each peer.
|
|
// Currently, this is limited to a 16GB framebuffer window size.
|
|
if (parent_gpu->nvswitch_info.is_nvswitch_connected)
|
|
parent_gpu->nvswitch_info.fabric_memory_window_start = gpu_caps.nvswitchMemoryWindowStart;
|
|
|
|
if (gpu_caps.numaEnabled) {
|
|
parent_gpu->numa_info.enabled = true;
|
|
parent_gpu->numa_info.node_id = gpu_caps.numaNodeId;
|
|
parent_gpu->numa_info.system_memory_window_start = gpu_caps.systemMemoryWindowStart;
|
|
parent_gpu->numa_info.system_memory_window_end = gpu_caps.systemMemoryWindowStart +
|
|
gpu_caps.systemMemoryWindowSize -
|
|
1;
|
|
}
|
|
else {
|
|
UVM_ASSERT(!g_uvm_global.ats.enabled);
|
|
}
|
|
|
|
return NV_OK;
|
|
}
|
|
|
|
static NV_STATUS alloc_and_init_address_space(uvm_gpu_t *gpu)
|
|
{
|
|
NV_STATUS status;
|
|
UvmGpuAddressSpaceInfo gpu_address_space_info = {0};
|
|
|
|
status = uvm_rm_locked_call(nvUvmInterfaceAddressSpaceCreate(uvm_gpu_device_handle(gpu),
|
|
gpu->parent->rm_va_base,
|
|
gpu->parent->rm_va_size,
|
|
&gpu->rm_address_space,
|
|
&gpu_address_space_info));
|
|
if (status != NV_OK)
|
|
return status;
|
|
|
|
gpu->big_page.internal_size = gpu_address_space_info.bigPageSize;
|
|
|
|
gpu->time.time0_register = gpu_address_space_info.time0Offset;
|
|
gpu->time.time1_register = gpu_address_space_info.time1Offset;
|
|
|
|
gpu->max_subcontexts = gpu_address_space_info.maxSubctxCount;
|
|
|
|
return NV_OK;
|
|
}
|
|
|
|
static NV_STATUS get_gpu_fb_info(uvm_gpu_t *gpu)
|
|
{
|
|
NV_STATUS status;
|
|
UvmGpuFbInfo fb_info = {0};
|
|
|
|
status = uvm_rm_locked_call(nvUvmInterfaceGetFbInfo(uvm_gpu_device_handle(gpu), &fb_info));
|
|
if (status != NV_OK)
|
|
return status;
|
|
|
|
if (!fb_info.bZeroFb) {
|
|
gpu->mem_info.size = ((NvU64)fb_info.heapSize + fb_info.reservedHeapSize) * 1024;
|
|
gpu->mem_info.max_allocatable_address = fb_info.maxAllocatableAddress;
|
|
}
|
|
|
|
return NV_OK;
|
|
}
|
|
|
|
static NV_STATUS get_gpu_ecc_info(uvm_gpu_t *gpu)
|
|
{
|
|
NV_STATUS status;
|
|
UvmGpuEccInfo ecc_info = {0};
|
|
|
|
status = uvm_rm_locked_call(nvUvmInterfaceGetEccInfo(uvm_gpu_device_handle(gpu), &ecc_info));
|
|
if (status != NV_OK)
|
|
return status;
|
|
|
|
gpu->ecc.enabled = ecc_info.bEccEnabled;
|
|
if (gpu->ecc.enabled) {
|
|
gpu->ecc.hw_interrupt_tree_location = (volatile NvU32*)((char*)ecc_info.eccReadLocation + ecc_info.eccOffset);
|
|
UVM_ASSERT(gpu->ecc.hw_interrupt_tree_location != NULL);
|
|
|
|
gpu->ecc.mask = ecc_info.eccMask;
|
|
UVM_ASSERT(gpu->ecc.mask != 0);
|
|
|
|
gpu->ecc.error_notifier = ecc_info.eccErrorNotifier;
|
|
UVM_ASSERT(gpu->ecc.error_notifier != NULL);
|
|
}
|
|
|
|
return NV_OK;
|
|
}
|
|
|
|
static bool gpu_supports_uvm(uvm_parent_gpu_t *parent_gpu)
|
|
{
|
|
// TODO: Bug 1757136: Add Linux SLI support. Until then, explicitly disable
|
|
// UVM on SLI.
|
|
return parent_gpu->rm_info.subdeviceCount == 1;
|
|
}
|
|
|
|
static bool parent_gpu_uses_canonical_form_address(uvm_parent_gpu_t *parent_gpu)
|
|
{
|
|
NvU64 gpu_addr_shift;
|
|
NvU64 cpu_addr_shift;
|
|
|
|
// PPC64LE doesn't use canonical form addresses.
|
|
if (NVCPU_IS_PPC64LE)
|
|
return false;
|
|
|
|
// We use big_page_size as UVM_PAGE_SIZE_64K because num_va_bits() is
|
|
// big_page_size invariant in the MMU HAL.
|
|
UVM_ASSERT(!parent_gpu->arch_hal->mmu_mode_hal(UVM_PAGE_SIZE_128K) ||
|
|
(parent_gpu->arch_hal->mmu_mode_hal(UVM_PAGE_SIZE_64K)->num_va_bits() ==
|
|
parent_gpu->arch_hal->mmu_mode_hal(UVM_PAGE_SIZE_128K)->num_va_bits()));
|
|
|
|
gpu_addr_shift = parent_gpu->arch_hal->mmu_mode_hal(UVM_PAGE_SIZE_64K)->num_va_bits();
|
|
cpu_addr_shift = fls64(TASK_SIZE - 1) + 1;
|
|
|
|
// Refer to the comments and diagram in uvm_gpu.c:uvm_gpu_can_address().
|
|
return gpu_addr_shift >= cpu_addr_shift;
|
|
|
|
}
|
|
|
|
bool uvm_gpu_can_address(uvm_gpu_t *gpu, NvU64 addr, NvU64 size)
|
|
{
|
|
// Lower and upper address spaces are typically found in platforms that use
|
|
// the canonical address form.
|
|
NvU64 max_va_lower;
|
|
NvU64 addr_end = addr + size - 1;
|
|
|
|
// Watch out for calling this too early in init
|
|
UVM_ASSERT(gpu->address_space_tree.hal);
|
|
UVM_ASSERT(gpu->address_space_tree.hal->num_va_bits() < 64);
|
|
UVM_ASSERT(addr <= addr_end);
|
|
UVM_ASSERT(size > 0);
|
|
|
|
// Pascal+ GPUs are capable of accessing kernel pointers in various modes
|
|
// by applying the same upper-bit checks that x86, ARM, and Power
|
|
// processors do. x86 and ARM use canonical form addresses. For ARM, even
|
|
// with Top-Byte Ignore enabled, the following logic validates addresses
|
|
// from the kernel VA range. PowerPC does not use canonical form address.
|
|
// The following diagram illustrates the valid (V) VA regions that can be
|
|
// mapped (or addressed) by the GPU/CPU when the CPU uses canonical form.
|
|
// (C) regions are only accessible by the CPU. Similarly, (G) regions
|
|
// are only accessible by the GPU. (X) regions are not addressible.
|
|
//
|
|
// GPU MAX VA < CPU MAX VA GPU MAX VA >= CPU MAX VA
|
|
// 0xF..F +----------------+ 0xF..F +----------------+
|
|
// |CCCCCCCCCCCCCCCC| |VVVVVVVVVVVVVVVV|
|
|
// |CCCCCCCCCCCCCCCC| |VVVVVVVVVVVVVVVV|
|
|
// |CCCCCCCCCCCCCCCC| |VVVVVVVVVVVVVVVV|
|
|
// |CCCCCCCCCCCCCCCC| CPU MIN UPPER VA|----------------|
|
|
// |CCCCCCCCCCCCCCCC| |GGGGGGGGGGGGGGGG|
|
|
// |CCCCCCCCCCCCCCCC| |GGGGGGGGGGGGGGGG|
|
|
// CPU MIN UPPER VA|----------------| GPU MIN UPPER VA|----------------|
|
|
// |XXXXXXXXXXXXXXXX| |XXXXXXXXXXXXXXXX|
|
|
// |XXXXXXXXXXXXXXXX| |XXXXXXXXXXXXXXXX|
|
|
// CPU MAX LOWER VA|----------------| GPU MAX LOWER VA|----------------|
|
|
// |CCCCCCCCCCCCCCCC| |GGGGGGGGGGGGGGGG|
|
|
// |CCCCCCCCCCCCCCCC| |GGGGGGGGGGGGGGGG|
|
|
// GPU MAX VA|----------------| CPU MAX LOWER VA|----------------|
|
|
// |VVVVVVVVVVVVVVVV| |VVVVVVVVVVVVVVVV|
|
|
// |VVVVVVVVVVVVVVVV| |VVVVVVVVVVVVVVVV|
|
|
// |VVVVVVVVVVVVVVVV| |VVVVVVVVVVVVVVVV|
|
|
// 0 +----------------+ 0 +----------------+
|
|
|
|
if (parent_gpu_uses_canonical_form_address(gpu->parent)) {
|
|
NvU64 min_va_upper = (NvU64)((NvS64)(1ULL << 63) >> (64 - gpu->address_space_tree.hal->num_va_bits()));
|
|
max_va_lower = 1ULL << (gpu->address_space_tree.hal->num_va_bits() - 1);
|
|
return (addr_end < max_va_lower) || (addr >= min_va_upper);
|
|
}
|
|
else {
|
|
max_va_lower = 1ULL << gpu->address_space_tree.hal->num_va_bits();
|
|
return addr_end < max_va_lower;
|
|
}
|
|
}
|
|
|
|
NvU64 uvm_parent_gpu_canonical_address(uvm_parent_gpu_t *parent_gpu, NvU64 addr)
|
|
{
|
|
NvU32 gpu_va_bits;
|
|
NvU32 shift;
|
|
|
|
if (parent_gpu_uses_canonical_form_address(parent_gpu)) {
|
|
gpu_va_bits = parent_gpu->arch_hal->mmu_mode_hal(UVM_PAGE_SIZE_64K)->num_va_bits();
|
|
shift = 64 - gpu_va_bits;
|
|
addr = (NvU64)((NvS64)(addr << shift) >> shift);
|
|
}
|
|
|
|
return addr;
|
|
}
|
|
|
|
static void gpu_info_print_ce_caps(uvm_gpu_t *gpu, struct seq_file *s)
|
|
{
|
|
NvU32 i;
|
|
UvmGpuCopyEnginesCaps ces_caps;
|
|
NV_STATUS status;
|
|
|
|
memset(&ces_caps, 0, sizeof(ces_caps));
|
|
status = uvm_rm_locked_call(nvUvmInterfaceQueryCopyEnginesCaps(uvm_gpu_device_handle(gpu), &ces_caps));
|
|
|
|
if (status != NV_OK) {
|
|
UVM_SEQ_OR_DBG_PRINT(s, "supported_ces: unavailable (query failed)\n");
|
|
return;
|
|
}
|
|
|
|
UVM_SEQ_OR_DBG_PRINT(s, "supported_ces:\n");
|
|
for (i = 0; i < UVM_COPY_ENGINE_COUNT_MAX; ++i) {
|
|
UvmGpuCopyEngineCaps *ce_caps = ces_caps.copyEngineCaps + i;
|
|
|
|
if (!ce_caps->supported)
|
|
continue;
|
|
|
|
UVM_SEQ_OR_DBG_PRINT(s, " ce %u pce mask 0x%08x grce %u shared %u sysmem read %u sysmem write %u sysmem %u nvlink p2p %u "
|
|
"p2p %u\n",
|
|
i,
|
|
ce_caps->cePceMask,
|
|
ce_caps->grce,
|
|
ce_caps->shared,
|
|
ce_caps->sysmemRead,
|
|
ce_caps->sysmemWrite,
|
|
ce_caps->sysmem,
|
|
ce_caps->nvlinkP2p,
|
|
ce_caps->p2p);
|
|
}
|
|
}
|
|
|
|
static const char *uvm_gpu_virt_type_string(UVM_VIRT_MODE virtMode)
|
|
{
|
|
BUILD_BUG_ON(UVM_VIRT_MODE_COUNT != 4);
|
|
|
|
switch (virtMode) {
|
|
UVM_ENUM_STRING_CASE(UVM_VIRT_MODE_NONE);
|
|
UVM_ENUM_STRING_CASE(UVM_VIRT_MODE_LEGACY);
|
|
UVM_ENUM_STRING_CASE(UVM_VIRT_MODE_SRIOV_HEAVY);
|
|
UVM_ENUM_STRING_CASE(UVM_VIRT_MODE_SRIOV_STANDARD);
|
|
UVM_ENUM_STRING_DEFAULT();
|
|
}
|
|
}
|
|
|
|
static const char *uvm_gpu_link_type_string(uvm_gpu_link_type_t link_type)
|
|
{
|
|
|
|
|
|
|
|
BUILD_BUG_ON(UVM_GPU_LINK_MAX != 5);
|
|
|
|
|
|
switch (link_type) {
|
|
UVM_ENUM_STRING_CASE(UVM_GPU_LINK_INVALID);
|
|
UVM_ENUM_STRING_CASE(UVM_GPU_LINK_PCIE);
|
|
UVM_ENUM_STRING_CASE(UVM_GPU_LINK_NVLINK_1);
|
|
UVM_ENUM_STRING_CASE(UVM_GPU_LINK_NVLINK_2);
|
|
UVM_ENUM_STRING_CASE(UVM_GPU_LINK_NVLINK_3);
|
|
|
|
|
|
|
|
|
|
UVM_ENUM_STRING_DEFAULT();
|
|
}
|
|
}
|
|
|
|
static void gpu_info_print_common(uvm_gpu_t *gpu, struct seq_file *s)
|
|
{
|
|
const UvmGpuInfo *gpu_info = &gpu->parent->rm_info;
|
|
uvm_numa_info_t *numa_info = &gpu->parent->numa_info;
|
|
NvU64 num_pages_in;
|
|
NvU64 num_pages_out;
|
|
NvU64 mapped_cpu_pages_size;
|
|
NvU32 get, put;
|
|
unsigned int cpu;
|
|
|
|
UVM_SEQ_OR_DBG_PRINT(s, "GPU %s\n", uvm_gpu_name(gpu));
|
|
UVM_SEQ_OR_DBG_PRINT(s, "retained_count %llu\n", uvm_gpu_retained_count(gpu));
|
|
UVM_SEQ_OR_DBG_PRINT(s, "ecc %s\n", gpu->ecc.enabled ? "enabled" : "disabled");
|
|
if (gpu->parent->closest_cpu_numa_node == -1)
|
|
UVM_SEQ_OR_DBG_PRINT(s, "closest_cpu_numa_node n/a\n");
|
|
else
|
|
UVM_SEQ_OR_DBG_PRINT(s, "closest_cpu_numa_node %d\n", gpu->parent->closest_cpu_numa_node);
|
|
|
|
if (!uvm_procfs_is_debug_enabled())
|
|
return;
|
|
|
|
UVM_SEQ_OR_DBG_PRINT(s, "CPU link type %s\n",
|
|
uvm_gpu_link_type_string(gpu->parent->sysmem_link));
|
|
UVM_SEQ_OR_DBG_PRINT(s, "CPU link bandwidth %uMBps\n",
|
|
gpu->parent->sysmem_link_rate_mbyte_per_s);
|
|
|
|
UVM_SEQ_OR_DBG_PRINT(s, "architecture 0x%X\n", gpu_info->gpuArch);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "implementation 0x%X\n", gpu_info->gpuImplementation);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "gpcs %u\n", gpu_info->gpcCount);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "max_gpcs %u\n", gpu_info->maxGpcCount);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "tpcs %u\n", gpu_info->tpcCount);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "max_tpcs_per_gpc %u\n", gpu_info->maxTpcPerGpcCount);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "host_class 0x%X\n", gpu_info->hostClass);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "ce_class 0x%X\n", gpu_info->ceClass);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "virtualization_mode %s\n",
|
|
uvm_gpu_virt_type_string(gpu_info->virtMode));
|
|
UVM_SEQ_OR_DBG_PRINT(s, "big_page_size %u\n", gpu->big_page.internal_size);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "rm_va_base 0x%llx\n", gpu->parent->rm_va_base);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "rm_va_size 0x%llx\n", gpu->parent->rm_va_size);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "vidmem_size %llu (%llu MBs)\n",
|
|
gpu->mem_info.size,
|
|
gpu->mem_info.size / (1024 * 1024));
|
|
UVM_SEQ_OR_DBG_PRINT(s, "vidmem_max_allocatable 0x%llx (%llu MBs)\n",
|
|
gpu->mem_info.max_allocatable_address,
|
|
gpu->mem_info.max_allocatable_address / (1024 * 1024));
|
|
|
|
if (numa_info->enabled) {
|
|
NvU64 window_size = numa_info->system_memory_window_end - numa_info->system_memory_window_start + 1;
|
|
UVM_SEQ_OR_DBG_PRINT(s, "numa_node_id %u\n", numa_info->node_id);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "system_memory_window_start 0x%llx\n",
|
|
numa_info->system_memory_window_start);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "system_memory_window_end 0x%llx\n",
|
|
numa_info->system_memory_window_end);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "system_memory_window_size 0x%llx (%llu MBs)\n",
|
|
window_size,
|
|
window_size / (1024 * 1024));
|
|
}
|
|
|
|
if (gpu->parent->npu)
|
|
UVM_SEQ_OR_DBG_PRINT(s, "npu_domain %d\n", gpu->parent->npu->pci_domain);
|
|
|
|
UVM_SEQ_OR_DBG_PRINT(s, "interrupts %llu\n", gpu->parent->isr.interrupt_count);
|
|
|
|
if (gpu->parent->isr.replayable_faults.handling) {
|
|
UVM_SEQ_OR_DBG_PRINT(s, "replayable_faults_bh %llu\n",
|
|
gpu->parent->isr.replayable_faults.stats.bottom_half_count);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "replayable_faults_bh/cpu\n");
|
|
for_each_cpu(cpu, &gpu->parent->isr.replayable_faults.stats.cpus_used_mask) {
|
|
UVM_SEQ_OR_DBG_PRINT(s, " cpu%02u %llu\n",
|
|
cpu,
|
|
gpu->parent->isr.replayable_faults.stats.cpu_exec_count[cpu]);
|
|
}
|
|
UVM_SEQ_OR_DBG_PRINT(s, "replayable_faults_buffer_entries %u\n",
|
|
gpu->parent->fault_buffer_info.replayable.max_faults);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "replayable_faults_cached_get %u\n",
|
|
gpu->parent->fault_buffer_info.replayable.cached_get);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "replayable_faults_cached_put %u\n",
|
|
gpu->parent->fault_buffer_info.replayable.cached_put);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "replayable_faults_get %u\n",
|
|
gpu->parent->fault_buffer_hal->read_get(gpu->parent));
|
|
UVM_SEQ_OR_DBG_PRINT(s, "replayable_faults_put %u\n",
|
|
gpu->parent->fault_buffer_hal->read_put(gpu->parent));
|
|
UVM_SEQ_OR_DBG_PRINT(s, "replayable_faults_fault_batch_size %u\n",
|
|
gpu->parent->fault_buffer_info.max_batch_size);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "replayable_faults_replay_policy %s\n",
|
|
uvm_perf_fault_replay_policy_string(gpu->parent->fault_buffer_info.replayable.replay_policy));
|
|
UVM_SEQ_OR_DBG_PRINT(s, "replayable_faults_num_faults %llu\n",
|
|
gpu->parent->stats.num_replayable_faults);
|
|
}
|
|
if (gpu->parent->isr.non_replayable_faults.handling) {
|
|
UVM_SEQ_OR_DBG_PRINT(s, "non_replayable_faults_bh %llu\n",
|
|
gpu->parent->isr.non_replayable_faults.stats.bottom_half_count);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "non_replayable_faults_bh/cpu\n");
|
|
for_each_cpu(cpu, &gpu->parent->isr.non_replayable_faults.stats.cpus_used_mask) {
|
|
UVM_SEQ_OR_DBG_PRINT(s, " cpu%02u %llu\n",
|
|
cpu,
|
|
gpu->parent->isr.non_replayable_faults.stats.cpu_exec_count[cpu]);
|
|
}
|
|
UVM_SEQ_OR_DBG_PRINT(s, "non_replayable_faults_buffer_entries %u\n",
|
|
gpu->parent->fault_buffer_info.non_replayable.max_faults);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "non_replayable_faults_num_faults %llu\n",
|
|
gpu->parent->stats.num_non_replayable_faults);
|
|
}
|
|
|
|
if (gpu->parent->isr.access_counters.handling_ref_count > 0) {
|
|
UVM_SEQ_OR_DBG_PRINT(s, "access_counters_bh %llu\n",
|
|
gpu->parent->isr.access_counters.stats.bottom_half_count);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "access_counters_bh/cpu\n");
|
|
for_each_cpu(cpu, &gpu->parent->isr.access_counters.stats.cpus_used_mask) {
|
|
UVM_SEQ_OR_DBG_PRINT(s, " cpu%02u %llu\n",
|
|
cpu,
|
|
gpu->parent->isr.access_counters.stats.cpu_exec_count[cpu]);
|
|
}
|
|
UVM_SEQ_OR_DBG_PRINT(s, "access_counters_buffer_entries %u\n",
|
|
gpu->parent->access_counter_buffer_info.max_notifications);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "access_counters_cached_get %u\n",
|
|
gpu->parent->access_counter_buffer_info.cached_get);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "access_counters_cached_put %u\n",
|
|
gpu->parent->access_counter_buffer_info.cached_put);
|
|
|
|
get = UVM_GPU_READ_ONCE(*gpu->parent->access_counter_buffer_info.rm_info.pAccessCntrBufferGet);
|
|
put = UVM_GPU_READ_ONCE(*gpu->parent->access_counter_buffer_info.rm_info.pAccessCntrBufferPut);
|
|
|
|
UVM_SEQ_OR_DBG_PRINT(s, "access_counters_get %u\n", get);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "access_counters_put %u\n", put);
|
|
}
|
|
|
|
num_pages_out = atomic64_read(&gpu->parent->stats.num_pages_out);
|
|
num_pages_in = atomic64_read(&gpu->parent->stats.num_pages_in);
|
|
mapped_cpu_pages_size = atomic64_read(&gpu->parent->mapped_cpu_pages_size);
|
|
|
|
UVM_SEQ_OR_DBG_PRINT(s, "migrated_pages_in %llu (%llu MB)\n",
|
|
num_pages_in,
|
|
(num_pages_in * (NvU64)PAGE_SIZE) / (1024u * 1024u));
|
|
UVM_SEQ_OR_DBG_PRINT(s, "migrated_pages_out %llu (%llu MB)\n",
|
|
num_pages_out,
|
|
(num_pages_out * (NvU64)PAGE_SIZE) / (1024u * 1024u));
|
|
UVM_SEQ_OR_DBG_PRINT(s, "mapped_cpu_pages_dma %llu (%llu MB)\n",
|
|
mapped_cpu_pages_size / PAGE_SIZE,
|
|
mapped_cpu_pages_size / (1024u * 1024u));
|
|
|
|
gpu_info_print_ce_caps(gpu, s);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
static void
|
|
gpu_fault_stats_print_common(uvm_parent_gpu_t *parent_gpu, struct seq_file *s)
|
|
{
|
|
NvU64 num_pages_in;
|
|
NvU64 num_pages_out;
|
|
|
|
UVM_ASSERT(uvm_procfs_is_debug_enabled());
|
|
|
|
UVM_SEQ_OR_DBG_PRINT(s, "replayable_faults %llu\n", parent_gpu->stats.num_replayable_faults);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "duplicates %llu\n",
|
|
parent_gpu->fault_buffer_info.replayable.stats.num_duplicate_faults);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "faults_by_access_type:\n");
|
|
UVM_SEQ_OR_DBG_PRINT(s, " prefetch %llu\n",
|
|
parent_gpu->fault_buffer_info.replayable.stats.num_prefetch_faults);
|
|
UVM_SEQ_OR_DBG_PRINT(s, " read %llu\n",
|
|
parent_gpu->fault_buffer_info.replayable.stats.num_read_faults);
|
|
UVM_SEQ_OR_DBG_PRINT(s, " write %llu\n",
|
|
parent_gpu->fault_buffer_info.replayable.stats.num_write_faults);
|
|
UVM_SEQ_OR_DBG_PRINT(s, " atomic %llu\n",
|
|
parent_gpu->fault_buffer_info.replayable.stats.num_atomic_faults);
|
|
num_pages_out = atomic64_read(&parent_gpu->fault_buffer_info.replayable.stats.num_pages_out);
|
|
num_pages_in = atomic64_read(&parent_gpu->fault_buffer_info.replayable.stats.num_pages_in);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "migrations:\n");
|
|
UVM_SEQ_OR_DBG_PRINT(s, " num_pages_in %llu (%llu MB)\n", num_pages_in,
|
|
(num_pages_in * (NvU64)PAGE_SIZE) / (1024u * 1024u));
|
|
UVM_SEQ_OR_DBG_PRINT(s, " num_pages_out %llu (%llu MB)\n", num_pages_out,
|
|
(num_pages_out * (NvU64)PAGE_SIZE) / (1024u * 1024u));
|
|
UVM_SEQ_OR_DBG_PRINT(s, "replays:\n");
|
|
UVM_SEQ_OR_DBG_PRINT(s, " start %llu\n",
|
|
parent_gpu->fault_buffer_info.replayable.stats.num_replays);
|
|
UVM_SEQ_OR_DBG_PRINT(s, " start_ack_all %llu\n",
|
|
parent_gpu->fault_buffer_info.replayable.stats.num_replays_ack_all);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "non_replayable_faults %llu\n", parent_gpu->stats.num_non_replayable_faults);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "faults_by_access_type:\n");
|
|
UVM_SEQ_OR_DBG_PRINT(s, " read %llu\n",
|
|
parent_gpu->fault_buffer_info.non_replayable.stats.num_read_faults);
|
|
UVM_SEQ_OR_DBG_PRINT(s, " write %llu\n",
|
|
parent_gpu->fault_buffer_info.non_replayable.stats.num_write_faults);
|
|
UVM_SEQ_OR_DBG_PRINT(s, " atomic %llu\n",
|
|
parent_gpu->fault_buffer_info.non_replayable.stats.num_atomic_faults);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "faults_by_addressing:\n");
|
|
UVM_SEQ_OR_DBG_PRINT(s, " virtual %llu\n",
|
|
parent_gpu->stats.num_non_replayable_faults -
|
|
parent_gpu->fault_buffer_info.non_replayable.stats.num_physical_faults);
|
|
UVM_SEQ_OR_DBG_PRINT(s, " physical %llu\n",
|
|
parent_gpu->fault_buffer_info.non_replayable.stats.num_physical_faults);
|
|
num_pages_out = atomic64_read(&parent_gpu->fault_buffer_info.non_replayable.stats.num_pages_out);
|
|
num_pages_in = atomic64_read(&parent_gpu->fault_buffer_info.non_replayable.stats.num_pages_in);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "migrations:\n");
|
|
UVM_SEQ_OR_DBG_PRINT(s, " num_pages_in %llu (%llu MB)\n", num_pages_in,
|
|
(num_pages_in * (NvU64)PAGE_SIZE) / (1024u * 1024u));
|
|
UVM_SEQ_OR_DBG_PRINT(s, " num_pages_out %llu (%llu MB)\n", num_pages_out,
|
|
(num_pages_out * (NvU64)PAGE_SIZE) / (1024u * 1024u));
|
|
}
|
|
|
|
static void gpu_access_counters_print_common(uvm_parent_gpu_t *parent_gpu, struct seq_file *s)
|
|
{
|
|
NvU64 num_pages_in;
|
|
NvU64 num_pages_out;
|
|
|
|
UVM_ASSERT(uvm_procfs_is_debug_enabled());
|
|
|
|
num_pages_out = atomic64_read(&parent_gpu->access_counter_buffer_info.stats.num_pages_out);
|
|
num_pages_in = atomic64_read(&parent_gpu->access_counter_buffer_info.stats.num_pages_in);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "migrations:\n");
|
|
UVM_SEQ_OR_DBG_PRINT(s, " num_pages_in %llu (%llu MB)\n", num_pages_in,
|
|
(num_pages_in * (NvU64)PAGE_SIZE) / (1024u * 1024u));
|
|
UVM_SEQ_OR_DBG_PRINT(s, " num_pages_out %llu (%llu MB)\n", num_pages_out,
|
|
(num_pages_out * (NvU64)PAGE_SIZE) / (1024u * 1024u));
|
|
}
|
|
|
|
void uvm_gpu_print(uvm_gpu_t *gpu)
|
|
{
|
|
gpu_info_print_common(gpu, NULL);
|
|
}
|
|
|
|
static void gpu_peer_caps_print(uvm_gpu_t **gpu_pair, struct seq_file *s)
|
|
{
|
|
bool nvswitch_connected;
|
|
uvm_aperture_t aperture;
|
|
uvm_gpu_peer_t *peer_caps;
|
|
uvm_gpu_t *local;
|
|
uvm_gpu_t *remote;
|
|
|
|
UVM_ASSERT(uvm_procfs_is_debug_enabled());
|
|
|
|
local = gpu_pair[0];
|
|
remote = gpu_pair[1];
|
|
peer_caps = uvm_gpu_peer_caps(local, remote);
|
|
aperture = uvm_gpu_peer_aperture(local, remote);
|
|
nvswitch_connected = uvm_gpus_are_nvswitch_connected(local, remote);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "Link type %s\n", uvm_gpu_link_type_string(peer_caps->link_type));
|
|
UVM_SEQ_OR_DBG_PRINT(s, "Bandwidth %uMBps\n", peer_caps->total_link_line_rate_mbyte_per_s);
|
|
UVM_SEQ_OR_DBG_PRINT(s, "Aperture %s\n", uvm_aperture_string(aperture));
|
|
UVM_SEQ_OR_DBG_PRINT(s, "Connected through NVSWITCH %s\n", nvswitch_connected ? "True" : "False");
|
|
UVM_SEQ_OR_DBG_PRINT(s, "Refcount %llu\n", UVM_READ_ONCE(peer_caps->ref_count));
|
|
}
|
|
|
|
static int nv_procfs_read_gpu_info(struct seq_file *s, void *v)
|
|
{
|
|
uvm_gpu_t *gpu = (uvm_gpu_t *)s->private;
|
|
|
|
if (!uvm_down_read_trylock(&g_uvm_global.pm.lock))
|
|
return -EAGAIN;
|
|
|
|
gpu_info_print_common(gpu, s);
|
|
|
|
uvm_up_read(&g_uvm_global.pm.lock);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int nv_procfs_read_gpu_info_entry(struct seq_file *s, void *v)
|
|
{
|
|
UVM_ENTRY_RET(nv_procfs_read_gpu_info(s, v));
|
|
}
|
|
|
|
static int nv_procfs_read_gpu_fault_stats(struct seq_file *s, void *v)
|
|
{
|
|
uvm_parent_gpu_t *parent_gpu = (uvm_parent_gpu_t *)s->private;
|
|
|
|
if (!uvm_down_read_trylock(&g_uvm_global.pm.lock))
|
|
return -EAGAIN;
|
|
|
|
gpu_fault_stats_print_common(parent_gpu, s);
|
|
|
|
uvm_up_read(&g_uvm_global.pm.lock);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int nv_procfs_read_gpu_fault_stats_entry(struct seq_file *s, void *v)
|
|
{
|
|
UVM_ENTRY_RET(nv_procfs_read_gpu_fault_stats(s, v));
|
|
}
|
|
|
|
static int nv_procfs_read_gpu_access_counters(struct seq_file *s, void *v)
|
|
{
|
|
uvm_parent_gpu_t *parent_gpu = (uvm_parent_gpu_t *)s->private;
|
|
|
|
if (!uvm_down_read_trylock(&g_uvm_global.pm.lock))
|
|
return -EAGAIN;
|
|
|
|
gpu_access_counters_print_common(parent_gpu, s);
|
|
|
|
uvm_up_read(&g_uvm_global.pm.lock);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int nv_procfs_read_gpu_access_counters_entry(struct seq_file *s, void *v)
|
|
{
|
|
UVM_ENTRY_RET(nv_procfs_read_gpu_access_counters(s, v));
|
|
}
|
|
|
|
UVM_DEFINE_SINGLE_PROCFS_FILE(gpu_info_entry);
|
|
UVM_DEFINE_SINGLE_PROCFS_FILE(gpu_fault_stats_entry);
|
|
UVM_DEFINE_SINGLE_PROCFS_FILE(gpu_access_counters_entry);
|
|
|
|
static NV_STATUS init_parent_procfs_dir(uvm_parent_gpu_t *parent_gpu)
|
|
{
|
|
struct proc_dir_entry *gpu_base_dir_entry;
|
|
char uuid_text_buffer[UVM_GPU_UUID_TEXT_BUFFER_LENGTH];
|
|
char gpu_dir_name[sizeof(uuid_text_buffer) + 1];
|
|
|
|
if (!uvm_procfs_is_enabled())
|
|
return NV_OK;
|
|
|
|
gpu_base_dir_entry = uvm_procfs_get_gpu_base_dir();
|
|
|
|
format_uuid_to_buffer(uuid_text_buffer, sizeof(uuid_text_buffer), &parent_gpu->uuid);
|
|
|
|
// Create UVM-GPU-${UUID} directory
|
|
snprintf(gpu_dir_name, sizeof(gpu_dir_name), "%s", uuid_text_buffer);
|
|
|
|
parent_gpu->procfs.dir = NV_CREATE_PROC_DIR(gpu_dir_name, gpu_base_dir_entry);
|
|
if (parent_gpu->procfs.dir == NULL)
|
|
return NV_ERR_OPERATING_SYSTEM;
|
|
|
|
return NV_OK;
|
|
}
|
|
|
|
static void deinit_parent_procfs_dir(uvm_parent_gpu_t *parent_gpu)
|
|
{
|
|
uvm_procfs_destroy_entry(parent_gpu->procfs.dir);
|
|
}
|
|
|
|
static NV_STATUS init_parent_procfs_files(uvm_parent_gpu_t *parent_gpu)
|
|
{
|
|
// Fault and access counter files are debug only
|
|
if (!uvm_procfs_is_debug_enabled())
|
|
return NV_OK;
|
|
|
|
parent_gpu->procfs.fault_stats_file = NV_CREATE_PROC_FILE("fault_stats",
|
|
parent_gpu->procfs.dir,
|
|
gpu_fault_stats_entry,
|
|
parent_gpu);
|
|
if (parent_gpu->procfs.fault_stats_file == NULL)
|
|
return NV_ERR_OPERATING_SYSTEM;
|
|
|
|
parent_gpu->procfs.access_counters_file = NV_CREATE_PROC_FILE("access_counters",
|
|
parent_gpu->procfs.dir,
|
|
gpu_access_counters_entry,
|
|
parent_gpu);
|
|
if (parent_gpu->procfs.access_counters_file == NULL)
|
|
return NV_ERR_OPERATING_SYSTEM;
|
|
|
|
return NV_OK;
|
|
}
|
|
|
|
static void deinit_parent_procfs_files(uvm_parent_gpu_t *parent_gpu)
|
|
{
|
|
uvm_procfs_destroy_entry(parent_gpu->procfs.access_counters_file);
|
|
uvm_procfs_destroy_entry(parent_gpu->procfs.fault_stats_file);
|
|
}
|
|
|
|
static NV_STATUS init_procfs_dirs(uvm_gpu_t *gpu)
|
|
{
|
|
struct proc_dir_entry *gpu_base_dir_entry;
|
|
char symlink_name[16]; // Hold a global_gpu_id_t value in decimal.
|
|
char uuid_text_buffer[UVM_GPU_UUID_TEXT_BUFFER_LENGTH];
|
|
char gpu_dir_name[sizeof(symlink_name) + sizeof(uuid_text_buffer) + 1];
|
|
|
|
if (!uvm_procfs_is_enabled())
|
|
return NV_OK;
|
|
|
|
format_uuid_to_buffer(uuid_text_buffer, sizeof(uuid_text_buffer), uvm_gpu_uuid(gpu));
|
|
|
|
gpu_base_dir_entry = uvm_procfs_get_gpu_base_dir();
|
|
|
|
// Create UVM-GPU-${UUID}/${sub_processor_index} directory
|
|
snprintf(gpu_dir_name, sizeof(gpu_dir_name), "%u", uvm_global_id_sub_processor_index(gpu->global_id));
|
|
|
|
gpu->procfs.dir = NV_CREATE_PROC_DIR(gpu_dir_name, gpu->parent->procfs.dir);
|
|
if (gpu->procfs.dir == NULL)
|
|
return NV_ERR_OPERATING_SYSTEM;
|
|
|
|
// Create symlink from ${global_gpu_id} to
|
|
// gpus/UVM-GPU-${UUID}/${sub_processor_index}
|
|
snprintf(symlink_name, sizeof(symlink_name), "%u", uvm_global_id_value(gpu->global_id));
|
|
snprintf(gpu_dir_name,
|
|
sizeof(gpu_dir_name),
|
|
"%s/%u",
|
|
uuid_text_buffer,
|
|
uvm_global_id_sub_processor_index(gpu->global_id));
|
|
|
|
gpu->procfs.dir_symlink = proc_symlink(symlink_name, gpu_base_dir_entry, gpu_dir_name);
|
|
if (gpu->procfs.dir_symlink == NULL)
|
|
return NV_ERR_OPERATING_SYSTEM;
|
|
|
|
// GPU peer files are debug only
|
|
if (!uvm_procfs_is_debug_enabled())
|
|
return NV_OK;
|
|
|
|
gpu->procfs.dir_peers = NV_CREATE_PROC_DIR(UVM_PROC_GPUS_PEER_DIR_NAME, gpu->procfs.dir);
|
|
if (gpu->procfs.dir_peers == NULL)
|
|
return NV_ERR_OPERATING_SYSTEM;
|
|
|
|
return NV_OK;
|
|
}
|
|
|
|
// The kernel waits on readers to finish before returning from those calls
|
|
static void deinit_procfs_dirs(uvm_gpu_t *gpu)
|
|
{
|
|
uvm_procfs_destroy_entry(gpu->procfs.dir_peers);
|
|
uvm_procfs_destroy_entry(gpu->procfs.dir_symlink);
|
|
uvm_procfs_destroy_entry(gpu->procfs.dir);
|
|
}
|
|
|
|
static NV_STATUS init_procfs_files(uvm_gpu_t *gpu)
|
|
{
|
|
gpu->procfs.info_file = NV_CREATE_PROC_FILE("info", gpu->procfs.dir, gpu_info_entry, gpu);
|
|
if (gpu->procfs.info_file == NULL)
|
|
return NV_ERR_OPERATING_SYSTEM;
|
|
|
|
return NV_OK;
|
|
}
|
|
|
|
static void deinit_procfs_files(uvm_gpu_t *gpu)
|
|
{
|
|
uvm_procfs_destroy_entry(gpu->procfs.info_file);
|
|
}
|
|
|
|
static void deinit_procfs_peer_cap_files(uvm_gpu_peer_t *peer_caps)
|
|
{
|
|
uvm_procfs_destroy_entry(peer_caps->procfs.peer_symlink_file[0]);
|
|
uvm_procfs_destroy_entry(peer_caps->procfs.peer_symlink_file[1]);
|
|
uvm_procfs_destroy_entry(peer_caps->procfs.peer_file[0]);
|
|
uvm_procfs_destroy_entry(peer_caps->procfs.peer_file[1]);
|
|
}
|
|
|
|
static NV_STATUS init_semaphore_pool(uvm_gpu_t *gpu)
|
|
{
|
|
NV_STATUS status;
|
|
uvm_gpu_t *other_gpu;
|
|
|
|
status = uvm_gpu_semaphore_pool_create(gpu, &gpu->semaphore_pool);
|
|
if (status != NV_OK)
|
|
return status;
|
|
|
|
for_each_global_gpu(other_gpu) {
|
|
if (other_gpu == gpu)
|
|
continue;
|
|
status = uvm_gpu_semaphore_pool_map_gpu(other_gpu->semaphore_pool, gpu);
|
|
if (status != NV_OK)
|
|
return status;
|
|
}
|
|
|
|
return NV_OK;
|
|
}
|
|
|
|
static void deinit_semaphore_pool(uvm_gpu_t *gpu)
|
|
{
|
|
uvm_gpu_t *other_gpu;
|
|
|
|
for_each_global_gpu(other_gpu) {
|
|
if (other_gpu == gpu)
|
|
continue;
|
|
uvm_gpu_semaphore_pool_unmap_gpu(other_gpu->semaphore_pool, gpu);
|
|
}
|
|
|
|
uvm_gpu_semaphore_pool_destroy(gpu->semaphore_pool);
|
|
}
|
|
|
|
static NV_STATUS find_unused_global_gpu_id(uvm_parent_gpu_t *parent_gpu, uvm_global_gpu_id_t *out_id)
|
|
{
|
|
NvU32 i;
|
|
|
|
uvm_assert_mutex_locked(&g_uvm_global.global_lock);
|
|
|
|
if (!parent_gpu) {
|
|
for (i = 0; i < UVM_MAX_GPUS; i++) {
|
|
if (!g_uvm_global.parent_gpus[i]) {
|
|
*out_id = uvm_global_gpu_id_from_parent_index(i);
|
|
return NV_OK;
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
NvU32 sub_processor_index = find_first_zero_bit(parent_gpu->valid_gpus, UVM_ID_MAX_SUB_PROCESSORS);
|
|
if (sub_processor_index < UVM_ID_MAX_SUB_PROCESSORS) {
|
|
*out_id = uvm_global_gpu_id_from_sub_processor_index(parent_gpu->id, sub_processor_index);
|
|
return NV_OK;
|
|
}
|
|
}
|
|
|
|
return NV_ERR_INSUFFICIENT_RESOURCES;
|
|
}
|
|
|
|
// Allocates a uvm_parent_gpu_t, assigns the GPU ID, and sets up basic data
|
|
// structures, but leaves all other initialization up to the caller.
|
|
static NV_STATUS alloc_parent_gpu(const NvProcessorUuid *gpu_uuid,
|
|
uvm_gpu_id_t gpu_id,
|
|
uvm_parent_gpu_t **parent_gpu_out)
|
|
{
|
|
uvm_parent_gpu_t *parent_gpu;
|
|
|
|
parent_gpu = uvm_kvmalloc_zero(sizeof(*parent_gpu));
|
|
if (!parent_gpu)
|
|
return NV_ERR_NO_MEMORY;
|
|
|
|
parent_gpu->id = gpu_id;
|
|
|
|
uvm_processor_uuid_copy(&parent_gpu->uuid, gpu_uuid);
|
|
uvm_sema_init(&parent_gpu->isr.replayable_faults.service_lock, 1, UVM_LOCK_ORDER_ISR);
|
|
uvm_sema_init(&parent_gpu->isr.non_replayable_faults.service_lock, 1, UVM_LOCK_ORDER_ISR);
|
|
uvm_sema_init(&parent_gpu->isr.access_counters.service_lock, 1, UVM_LOCK_ORDER_ISR);
|
|
uvm_spin_lock_irqsave_init(&parent_gpu->isr.interrupts_lock, UVM_LOCK_ORDER_LEAF);
|
|
uvm_spin_lock_init(&parent_gpu->instance_ptr_table_lock, UVM_LOCK_ORDER_LEAF);
|
|
uvm_rb_tree_init(&parent_gpu->instance_ptr_table);
|
|
uvm_rb_tree_init(&parent_gpu->tsg_table);
|
|
|
|
nv_kref_init(&parent_gpu->gpu_kref);
|
|
|
|
*parent_gpu_out = parent_gpu;
|
|
|
|
return NV_OK;
|
|
}
|
|
|
|
// Allocates a uvm_gpu_t struct and initializes the basic fields and leaves all
|
|
// other initialization up to the caller.
|
|
static uvm_gpu_t *alloc_gpu(uvm_parent_gpu_t *parent_gpu, uvm_global_gpu_id_t global_gpu_id)
|
|
{
|
|
NvU32 sub_processor_index;
|
|
uvm_gpu_t *gpu;
|
|
|
|
gpu = uvm_kvmalloc_zero(sizeof(*gpu));
|
|
if (!gpu)
|
|
return gpu;
|
|
|
|
gpu->id = parent_gpu->id;
|
|
gpu->global_id = global_gpu_id;
|
|
gpu->parent = parent_gpu;
|
|
|
|
// Initialize enough of the gpu struct for remove_gpu to be called
|
|
gpu->magic = UVM_GPU_MAGIC_VALUE;
|
|
uvm_spin_lock_init(&gpu->peer_info.peer_gpus_lock, UVM_LOCK_ORDER_LEAF);
|
|
|
|
sub_processor_index = uvm_global_id_sub_processor_index(global_gpu_id);
|
|
parent_gpu->gpus[sub_processor_index] = gpu;
|
|
|
|
return gpu;
|
|
}
|
|
|
|
static NV_STATUS configure_address_space(uvm_gpu_t *gpu)
|
|
{
|
|
NV_STATUS status;
|
|
NvU32 num_entries;
|
|
NvU64 va_size;
|
|
NvU64 va_per_entry;
|
|
|
|
status = uvm_page_tree_init(gpu,
|
|
NULL,
|
|
UVM_PAGE_TREE_TYPE_KERNEL,
|
|
gpu->big_page.internal_size,
|
|
uvm_gpu_page_tree_init_location(gpu),
|
|
&gpu->address_space_tree);
|
|
if (status != NV_OK) {
|
|
UVM_ERR_PRINT("Initializing the page tree failed: %s, GPU %s\n", nvstatusToString(status), uvm_gpu_name(gpu));
|
|
return status;
|
|
}
|
|
|
|
num_entries = uvm_mmu_page_tree_entries(&gpu->address_space_tree, 0, UVM_PAGE_SIZE_AGNOSTIC);
|
|
|
|
UVM_ASSERT(gpu->address_space_tree.hal->num_va_bits() < 64);
|
|
va_size = 1ull << gpu->address_space_tree.hal->num_va_bits();
|
|
va_per_entry = va_size / num_entries;
|
|
|
|
// Make sure that RM's part of the VA is aligned to the VA covered by a
|
|
// single top level PDE.
|
|
UVM_ASSERT_MSG(gpu->parent->rm_va_base % va_per_entry == 0,
|
|
"va_base 0x%llx va_per_entry 0x%llx\n", gpu->parent->rm_va_base, va_per_entry);
|
|
UVM_ASSERT_MSG(gpu->parent->rm_va_size % va_per_entry == 0,
|
|
"va_size 0x%llx va_per_entry 0x%llx\n", gpu->parent->rm_va_size, va_per_entry);
|
|
|
|
status = uvm_rm_locked_call(nvUvmInterfaceSetPageDirectory(gpu->rm_address_space,
|
|
uvm_page_tree_pdb(&gpu->address_space_tree)->addr.address, num_entries,
|
|
uvm_page_tree_pdb(&gpu->address_space_tree)->addr.aperture == UVM_APERTURE_VID,
|
|
-1U /* Invalid PASID for internal RM address space */));
|
|
if (status != NV_OK) {
|
|
UVM_ERR_PRINT("nvUvmInterfaceSetPageDirectory() failed: %s, GPU %s\n",
|
|
nvstatusToString(status),
|
|
uvm_gpu_name(gpu));
|
|
return status;
|
|
}
|
|
gpu->rm_address_space_moved_to_page_tree = true;
|
|
|
|
return NV_OK;
|
|
}
|
|
|
|
static void deconfigure_address_space(uvm_gpu_t *gpu)
|
|
{
|
|
if (gpu->rm_address_space_moved_to_page_tree)
|
|
uvm_rm_locked_call_void(nvUvmInterfaceUnsetPageDirectory(gpu->rm_address_space));
|
|
|
|
if (gpu->address_space_tree.root)
|
|
uvm_page_tree_deinit(&gpu->address_space_tree);
|
|
}
|
|
|
|
static NV_STATUS service_interrupts(uvm_parent_gpu_t *parent_gpu)
|
|
{
|
|
// Asking RM to service interrupts from top half interrupt handler would
|
|
// very likely deadlock.
|
|
UVM_ASSERT(!in_interrupt());
|
|
|
|
return uvm_rm_locked_call(nvUvmInterfaceServiceDeviceInterruptsRM(parent_gpu->rm_device));
|
|
}
|
|
|
|
NV_STATUS uvm_gpu_check_ecc_error(uvm_gpu_t *gpu)
|
|
{
|
|
NV_STATUS status = uvm_gpu_check_ecc_error_no_rm(gpu);
|
|
|
|
if (status == NV_OK || status != NV_WARN_MORE_PROCESSING_REQUIRED)
|
|
return status;
|
|
|
|
// An interrupt that might mean an ECC error needs to be serviced.
|
|
UVM_ASSERT(status == NV_WARN_MORE_PROCESSING_REQUIRED);
|
|
|
|
status = service_interrupts(gpu->parent);
|
|
if (status != NV_OK) {
|
|
UVM_ERR_PRINT("Servicing interrupts failed: %s, GPU %s\n", nvstatusToString(status), uvm_gpu_name(gpu));
|
|
return status;
|
|
}
|
|
|
|
// After servicing interrupts the ECC error notifier should be current.
|
|
if (*gpu->ecc.error_notifier) {
|
|
UVM_ERR_PRINT("ECC error encountered, GPU %s\n", uvm_gpu_name(gpu));
|
|
uvm_global_set_fatal_error(NV_ERR_ECC_ERROR);
|
|
return NV_ERR_ECC_ERROR;
|
|
}
|
|
|
|
return NV_OK;
|
|
}
|
|
|
|
static NV_STATUS init_parent_gpu(uvm_parent_gpu_t *parent_gpu,
|
|
const NvProcessorUuid *gpu_uuid,
|
|
const UvmGpuInfo *gpu_info,
|
|
const UvmGpuPlatformInfo *gpu_platform_info)
|
|
{
|
|
NV_STATUS status;
|
|
|
|
status = uvm_rm_locked_call(nvUvmInterfaceDeviceCreate(g_uvm_global.rm_session_handle,
|
|
gpu_info,
|
|
gpu_uuid,
|
|
&parent_gpu->rm_device,
|
|
NV_FALSE));
|
|
if (status != NV_OK) {
|
|
UVM_ERR_PRINT("Creating RM device failed: %s, GPU %s\n", nvstatusToString(status), parent_gpu->name);
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
parent_gpu->pci_dev = gpu_platform_info->pci_dev;
|
|
parent_gpu->closest_cpu_numa_node = dev_to_node(&parent_gpu->pci_dev->dev);
|
|
parent_gpu->dma_addressable_start = gpu_platform_info->dma_addressable_start;
|
|
parent_gpu->dma_addressable_limit = gpu_platform_info->dma_addressable_limit;
|
|
|
|
parent_gpu->sli_enabled = (gpu_info->subdeviceCount > 1);
|
|
|
|
parent_gpu->virt_mode = gpu_info->virtMode;
|
|
if (parent_gpu->virt_mode == UVM_VIRT_MODE_LEGACY) {
|
|
UVM_ERR_PRINT("Failed to init GPU %s. UVM is not supported in legacy virtualization mode\n", parent_gpu->name);
|
|
return NV_ERR_NOT_SUPPORTED;
|
|
}
|
|
|
|
if (gpu_info->isSimulated)
|
|
++g_uvm_global.num_simulated_devices;
|
|
|
|
status = init_parent_procfs_dir(parent_gpu);
|
|
if (status != NV_OK) {
|
|
UVM_ERR_PRINT("Failed to init parent procfs dir: %s, GPU %s\n", nvstatusToString(status), parent_gpu->name);
|
|
return status;
|
|
}
|
|
|
|
status = uvm_hal_init_gpu(parent_gpu);
|
|
if (status != NV_OK) {
|
|
UVM_ERR_PRINT("Failed to init GPU hal: %s, GPU %s\n", nvstatusToString(status), parent_gpu->name);
|
|
return status;
|
|
}
|
|
|
|
uvm_hal_init_properties(parent_gpu);
|
|
|
|
UVM_ASSERT(!parent_gpu->rm_info.smcEnabled || parent_gpu->smc.supported);
|
|
parent_gpu->smc.enabled = !!parent_gpu->rm_info.smcEnabled;
|
|
|
|
uvm_mmu_init_gpu_chunk_sizes(parent_gpu);
|
|
|
|
status = get_gpu_caps(parent_gpu);
|
|
if (status != NV_OK) {
|
|
UVM_ERR_PRINT("Failed to get GPU caps: %s, GPU %s\n", nvstatusToString(status), parent_gpu->name);
|
|
return status;
|
|
}
|
|
|
|
status = uvm_ats_add_gpu(parent_gpu);
|
|
if (status != NV_OK) {
|
|
UVM_ERR_PRINT("uvm_ats_add_gpu failed: %s, GPU %s\n", nvstatusToString(status), parent_gpu->name);
|
|
return status;
|
|
}
|
|
|
|
status = init_parent_procfs_files(parent_gpu);
|
|
if (status != NV_OK) {
|
|
UVM_ERR_PRINT("Failed to init parent procfs files: %s, GPU %s\n", nvstatusToString(status), parent_gpu->name);
|
|
return status;
|
|
}
|
|
|
|
status = uvm_gpu_init_isr(parent_gpu);
|
|
if (status != NV_OK) {
|
|
UVM_ERR_PRINT("Failed to init ISR: %s, GPU %s\n", nvstatusToString(status), parent_gpu->name);
|
|
return status;
|
|
}
|
|
|
|
return NV_OK;
|
|
}
|
|
|
|
static NV_STATUS init_gpu(uvm_gpu_t *gpu, const UvmGpuInfo *gpu_info)
|
|
{
|
|
NV_STATUS status;
|
|
|
|
// Presently, an RM client can only subscribe to a single partition per
|
|
// GPU. Therefore, UVM needs to create several RM clients. For simplicity,
|
|
// and since P2P is not supported when SMC partitions are created, we
|
|
// create a client (session) per GPU partition.
|
|
if (gpu->parent->smc.enabled) {
|
|
UvmPlatformInfo platform_info;
|
|
status = uvm_rm_locked_call(nvUvmInterfaceSessionCreate(&gpu->smc.rm_session_handle, &platform_info));
|
|
if (status != NV_OK) {
|
|
UVM_ERR_PRINT("Creating RM session failed: %s\n", nvstatusToString(status));
|
|
return status;
|
|
}
|
|
|
|
status = uvm_rm_locked_call(nvUvmInterfaceDeviceCreate(uvm_gpu_session_handle(gpu),
|
|
gpu_info,
|
|
uvm_gpu_uuid(gpu),
|
|
&gpu->smc.rm_device,
|
|
NV_TRUE));
|
|
if (status != NV_OK) {
|
|
UVM_ERR_PRINT("Creating RM device failed: %s, GPU %s\n", nvstatusToString(status), uvm_gpu_name(gpu));
|
|
return status;
|
|
}
|
|
}
|
|
|
|
gpu->smc.swizz_id = gpu_info->smcSwizzId;
|
|
|
|
// Initialize the per-GPU procfs dirs as early as possible so that other
|
|
// parts of the driver can add files in them as part of their per-GPU init.
|
|
status = init_procfs_dirs(gpu);
|
|
if (status != NV_OK) {
|
|
UVM_ERR_PRINT("Failed to init procfs dirs: %s, GPU %s\n", nvstatusToString(status), uvm_gpu_name(gpu));
|
|
return status;
|
|
}
|
|
|
|
uvm_mmu_init_gpu_peer_addresses(gpu);
|
|
|
|
status = alloc_and_init_address_space(gpu);
|
|
if (status != NV_OK) {
|
|
UVM_ERR_PRINT("Creating RM address space failed: %s, GPU %s\n", nvstatusToString(status), uvm_gpu_name(gpu));
|
|
return status;
|
|
}
|
|
|
|
status = get_gpu_fb_info(gpu);
|
|
if (status != NV_OK) {
|
|
UVM_ERR_PRINT("Failed to get GPU FB info: %s, GPU %s\n", nvstatusToString(status), uvm_gpu_name(gpu));
|
|
return status;
|
|
}
|
|
|
|
status = get_gpu_ecc_info(gpu);
|
|
if (status != NV_OK) {
|
|
UVM_ERR_PRINT("Failed to get GPU ECC info: %s, GPU %s\n", nvstatusToString(status), uvm_gpu_name(gpu));
|
|
return status;
|
|
}
|
|
|
|
status = uvm_pmm_gpu_init(&gpu->pmm);
|
|
if (status != NV_OK) {
|
|
UVM_ERR_PRINT("PMM initialization failed: %s, GPU %s\n", nvstatusToString(status), uvm_gpu_name(gpu));
|
|
return status;
|
|
}
|
|
|
|
status = uvm_pmm_sysmem_mappings_init(gpu, &gpu->pmm_reverse_sysmem_mappings);
|
|
if (status != NV_OK) {
|
|
UVM_ERR_PRINT("CPU PMM MMIO initialization failed: %s, GPU %s\n", nvstatusToString(status), uvm_gpu_name(gpu));
|
|
return status;
|
|
}
|
|
|
|
status = init_semaphore_pool(gpu);
|
|
if (status != NV_OK) {
|
|
UVM_ERR_PRINT("Failed to initialize the semaphore pool: %s, GPU %s\n",
|
|
nvstatusToString(status),
|
|
uvm_gpu_name(gpu));
|
|
return status;
|
|
}
|
|
|
|
status = uvm_channel_manager_create(gpu, &gpu->channel_manager);
|
|
if (status != NV_OK) {
|
|
UVM_ERR_PRINT("Failed to initialize the channel manager: %s, GPU %s\n",
|
|
nvstatusToString(status),
|
|
uvm_gpu_name(gpu));
|
|
return status;
|
|
}
|
|
|
|
status = configure_address_space(gpu);
|
|
if (status != NV_OK) {
|
|
UVM_ERR_PRINT("Failed to configure the GPU address space: %s, GPU %s\n",
|
|
nvstatusToString(status),
|
|
uvm_gpu_name(gpu));
|
|
return status;
|
|
}
|
|
|
|
status = uvm_mmu_create_flat_mappings(gpu);
|
|
if (status != NV_OK) {
|
|
UVM_ERR_PRINT("Creating flat mappings failed: %s, GPU %s\n", nvstatusToString(status), uvm_gpu_name(gpu));
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
status = init_procfs_files(gpu);
|
|
if (status != NV_OK) {
|
|
UVM_ERR_PRINT("Failed to init procfs files: %s, GPU %s\n", nvstatusToString(status), uvm_gpu_name(gpu));
|
|
return status;
|
|
}
|
|
|
|
status = uvm_perf_heuristics_add_gpu(gpu);
|
|
if (status != NV_OK) {
|
|
UVM_ERR_PRINT("Failed to init heuristics: %s, GPU %s\n", nvstatusToString(status), uvm_gpu_name(gpu));
|
|
return status;
|
|
}
|
|
|
|
return NV_OK;
|
|
}
|
|
|
|
// Add a new gpu and register it with RM
|
|
// TODO: Bug 2844714: Split parent-specific parts of this function out into a
|
|
// separate add_parent_gpu() function.
|
|
static NV_STATUS add_gpu(const NvProcessorUuid *gpu_uuid,
|
|
const uvm_global_gpu_id_t global_gpu_id,
|
|
const UvmGpuInfo *gpu_info,
|
|
const UvmGpuPlatformInfo *gpu_platform_info,
|
|
uvm_parent_gpu_t *parent_gpu,
|
|
uvm_gpu_t **gpu_out)
|
|
{
|
|
NV_STATUS status;
|
|
bool alloc_parent = (parent_gpu == NULL);
|
|
uvm_gpu_t *gpu = NULL;
|
|
|
|
uvm_assert_mutex_locked(&g_uvm_global.global_lock);
|
|
|
|
if (alloc_parent) {
|
|
status = alloc_parent_gpu(gpu_uuid, uvm_gpu_id_from_global_gpu_id(global_gpu_id), &parent_gpu);
|
|
if (status != NV_OK)
|
|
return status;
|
|
}
|
|
|
|
gpu = alloc_gpu(parent_gpu, global_gpu_id);
|
|
if (!gpu) {
|
|
if (alloc_parent)
|
|
uvm_parent_gpu_kref_put(parent_gpu);
|
|
|
|
return NV_ERR_NO_MEMORY;
|
|
}
|
|
|
|
parent_gpu->num_retained_gpus++;
|
|
|
|
if (alloc_parent)
|
|
fill_gpu_info(parent_gpu, gpu_info);
|
|
|
|
// After this point all error clean up should be handled by remove_gpu()
|
|
|
|
if (!gpu_supports_uvm(parent_gpu)) {
|
|
UVM_DBG_PRINT("Registration of non-UVM-capable GPU attempted: GPU %s\n", uvm_gpu_name(gpu));
|
|
status = NV_ERR_NOT_SUPPORTED;
|
|
goto error;
|
|
}
|
|
|
|
if (alloc_parent) {
|
|
status = init_parent_gpu(parent_gpu, gpu_uuid, gpu_info, gpu_platform_info);
|
|
if (status != NV_OK)
|
|
goto error;
|
|
}
|
|
|
|
status = init_gpu(gpu, gpu_info);
|
|
if (status != NV_OK)
|
|
goto error;
|
|
|
|
status = uvm_gpu_check_ecc_error(gpu);
|
|
if (status != NV_OK)
|
|
goto error;
|
|
|
|
atomic64_set(&gpu->retained_count, 1);
|
|
uvm_global_processor_mask_set(&g_uvm_global.retained_gpus, gpu->global_id);
|
|
|
|
uvm_spin_lock_irqsave(&g_uvm_global.gpu_table_lock);
|
|
|
|
if (alloc_parent)
|
|
uvm_global_add_parent_gpu(parent_gpu);
|
|
|
|
// Mark the GPU as valid in the parent GPU's GPU table.
|
|
UVM_ASSERT(!test_bit(uvm_global_id_sub_processor_index(gpu->global_id), parent_gpu->valid_gpus));
|
|
__set_bit(uvm_global_id_sub_processor_index(gpu->global_id), parent_gpu->valid_gpus);
|
|
|
|
// Although locking correctness does not, at this early point (before the
|
|
// GPU is visible in the table) strictly require holding the gpu_table_lock
|
|
// in order to read gpu->isr.replayable_faults.handling, nor to enable page
|
|
// fault interrupts (this could have been done earlier), it is best to do it
|
|
// here, in order to avoid an interrupt storm. That way, we take advantage
|
|
// of the spinlock_irqsave side effect of turning off local CPU interrupts,
|
|
// part of holding the gpu_table_lock. That means that the local CPU won't
|
|
// receive any of these interrupts, until the GPU is safely added to the
|
|
// table (where the top half ISR can find it).
|
|
//
|
|
// As usual with spinlock_irqsave behavior, *other* CPUs can still handle
|
|
// these interrupts, but the local CPU will not be slowed down (interrupted)
|
|
// by such handling, and can quickly release the gpu_table_lock, thus
|
|
// unblocking any other CPU's top half (which waits for the gpu_table_lock).
|
|
if (alloc_parent && parent_gpu->isr.replayable_faults.handling) {
|
|
parent_gpu->fault_buffer_hal->enable_replayable_faults(parent_gpu);
|
|
|
|
// Clear the interrupt bit and force the re-evaluation of the interrupt
|
|
// condition to ensure that we don't miss any pending interrupt
|
|
parent_gpu->fault_buffer_hal->clear_replayable_faults(parent_gpu,
|
|
parent_gpu->fault_buffer_info.replayable.cached_get);
|
|
}
|
|
|
|
// Access counters are enabled on demand
|
|
|
|
uvm_spin_unlock_irqrestore(&g_uvm_global.gpu_table_lock);
|
|
|
|
if (alloc_parent) {
|
|
status = discover_nvlink_peers(gpu);
|
|
if (status != NV_OK) {
|
|
UVM_ERR_PRINT("Failed to discover NVLINK peers: %s, GPU %s\n", nvstatusToString(status), uvm_gpu_name(gpu));
|
|
|
|
// Nobody can have retained the GPU yet, since we still hold the global
|
|
// lock.
|
|
UVM_ASSERT(uvm_gpu_retained_count(gpu) == 1);
|
|
atomic64_set(&gpu->retained_count, 0);
|
|
goto error;
|
|
}
|
|
}
|
|
|
|
*gpu_out = gpu;
|
|
|
|
return NV_OK;
|
|
|
|
error:
|
|
remove_gpu(gpu);
|
|
|
|
return status;
|
|
}
|
|
|
|
static void sync_parent_gpu_trackers(uvm_parent_gpu_t *parent_gpu,
|
|
bool sync_replay_tracker,
|
|
bool sync_clear_faulted_tracker)
|
|
{
|
|
NV_STATUS status;
|
|
|
|
// Sync the replay tracker since it inherits dependencies from the VA block
|
|
// trackers.
|
|
if (sync_replay_tracker) {
|
|
uvm_gpu_replayable_faults_isr_lock(parent_gpu);
|
|
status = uvm_tracker_wait(&parent_gpu->fault_buffer_info.replayable.replay_tracker);
|
|
uvm_gpu_replayable_faults_isr_unlock(parent_gpu);
|
|
|
|
if (status != NV_OK)
|
|
UVM_ASSERT(status == uvm_global_get_status());
|
|
}
|
|
|
|
// Sync the clear_faulted tracker since it inherits dependencies from the
|
|
// VA block trackers, too.
|
|
if (sync_clear_faulted_tracker) {
|
|
uvm_gpu_non_replayable_faults_isr_lock(parent_gpu);
|
|
status = uvm_tracker_wait(&parent_gpu->fault_buffer_info.non_replayable.clear_faulted_tracker);
|
|
uvm_gpu_non_replayable_faults_isr_unlock(parent_gpu);
|
|
|
|
if (status != NV_OK)
|
|
UVM_ASSERT(status == uvm_global_get_status());
|
|
}
|
|
}
|
|
|
|
// Remove all references the given GPU has to other GPUs, since one of those
|
|
// other GPUs is getting removed. This involves waiting for any unfinished
|
|
// trackers contained by this GPU.
|
|
static void remove_gpus_from_gpu(uvm_gpu_t *gpu)
|
|
{
|
|
sync_parent_gpu_trackers(gpu->parent,
|
|
gpu->parent->isr.replayable_faults.handling,
|
|
gpu->parent->isr.non_replayable_faults.handling);
|
|
|
|
// Sync all trackers in PMM
|
|
uvm_pmm_gpu_sync(&gpu->pmm);
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
// Remove all references to the given GPU from its parent, since it is being
|
|
// removed. This involves waiting for any unfinished trackers contained
|
|
// by the parent GPU.
|
|
static void remove_gpu_from_parent_gpu(uvm_gpu_t *gpu)
|
|
{
|
|
// We use *.was_handling instead of *.handling here since this function is
|
|
// called after uvm_gpu_disable_isr(), and the *.handling flags will
|
|
// already have been copied to *.was_handling, and then set to false.
|
|
sync_parent_gpu_trackers(gpu->parent,
|
|
gpu->parent->isr.replayable_faults.was_handling,
|
|
gpu->parent->isr.non_replayable_faults.was_handling);
|
|
}
|
|
|
|
static void deinit_parent_gpu(uvm_parent_gpu_t *parent_gpu)
|
|
{
|
|
// All channels should have been removed before the retained count went to 0
|
|
UVM_ASSERT(uvm_rb_tree_empty(&parent_gpu->instance_ptr_table));
|
|
UVM_ASSERT(uvm_rb_tree_empty(&parent_gpu->tsg_table));
|
|
|
|
// Access counters should have been disabled when the GPU is no longer
|
|
// registered in any VA space.
|
|
UVM_ASSERT(parent_gpu->isr.access_counters.handling_ref_count == 0);
|
|
|
|
// Return ownership to RM
|
|
uvm_gpu_deinit_isr(parent_gpu);
|
|
|
|
deinit_parent_procfs_files(parent_gpu);
|
|
|
|
uvm_ats_remove_gpu(parent_gpu);
|
|
|
|
UVM_ASSERT(atomic64_read(&parent_gpu->mapped_cpu_pages_size) == 0);
|
|
|
|
// After calling nvUvmInterfaceUnregisterGpu() the reference to pci_dev may
|
|
// not be valid any more so clear it ahead of time.
|
|
parent_gpu->pci_dev = NULL;
|
|
|
|
deinit_parent_procfs_dir(parent_gpu);
|
|
|
|
if (parent_gpu->rm_info.isSimulated)
|
|
--g_uvm_global.num_simulated_devices;
|
|
|
|
if (parent_gpu->rm_device != 0)
|
|
uvm_rm_locked_call_void(nvUvmInterfaceDeviceDestroy(parent_gpu->rm_device));
|
|
|
|
uvm_parent_gpu_kref_put(parent_gpu);
|
|
}
|
|
|
|
static void deinit_gpu(uvm_gpu_t *gpu)
|
|
{
|
|
uvm_gpu_t *other_gpu;
|
|
|
|
// Remove any pointers to this GPU from other GPUs' trackers.
|
|
for_each_global_gpu(other_gpu) {
|
|
UVM_ASSERT(other_gpu != gpu);
|
|
remove_gpus_from_gpu(other_gpu);
|
|
}
|
|
|
|
// Further, remove any pointers to this GPU from its parent's trackers.
|
|
remove_gpu_from_parent_gpu(gpu);
|
|
|
|
uvm_perf_heuristics_remove_gpu(gpu);
|
|
|
|
deinit_procfs_files(gpu);
|
|
|
|
// TODO Bug 3429163: [UVM] Move uvm_mmu_destroy_flat_mapping() to the
|
|
// correct spot
|
|
uvm_mmu_destroy_flat_mappings(gpu);
|
|
|
|
// Wait for any deferred frees and their associated trackers to be finished
|
|
// before tearing down channels.
|
|
uvm_pmm_gpu_sync(&gpu->pmm);
|
|
|
|
uvm_channel_manager_destroy(gpu->channel_manager);
|
|
|
|
// Deconfigure the address space only after destroying all the channels as
|
|
// in case any of them hit fatal errors, RM will assert that they are not
|
|
// idle during nvUvmInterfaceUnsetPageDirectory() and that's an unnecessary
|
|
// pain during development.
|
|
deconfigure_address_space(gpu);
|
|
|
|
deinit_semaphore_pool(gpu);
|
|
|
|
uvm_pmm_sysmem_mappings_deinit(&gpu->pmm_reverse_sysmem_mappings);
|
|
|
|
uvm_pmm_gpu_deinit(&gpu->pmm);
|
|
|
|
if (gpu->rm_address_space != 0)
|
|
uvm_rm_locked_call_void(nvUvmInterfaceAddressSpaceDestroy(gpu->rm_address_space));
|
|
|
|
deinit_procfs_dirs(gpu);
|
|
|
|
if (gpu->parent->smc.enabled) {
|
|
if (gpu->smc.rm_device != 0)
|
|
uvm_rm_locked_call_void(nvUvmInterfaceDeviceDestroy(gpu->smc.rm_device));
|
|
|
|
if (gpu->smc.rm_session_handle != 0)
|
|
uvm_rm_locked_call_void(nvUvmInterfaceSessionDestroy(gpu->smc.rm_session_handle));
|
|
}
|
|
|
|
gpu->magic = 0;
|
|
}
|
|
|
|
// Remove a gpu and unregister it from RM
|
|
// Note that this is also used in most error paths in add_gpu()
|
|
static void remove_gpu(uvm_gpu_t *gpu)
|
|
{
|
|
NvU32 sub_processor_index;
|
|
uvm_parent_gpu_t *parent_gpu;
|
|
bool free_parent;
|
|
|
|
uvm_assert_mutex_locked(&g_uvm_global.global_lock);
|
|
|
|
sub_processor_index = uvm_global_id_sub_processor_index(gpu->global_id);
|
|
parent_gpu = gpu->parent;
|
|
|
|
UVM_ASSERT_MSG(uvm_gpu_retained_count(gpu) == 0,
|
|
"gpu_id %u retained_count %llu\n",
|
|
uvm_id_value(gpu->id),
|
|
uvm_gpu_retained_count(gpu));
|
|
|
|
UVM_ASSERT(parent_gpu->num_retained_gpus > 0);
|
|
parent_gpu->num_retained_gpus--;
|
|
|
|
free_parent = (parent_gpu->num_retained_gpus == 0);
|
|
|
|
// NVLINK peers must be removed and the relevant access counter buffers must
|
|
// be flushed before removing this GPU from the global table. See the
|
|
// comment on discover_nvlink_peers in add_gpu.
|
|
if (free_parent)
|
|
destroy_nvlink_peers(gpu);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// TODO: Bug 2844714: If the parent is not being freed, the following
|
|
// gpu_table_lock is only needed to protect concurrent
|
|
// find_first_valid_gpu() in BH from the __clear_bit here. After
|
|
// find_first_valid_gpu() is removed, gpu_table_lock should only be acquired
|
|
// and released in the free_parent case.
|
|
//
|
|
// In the free_parent case, gpu_table_lock protects the top half from the
|
|
// uvm_global_remove_parent_gpu()
|
|
uvm_spin_lock_irqsave(&g_uvm_global.gpu_table_lock);
|
|
|
|
// Mark the GPU as invalid in the parent GPU's GPU table.
|
|
__clear_bit(sub_processor_index, parent_gpu->valid_gpus);
|
|
|
|
// Remove the GPU from the table.
|
|
if (free_parent)
|
|
uvm_global_remove_parent_gpu(parent_gpu);
|
|
|
|
uvm_spin_unlock_irqrestore(&g_uvm_global.gpu_table_lock);
|
|
|
|
uvm_global_processor_mask_clear(&g_uvm_global.retained_gpus, gpu->global_id);
|
|
|
|
// If the parent is being freed, stop scheduling new bottom halves and
|
|
// update relevant software state. Else flush any pending bottom halves
|
|
// before continuing.
|
|
if (free_parent)
|
|
uvm_gpu_disable_isr(parent_gpu);
|
|
else
|
|
uvm_gpu_flush_bottom_halves(parent_gpu);
|
|
|
|
deinit_gpu(gpu);
|
|
|
|
UVM_ASSERT(parent_gpu->gpus[sub_processor_index] == gpu);
|
|
parent_gpu->gpus[sub_processor_index] = NULL;
|
|
uvm_kvfree(gpu);
|
|
|
|
if (free_parent)
|
|
deinit_parent_gpu(parent_gpu);
|
|
}
|
|
|
|
// Do not not call this directly. It is called by nv_kref_put, when the
|
|
// GPU's ref count drops to zero.
|
|
static void uvm_parent_gpu_destroy(nv_kref_t *nv_kref)
|
|
{
|
|
uvm_parent_gpu_t *parent_gpu = container_of(nv_kref, uvm_parent_gpu_t, gpu_kref);
|
|
NvU32 sub_processor_index;
|
|
|
|
UVM_ASSERT(parent_gpu->num_retained_gpus == 0);
|
|
UVM_ASSERT(bitmap_empty(parent_gpu->valid_gpus, UVM_ID_MAX_SUB_PROCESSORS));
|
|
|
|
for (sub_processor_index = 0; sub_processor_index < UVM_ID_MAX_SUB_PROCESSORS; sub_processor_index++)
|
|
UVM_ASSERT(!parent_gpu->gpus[sub_processor_index]);
|
|
|
|
uvm_kvfree(parent_gpu);
|
|
}
|
|
|
|
void uvm_parent_gpu_kref_put(uvm_parent_gpu_t *parent_gpu)
|
|
{
|
|
nv_kref_put(&parent_gpu->gpu_kref, uvm_parent_gpu_destroy);
|
|
}
|
|
|
|
static void update_stats_gpu_fault_instance(uvm_gpu_t *gpu,
|
|
const uvm_fault_buffer_entry_t *fault_entry,
|
|
bool is_duplicate)
|
|
{
|
|
if (!fault_entry->is_replayable) {
|
|
switch (fault_entry->fault_access_type)
|
|
{
|
|
case UVM_FAULT_ACCESS_TYPE_READ:
|
|
++gpu->parent->fault_buffer_info.non_replayable.stats.num_read_faults;
|
|
break;
|
|
case UVM_FAULT_ACCESS_TYPE_WRITE:
|
|
++gpu->parent->fault_buffer_info.non_replayable.stats.num_write_faults;
|
|
break;
|
|
case UVM_FAULT_ACCESS_TYPE_ATOMIC_WEAK:
|
|
case UVM_FAULT_ACCESS_TYPE_ATOMIC_STRONG:
|
|
++gpu->parent->fault_buffer_info.non_replayable.stats.num_atomic_faults;
|
|
break;
|
|
default:
|
|
UVM_ASSERT_MSG(false, "Invalid access type for non-replayable faults\n");
|
|
break;
|
|
}
|
|
|
|
if (!fault_entry->is_virtual)
|
|
++gpu->parent->fault_buffer_info.non_replayable.stats.num_physical_faults;
|
|
|
|
++gpu->parent->stats.num_non_replayable_faults;
|
|
|
|
return;
|
|
}
|
|
|
|
UVM_ASSERT(fault_entry->is_virtual);
|
|
|
|
switch (fault_entry->fault_access_type)
|
|
{
|
|
case UVM_FAULT_ACCESS_TYPE_PREFETCH:
|
|
++gpu->parent->fault_buffer_info.replayable.stats.num_prefetch_faults;
|
|
break;
|
|
case UVM_FAULT_ACCESS_TYPE_READ:
|
|
++gpu->parent->fault_buffer_info.replayable.stats.num_read_faults;
|
|
break;
|
|
case UVM_FAULT_ACCESS_TYPE_WRITE:
|
|
++gpu->parent->fault_buffer_info.replayable.stats.num_write_faults;
|
|
break;
|
|
case UVM_FAULT_ACCESS_TYPE_ATOMIC_WEAK:
|
|
case UVM_FAULT_ACCESS_TYPE_ATOMIC_STRONG:
|
|
++gpu->parent->fault_buffer_info.replayable.stats.num_atomic_faults;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
if (is_duplicate || fault_entry->filtered)
|
|
++gpu->parent->fault_buffer_info.replayable.stats.num_duplicate_faults;
|
|
|
|
++gpu->parent->stats.num_replayable_faults;
|
|
}
|
|
|
|
static void update_stats_fault_cb(uvm_perf_event_t event_id, uvm_perf_event_data_t *event_data)
|
|
{
|
|
uvm_gpu_t *gpu;
|
|
const uvm_fault_buffer_entry_t *fault_entry, *fault_instance;
|
|
|
|
UVM_ASSERT(event_id == UVM_PERF_EVENT_FAULT);
|
|
|
|
if (UVM_ID_IS_CPU(event_data->fault.proc_id))
|
|
return;
|
|
|
|
// The reported fault entry must be the "representative" fault entry
|
|
UVM_ASSERT(!event_data->fault.gpu.buffer_entry->filtered);
|
|
|
|
gpu = uvm_va_space_get_gpu(event_data->fault.space, event_data->fault.proc_id);
|
|
|
|
fault_entry = event_data->fault.gpu.buffer_entry;
|
|
|
|
// Update the stats using the representative fault entry and the rest of
|
|
// instances
|
|
update_stats_gpu_fault_instance(gpu, fault_entry, event_data->fault.gpu.is_duplicate);
|
|
|
|
list_for_each_entry(fault_instance, &fault_entry->merged_instances_list, merged_instances_list)
|
|
update_stats_gpu_fault_instance(gpu, fault_instance, event_data->fault.gpu.is_duplicate);
|
|
}
|
|
|
|
static void update_stats_migration_cb(uvm_perf_event_t event_id, uvm_perf_event_data_t *event_data)
|
|
{
|
|
uvm_gpu_t *gpu_dst = NULL;
|
|
uvm_gpu_t *gpu_src = NULL;
|
|
NvU64 pages;
|
|
bool is_replayable_fault;
|
|
bool is_non_replayable_fault;
|
|
bool is_access_counter;
|
|
uvm_va_space_t *va_space = uvm_va_block_get_va_space(event_data->migration.block);
|
|
|
|
UVM_ASSERT(event_id == UVM_PERF_EVENT_MIGRATION);
|
|
|
|
if (UVM_ID_IS_GPU(event_data->migration.dst))
|
|
gpu_dst = uvm_va_space_get_gpu(va_space, event_data->migration.dst);
|
|
|
|
if (UVM_ID_IS_GPU(event_data->migration.src))
|
|
gpu_src = uvm_va_space_get_gpu(va_space, event_data->migration.src);
|
|
|
|
if (!gpu_dst && !gpu_src)
|
|
return;
|
|
|
|
// Page prefetching is also triggered by faults
|
|
is_replayable_fault =
|
|
event_data->migration.make_resident_context->cause == UVM_MAKE_RESIDENT_CAUSE_REPLAYABLE_FAULT;
|
|
is_non_replayable_fault =
|
|
event_data->migration.make_resident_context->cause == UVM_MAKE_RESIDENT_CAUSE_NON_REPLAYABLE_FAULT;
|
|
is_access_counter =
|
|
event_data->migration.make_resident_context->cause == UVM_MAKE_RESIDENT_CAUSE_ACCESS_COUNTER;
|
|
|
|
pages = event_data->migration.bytes / PAGE_SIZE;
|
|
UVM_ASSERT(event_data->migration.bytes % PAGE_SIZE == 0);
|
|
UVM_ASSERT(pages > 0);
|
|
|
|
if (gpu_dst) {
|
|
atomic64_add(pages, &gpu_dst->parent->stats.num_pages_in);
|
|
if (is_replayable_fault)
|
|
atomic64_add(pages, &gpu_dst->parent->fault_buffer_info.replayable.stats.num_pages_in);
|
|
else if (is_non_replayable_fault)
|
|
atomic64_add(pages, &gpu_dst->parent->fault_buffer_info.non_replayable.stats.num_pages_in);
|
|
else if (is_access_counter)
|
|
atomic64_add(pages, &gpu_dst->parent->access_counter_buffer_info.stats.num_pages_in);
|
|
}
|
|
if (gpu_src) {
|
|
atomic64_add(pages, &gpu_src->parent->stats.num_pages_out);
|
|
if (is_replayable_fault)
|
|
atomic64_add(pages, &gpu_src->parent->fault_buffer_info.replayable.stats.num_pages_out);
|
|
else if (is_non_replayable_fault)
|
|
atomic64_add(pages, &gpu_src->parent->fault_buffer_info.non_replayable.stats.num_pages_out);
|
|
else if (is_access_counter)
|
|
atomic64_add(pages, &gpu_src->parent->access_counter_buffer_info.stats.num_pages_out);
|
|
}
|
|
}
|
|
|
|
// Override the UVM driver and GPU settings from the module loader
|
|
static void uvm_param_conf(void)
|
|
{
|
|
// uvm_peer_copy: Valid entries are "phys" and "virt" for Ampere+ GPUs.
|
|
// No effect in pre-Ampere GPUs
|
|
if (strcmp(uvm_peer_copy, UVM_PARAM_PEER_COPY_VIRTUAL) == 0) {
|
|
g_uvm_global.peer_copy_mode = UVM_GPU_PEER_COPY_MODE_VIRTUAL;
|
|
}
|
|
else {
|
|
if (strcmp(uvm_peer_copy, UVM_PARAM_PEER_COPY_PHYSICAL) != 0) {
|
|
pr_info("Invalid value for uvm_peer_copy = %s, using %s instead.\n",
|
|
uvm_peer_copy, UVM_PARAM_PEER_COPY_PHYSICAL);
|
|
}
|
|
|
|
g_uvm_global.peer_copy_mode = UVM_GPU_PEER_COPY_MODE_PHYSICAL;
|
|
}
|
|
}
|
|
|
|
NV_STATUS uvm_gpu_init(void)
|
|
{
|
|
NV_STATUS status;
|
|
|
|
uvm_param_conf();
|
|
|
|
status = uvm_hal_init_table();
|
|
if (status != NV_OK) {
|
|
UVM_ERR_PRINT("uvm_hal_init_table() failed: %s\n", nvstatusToString(status));
|
|
return status;
|
|
}
|
|
|
|
return NV_OK;
|
|
}
|
|
|
|
void uvm_gpu_exit(void)
|
|
{
|
|
uvm_parent_gpu_t *parent_gpu;
|
|
|
|
for_each_parent_gpu(parent_gpu)
|
|
UVM_ASSERT_MSG(false, "GPU still present: %s\n", parent_gpu->name);
|
|
|
|
// CPU should never be in the retained GPUs mask
|
|
UVM_ASSERT(!uvm_global_processor_mask_test(&g_uvm_global.retained_gpus, UVM_GLOBAL_ID_CPU));
|
|
}
|
|
|
|
NV_STATUS uvm_gpu_init_va_space(uvm_va_space_t *va_space)
|
|
{
|
|
NV_STATUS status;
|
|
|
|
if (uvm_procfs_is_debug_enabled()) {
|
|
status = uvm_perf_register_event_callback(&va_space->perf_events,
|
|
UVM_PERF_EVENT_FAULT,
|
|
update_stats_fault_cb);
|
|
if (status != NV_OK)
|
|
return status;
|
|
|
|
status = uvm_perf_register_event_callback(&va_space->perf_events,
|
|
UVM_PERF_EVENT_MIGRATION,
|
|
update_stats_migration_cb);
|
|
if (status != NV_OK)
|
|
return status;
|
|
}
|
|
|
|
return NV_OK;
|
|
}
|
|
|
|
uvm_parent_gpu_t *uvm_parent_gpu_get_by_uuid_locked(const NvProcessorUuid *gpu_uuid)
|
|
{
|
|
uvm_parent_gpu_t *parent_gpu;
|
|
|
|
for_each_parent_gpu(parent_gpu) {
|
|
if (uvm_processor_uuid_eq(&parent_gpu->uuid, gpu_uuid))
|
|
return parent_gpu;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
uvm_parent_gpu_t *uvm_parent_gpu_get_by_uuid(const NvProcessorUuid *gpu_uuid)
|
|
{
|
|
uvm_assert_mutex_locked(&g_uvm_global.global_lock);
|
|
|
|
return uvm_parent_gpu_get_by_uuid_locked(gpu_uuid);
|
|
}
|
|
|
|
static uvm_gpu_t *uvm_gpu_get_by_uuid_locked(const NvProcessorUuid *gpu_uuid)
|
|
{
|
|
uvm_gpu_id_t gpu_id;
|
|
uvm_global_gpu_id_t global_gpu_id;
|
|
uvm_gpu_t *gpu;
|
|
|
|
for_each_gpu_id(gpu_id) {
|
|
global_gpu_id = uvm_global_gpu_id_from_gpu_id(gpu_id);
|
|
gpu = uvm_gpu_get(global_gpu_id);
|
|
if (gpu) {
|
|
if (uvm_processor_uuid_eq(uvm_gpu_uuid(gpu), gpu_uuid)) {
|
|
UVM_ASSERT(!gpu->parent->smc.enabled);
|
|
return gpu;
|
|
}
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
uvm_gpu_t *uvm_gpu_get_by_uuid(const NvProcessorUuid *gpu_uuid)
|
|
{
|
|
uvm_assert_mutex_locked(&g_uvm_global.global_lock);
|
|
|
|
return uvm_gpu_get_by_uuid_locked(gpu_uuid);
|
|
}
|
|
|
|
uvm_gpu_t *uvm_gpu_get_by_parent_and_swizz_id_locked(uvm_parent_gpu_t *parent_gpu, NvU32 swizz_id)
|
|
{
|
|
uvm_gpu_t *gpu;
|
|
|
|
UVM_ASSERT(parent_gpu);
|
|
|
|
for_each_gpu_in_parent(parent_gpu, gpu) {
|
|
if (gpu->smc.swizz_id == swizz_id)
|
|
return gpu;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
uvm_gpu_t *uvm_gpu_get_by_parent_and_swizz_id(uvm_parent_gpu_t *parent_gpu, NvU32 swizz_id)
|
|
{
|
|
uvm_assert_mutex_locked(&g_uvm_global.global_lock);
|
|
|
|
return uvm_gpu_get_by_parent_and_swizz_id_locked(parent_gpu, swizz_id);
|
|
}
|
|
|
|
// Increment the refcount for the GPU with the given UUID. If this is the first
|
|
// time that this UUID is retained, the GPU is added to UVM.
|
|
// When SMC partitioning is enabled, user_rm_device contains the user handles
|
|
// that were created by the caller, and that can be used to identify and
|
|
// obtain information about the partition. nvUvmInterfaceGetGpuInfo returns, in
|
|
// gpu_info, whether SMC is enabled and the swizzId corresponding to the
|
|
// partition.
|
|
static NV_STATUS gpu_retain_by_uuid_locked(const NvProcessorUuid *gpu_uuid,
|
|
const uvm_rm_user_object_t *user_rm_device,
|
|
uvm_gpu_t **gpu_out)
|
|
{
|
|
NV_STATUS status = NV_OK;
|
|
uvm_gpu_t *gpu = NULL;
|
|
uvm_parent_gpu_t *parent_gpu;
|
|
UvmGpuInfo *gpu_info = NULL;
|
|
UvmGpuClientInfo client_info = {0};
|
|
UvmGpuPlatformInfo gpu_platform_info = {0};
|
|
uvm_global_gpu_id_t global_gpu_id;
|
|
|
|
client_info.hClient = user_rm_device->user_client;
|
|
client_info.hSmcPartRef = user_rm_device->user_object;
|
|
|
|
gpu_info = uvm_kvmalloc_zero(sizeof(*gpu_info));
|
|
if (!gpu_info)
|
|
return NV_ERR_NO_MEMORY;
|
|
|
|
uvm_assert_mutex_locked(&g_uvm_global.global_lock);
|
|
|
|
parent_gpu = uvm_parent_gpu_get_by_uuid(gpu_uuid);
|
|
|
|
if (parent_gpu == NULL) {
|
|
// If this is the first time the UUID is seen, register it on RM
|
|
status = uvm_rm_locked_call(nvUvmInterfaceRegisterGpu(gpu_uuid, &gpu_platform_info));
|
|
if (status != NV_OK)
|
|
goto error_free_gpu_info;
|
|
}
|
|
|
|
status = uvm_rm_locked_call(nvUvmInterfaceGetGpuInfo(gpu_uuid, &client_info, gpu_info));
|
|
if (status != NV_OK)
|
|
goto error_unregister;
|
|
|
|
if (parent_gpu != NULL) {
|
|
// If the UUID has been seen before, and if SMC is enabled, then check
|
|
// if this specific partition has been seen previously. The UUID-based
|
|
// look-up above may have succeeded for a different partition with the
|
|
// same parent GPU.
|
|
if (gpu_info->smcEnabled) {
|
|
gpu = uvm_gpu_get_by_parent_and_swizz_id(parent_gpu, gpu_info->smcSwizzId);
|
|
}
|
|
else {
|
|
gpu = parent_gpu->gpus[0];
|
|
UVM_ASSERT(gpu != NULL);
|
|
}
|
|
}
|
|
|
|
if (gpu == NULL) {
|
|
status = find_unused_global_gpu_id(parent_gpu, &global_gpu_id);
|
|
if (status != NV_OK)
|
|
goto error_unregister;
|
|
|
|
status = add_gpu(gpu_uuid, global_gpu_id, gpu_info, &gpu_platform_info, parent_gpu, &gpu);
|
|
if (status != NV_OK)
|
|
goto error_unregister;
|
|
}
|
|
else {
|
|
atomic64_inc(&gpu->retained_count);
|
|
}
|
|
|
|
*gpu_out = gpu;
|
|
|
|
uvm_kvfree(gpu_info);
|
|
|
|
return status;
|
|
|
|
error_unregister:
|
|
if (parent_gpu == NULL)
|
|
uvm_rm_locked_call_void(nvUvmInterfaceUnregisterGpu(gpu_uuid));
|
|
error_free_gpu_info:
|
|
uvm_kvfree(gpu_info);
|
|
|
|
return status;
|
|
}
|
|
|
|
NV_STATUS uvm_gpu_retain_by_uuid(const NvProcessorUuid *gpu_uuid,
|
|
const uvm_rm_user_object_t *user_rm_device,
|
|
uvm_gpu_t **gpu_out)
|
|
{
|
|
NV_STATUS status;
|
|
uvm_mutex_lock(&g_uvm_global.global_lock);
|
|
status = gpu_retain_by_uuid_locked(gpu_uuid, user_rm_device, gpu_out);
|
|
uvm_mutex_unlock(&g_uvm_global.global_lock);
|
|
return status;
|
|
}
|
|
|
|
void uvm_gpu_retain(uvm_gpu_t *gpu)
|
|
{
|
|
UVM_ASSERT(uvm_gpu_retained_count(gpu) > 0);
|
|
atomic64_inc(&gpu->retained_count);
|
|
}
|
|
|
|
void uvm_gpu_release_locked(uvm_gpu_t *gpu)
|
|
{
|
|
uvm_parent_gpu_t *parent_gpu = gpu->parent;
|
|
|
|
uvm_assert_mutex_locked(&g_uvm_global.global_lock);
|
|
UVM_ASSERT(uvm_gpu_retained_count(gpu) > 0);
|
|
|
|
if (atomic64_dec_and_test(&gpu->retained_count)) {
|
|
nv_kref_get(&parent_gpu->gpu_kref);
|
|
remove_gpu(gpu);
|
|
if (parent_gpu->num_retained_gpus == 0)
|
|
uvm_rm_locked_call_void(nvUvmInterfaceUnregisterGpu(&parent_gpu->uuid));
|
|
uvm_parent_gpu_kref_put(parent_gpu);
|
|
}
|
|
}
|
|
|
|
void uvm_gpu_release(uvm_gpu_t *gpu)
|
|
{
|
|
uvm_mutex_lock(&g_uvm_global.global_lock);
|
|
uvm_gpu_release_locked(gpu);
|
|
uvm_mutex_unlock(&g_uvm_global.global_lock);
|
|
}
|
|
|
|
// Note: Peer table is an upper triangular matrix packed into a flat array.
|
|
// This function converts an index of 2D array of size [N x N] into an index
|
|
// of upper triangular array of size [((N - 1) * ((N - 1) + 1)) / 2] which
|
|
// does not include diagonal elements.
|
|
NvU32 uvm_gpu_peer_table_index(const uvm_gpu_id_t gpu_id0, const uvm_gpu_id_t gpu_id1)
|
|
{
|
|
NvU32 square_index, triangular_index;
|
|
NvU32 gpu_index0 = uvm_id_gpu_index(gpu_id0);
|
|
NvU32 gpu_index1 = uvm_id_gpu_index(gpu_id1);
|
|
|
|
UVM_ASSERT(!uvm_id_equal(gpu_id0, gpu_id1));
|
|
|
|
// Calculate an index of 2D array by re-ordering indices to always point
|
|
// to the same entry.
|
|
square_index = min(gpu_index0, gpu_index1) * UVM_ID_MAX_GPUS +
|
|
max(gpu_index0, gpu_index1);
|
|
|
|
// Calculate and subtract number of lower triangular matrix elements till
|
|
// the current row (which includes diagonal elements) to get the correct
|
|
// index in an upper triangular matrix.
|
|
// Note: As gpu_id can be [1, N), no extra logic is needed to calculate
|
|
// diagonal elements.
|
|
triangular_index = square_index - SUM_FROM_0_TO_N(min(uvm_id_value(gpu_id0), uvm_id_value(gpu_id1)));
|
|
|
|
UVM_ASSERT(triangular_index < UVM_MAX_UNIQUE_GPU_PAIRS);
|
|
|
|
return triangular_index;
|
|
}
|
|
|
|
NV_STATUS uvm_gpu_check_ecc_error_no_rm(uvm_gpu_t *gpu)
|
|
{
|
|
// We may need to call service_interrupts() which cannot be done in the top
|
|
// half interrupt handler so assert here as well to catch improper use as
|
|
// early as possible.
|
|
UVM_ASSERT(!in_interrupt());
|
|
|
|
if (!gpu->ecc.enabled)
|
|
return NV_OK;
|
|
|
|
// Early out If a global ECC error is already set to not spam the logs with
|
|
// the same error.
|
|
if (uvm_global_get_status() == NV_ERR_ECC_ERROR)
|
|
return NV_ERR_ECC_ERROR;
|
|
|
|
if (*gpu->ecc.error_notifier) {
|
|
UVM_ERR_PRINT("ECC error encountered, GPU %s\n", uvm_gpu_name(gpu));
|
|
uvm_global_set_fatal_error(NV_ERR_ECC_ERROR);
|
|
return NV_ERR_ECC_ERROR;
|
|
}
|
|
|
|
// RM hasn't seen an ECC error yet, check whether there is a pending
|
|
// interrupt that might indicate one. We might get false positives because
|
|
// the interrupt bits we read are not ECC-specific. They're just the
|
|
// top-level bits for any interrupt on all engines which support ECC. On
|
|
// Pascal for example, RM returns us a mask with the bits for GR, L2, and
|
|
// FB, because any of those might raise an ECC interrupt. So if they're set
|
|
// we have to ask RM to check whether it was really an ECC error (and a
|
|
// double-bit ECC error at that), in which case it sets the notifier.
|
|
if ((*gpu->ecc.hw_interrupt_tree_location & gpu->ecc.mask) == 0) {
|
|
// No pending interrupts.
|
|
return NV_OK;
|
|
}
|
|
|
|
// An interrupt that might mean an ECC error needs to be serviced, signal
|
|
// that to the caller.
|
|
return NV_WARN_MORE_PROCESSING_REQUIRED;
|
|
}
|
|
|
|
static NV_STATUS get_p2p_caps(uvm_gpu_t *gpu0,
|
|
uvm_gpu_t *gpu1,
|
|
UvmGpuP2PCapsParams *p2p_caps_params)
|
|
{
|
|
NV_STATUS status;
|
|
uvmGpuDeviceHandle rm_device0, rm_device1;
|
|
|
|
if (uvm_id_value(gpu0->id) < uvm_id_value(gpu1->id)) {
|
|
rm_device0 = uvm_gpu_device_handle(gpu0);
|
|
rm_device1 = uvm_gpu_device_handle(gpu1);
|
|
}
|
|
else {
|
|
rm_device0 = uvm_gpu_device_handle(gpu1);
|
|
rm_device1 = uvm_gpu_device_handle(gpu0);
|
|
}
|
|
|
|
memset(p2p_caps_params, 0, sizeof(*p2p_caps_params));
|
|
status = uvm_rm_locked_call(nvUvmInterfaceGetP2PCaps(rm_device0, rm_device1, p2p_caps_params));
|
|
if (status != NV_OK) {
|
|
UVM_ERR_PRINT("nvUvmInterfaceGetP2PCaps() failed with error: %s, for GPU0:%s and GPU1:%s\n",
|
|
nvstatusToString(status),
|
|
uvm_gpu_name(gpu0),
|
|
uvm_gpu_name(gpu1));
|
|
return status;
|
|
}
|
|
|
|
if (p2p_caps_params->p2pLink != UVM_LINK_TYPE_NONE) {
|
|
// P2P is not supported under SMC partitioning
|
|
UVM_ASSERT(!gpu0->parent->smc.enabled);
|
|
UVM_ASSERT(!gpu1->parent->smc.enabled);
|
|
}
|
|
|
|
return NV_OK;
|
|
}
|
|
|
|
static NV_STATUS create_p2p_object(uvm_gpu_t *gpu0, uvm_gpu_t *gpu1, NvHandle *p2p_handle)
|
|
{
|
|
NV_STATUS status;
|
|
uvmGpuDeviceHandle rm_device0, rm_device1;
|
|
|
|
if (uvm_id_value(gpu0->id) < uvm_id_value(gpu1->id)) {
|
|
rm_device0 = uvm_gpu_device_handle(gpu0);
|
|
rm_device1 = uvm_gpu_device_handle(gpu1);
|
|
}
|
|
else {
|
|
rm_device0 = uvm_gpu_device_handle(gpu1);
|
|
rm_device1 = uvm_gpu_device_handle(gpu0);
|
|
}
|
|
|
|
*p2p_handle = 0;
|
|
|
|
status = uvm_rm_locked_call(nvUvmInterfaceP2pObjectCreate(rm_device0, rm_device1, p2p_handle));
|
|
if (status != NV_OK) {
|
|
UVM_ERR_PRINT("nvUvmInterfaceP2pObjectCreate() failed with error: %s, for GPU0:%s and GPU1:%s\n",
|
|
nvstatusToString(status),
|
|
uvm_gpu_name(gpu0),
|
|
uvm_gpu_name(gpu1));
|
|
return status;
|
|
}
|
|
|
|
UVM_ASSERT(*p2p_handle);
|
|
return NV_OK;
|
|
}
|
|
|
|
static void set_optimal_p2p_write_ces(const UvmGpuP2PCapsParams *p2p_caps_params,
|
|
const uvm_gpu_peer_t *peer_caps,
|
|
uvm_gpu_t *gpu0,
|
|
uvm_gpu_t *gpu1)
|
|
{
|
|
bool sorted;
|
|
NvU32 ce0, ce1;
|
|
|
|
if (peer_caps->link_type < UVM_GPU_LINK_NVLINK_1)
|
|
return;
|
|
|
|
sorted = uvm_id_value(gpu0->id) < uvm_id_value(gpu1->id);
|
|
ce0 = p2p_caps_params->optimalNvlinkWriteCEs[sorted ? 0 : 1];
|
|
ce1 = p2p_caps_params->optimalNvlinkWriteCEs[sorted ? 1 : 0];
|
|
|
|
// Indirect peers communicate through the CPU, so the optimal CE
|
|
// should match the one selected for writing to system memory
|
|
if (peer_caps->is_indirect_peer) {
|
|
uvm_channel_pool_t *pool;
|
|
|
|
pool = gpu0->channel_manager->pool_to_use.default_for_type[UVM_CHANNEL_TYPE_GPU_TO_CPU];
|
|
UVM_ASSERT(ce0 == pool->engine_index);
|
|
|
|
pool = gpu1->channel_manager->pool_to_use.default_for_type[UVM_CHANNEL_TYPE_GPU_TO_CPU];
|
|
UVM_ASSERT(ce1 == pool->engine_index);
|
|
}
|
|
|
|
uvm_channel_manager_set_p2p_ce(gpu0->channel_manager, gpu1, ce0);
|
|
uvm_channel_manager_set_p2p_ce(gpu1->channel_manager, gpu0, ce1);
|
|
}
|
|
|
|
static int nv_procfs_read_gpu_peer_caps(struct seq_file *s, void *v)
|
|
{
|
|
if (!uvm_down_read_trylock(&g_uvm_global.pm.lock))
|
|
return -EAGAIN;
|
|
|
|
gpu_peer_caps_print((uvm_gpu_t **)s->private, s);
|
|
|
|
uvm_up_read(&g_uvm_global.pm.lock);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int nv_procfs_read_gpu_peer_caps_entry(struct seq_file *s, void *v)
|
|
{
|
|
UVM_ENTRY_RET(nv_procfs_read_gpu_peer_caps(s, v));
|
|
}
|
|
|
|
UVM_DEFINE_SINGLE_PROCFS_FILE(gpu_peer_caps_entry);
|
|
|
|
static NV_STATUS init_procfs_peer_cap_files(uvm_gpu_t *local, uvm_gpu_t *remote, size_t local_idx)
|
|
{
|
|
// This needs to hold a gpu_id_t in decimal
|
|
char gpu_dir_name[16];
|
|
|
|
// This needs to hold a GPU UUID
|
|
char symlink_name[UVM_GPU_UUID_TEXT_BUFFER_LENGTH];
|
|
uvm_gpu_peer_t *peer_caps;
|
|
|
|
if (!uvm_procfs_is_enabled())
|
|
return NV_OK;
|
|
|
|
peer_caps = uvm_gpu_peer_caps(local, remote);
|
|
peer_caps->procfs.pairs[local_idx][0] = local;
|
|
peer_caps->procfs.pairs[local_idx][1] = remote;
|
|
|
|
// Create gpus/gpuA/peers/gpuB
|
|
snprintf(gpu_dir_name, sizeof(gpu_dir_name), "%u", uvm_id_value(remote->id));
|
|
peer_caps->procfs.peer_file[local_idx] = NV_CREATE_PROC_FILE(gpu_dir_name,
|
|
local->procfs.dir_peers,
|
|
gpu_peer_caps_entry,
|
|
&peer_caps->procfs.pairs[local_idx]);
|
|
|
|
if (peer_caps->procfs.peer_file[local_idx] == NULL)
|
|
return NV_ERR_OPERATING_SYSTEM;
|
|
|
|
// Create a symlink from UVM GPU UUID (UVM-GPU-...) to the UVM GPU ID gpuB
|
|
format_uuid_to_buffer(symlink_name, sizeof(symlink_name), uvm_gpu_uuid(remote));
|
|
peer_caps->procfs.peer_symlink_file[local_idx] = proc_symlink(symlink_name,
|
|
local->procfs.dir_peers,
|
|
gpu_dir_name);
|
|
if (peer_caps->procfs.peer_symlink_file[local_idx] == NULL)
|
|
return NV_ERR_OPERATING_SYSTEM;
|
|
|
|
return NV_OK;
|
|
}
|
|
|
|
static NV_STATUS init_peer_access(uvm_gpu_t *gpu0,
|
|
uvm_gpu_t *gpu1,
|
|
const UvmGpuP2PCapsParams *p2p_caps_params,
|
|
uvm_gpu_peer_t *peer_caps)
|
|
{
|
|
NV_STATUS status;
|
|
|
|
|
|
|
|
|
|
|
|
// check for peer-to-peer compatibility (PCI-E or NvLink).
|
|
peer_caps->link_type = get_gpu_link_type(p2p_caps_params->p2pLink);
|
|
if (peer_caps->link_type == UVM_GPU_LINK_INVALID
|
|
|
|
|
|
|
|
)
|
|
return NV_ERR_NOT_SUPPORTED;
|
|
|
|
peer_caps->total_link_line_rate_mbyte_per_s = p2p_caps_params->totalLinkLineRateMBps;
|
|
|
|
// Initialize peer ids and establish peer mappings
|
|
peer_caps->is_indirect_peer = (p2p_caps_params->indirectAccess == NV_TRUE);
|
|
|
|
if (peer_caps->is_indirect_peer) {
|
|
UVM_ASSERT(gpu0->parent->numa_info.enabled);
|
|
UVM_ASSERT(gpu1->parent->numa_info.enabled);
|
|
|
|
status = uvm_pmm_gpu_indirect_peer_init(&gpu0->pmm, gpu1);
|
|
if (status != NV_OK)
|
|
return status;
|
|
|
|
status = uvm_pmm_gpu_indirect_peer_init(&gpu1->pmm, gpu0);
|
|
if (status != NV_OK)
|
|
return status;
|
|
|
|
set_optimal_p2p_write_ces(p2p_caps_params, peer_caps, gpu0, gpu1);
|
|
UVM_ASSERT(peer_caps->total_link_line_rate_mbyte_per_s == 0);
|
|
}
|
|
else {
|
|
// Peer id from min(gpu_id0, gpu_id1) -> max(gpu_id0, gpu_id1)
|
|
peer_caps->peer_ids[0] = p2p_caps_params->peerIds[0];
|
|
|
|
// Peer id from max(gpu_id0, gpu_id1) -> min(gpu_id0, gpu_id1)
|
|
peer_caps->peer_ids[1] = p2p_caps_params->peerIds[1];
|
|
|
|
// Establish peer mappings from each GPU to the other. Indirect peers
|
|
// do not require identity mappings since they use sysmem aperture to
|
|
// communicate.
|
|
status = uvm_mmu_create_peer_identity_mappings(gpu0, gpu1);
|
|
if (status != NV_OK)
|
|
return status;
|
|
|
|
status = uvm_mmu_create_peer_identity_mappings(gpu1, gpu0);
|
|
if (status != NV_OK)
|
|
return status;
|
|
|
|
set_optimal_p2p_write_ces(p2p_caps_params, peer_caps, gpu0, gpu1);
|
|
|
|
UVM_ASSERT(uvm_gpu_get(gpu0->global_id) == gpu0);
|
|
UVM_ASSERT(uvm_gpu_get(gpu1->global_id) == gpu1);
|
|
|
|
// In the case of NVLINK peers, this initialization will happen during
|
|
// add_gpu. As soon as the peer info table is assigned below, the access
|
|
// counter bottom half could start operating on the GPU being newly
|
|
// added and inspecting the peer caps, so all of the appropriate
|
|
// initialization must happen before this point.
|
|
uvm_spin_lock(&gpu0->peer_info.peer_gpus_lock);
|
|
|
|
uvm_processor_mask_set(&gpu0->peer_info.peer_gpu_mask, gpu1->id);
|
|
UVM_ASSERT(gpu0->peer_info.peer_gpus[uvm_id_gpu_index(gpu1->id)] == NULL);
|
|
gpu0->peer_info.peer_gpus[uvm_id_gpu_index(gpu1->id)] = gpu1;
|
|
|
|
uvm_spin_unlock(&gpu0->peer_info.peer_gpus_lock);
|
|
uvm_spin_lock(&gpu1->peer_info.peer_gpus_lock);
|
|
|
|
uvm_processor_mask_set(&gpu1->peer_info.peer_gpu_mask, gpu0->id);
|
|
UVM_ASSERT(gpu1->peer_info.peer_gpus[uvm_id_gpu_index(gpu0->id)] == NULL);
|
|
gpu1->peer_info.peer_gpus[uvm_id_gpu_index(gpu0->id)] = gpu0;
|
|
|
|
uvm_spin_unlock(&gpu1->peer_info.peer_gpus_lock);
|
|
}
|
|
|
|
if (!uvm_procfs_is_debug_enabled())
|
|
return NV_OK;
|
|
|
|
status = init_procfs_peer_cap_files(gpu0, gpu1, 0);
|
|
if (status != NV_OK)
|
|
return status;
|
|
|
|
status = init_procfs_peer_cap_files(gpu1, gpu0, 1);
|
|
if (status != NV_OK)
|
|
return status;
|
|
|
|
return NV_OK;
|
|
}
|
|
|
|
static NV_STATUS enable_pcie_peer_access(uvm_gpu_t *gpu0, uvm_gpu_t *gpu1)
|
|
{
|
|
NV_STATUS status = NV_OK;
|
|
UvmGpuP2PCapsParams p2p_caps_params;
|
|
uvm_gpu_peer_t *peer_caps;
|
|
NvHandle p2p_handle;
|
|
|
|
UVM_ASSERT(gpu0);
|
|
UVM_ASSERT(gpu1);
|
|
uvm_assert_mutex_locked(&g_uvm_global.global_lock);
|
|
|
|
peer_caps = uvm_gpu_peer_caps(gpu0, gpu1);
|
|
UVM_ASSERT(peer_caps->link_type == UVM_GPU_LINK_INVALID);
|
|
UVM_ASSERT(peer_caps->ref_count == 0);
|
|
|
|
status = create_p2p_object(gpu0, gpu1, &p2p_handle);
|
|
if (status != NV_OK)
|
|
return status;
|
|
|
|
// Store the handle in the global table.
|
|
peer_caps->p2p_handle = p2p_handle;
|
|
|
|
status = get_p2p_caps(gpu0, gpu1, &p2p_caps_params);
|
|
if (status != NV_OK)
|
|
goto cleanup;
|
|
|
|
// Sanity checks
|
|
UVM_ASSERT(p2p_caps_params.indirectAccess == NV_FALSE);
|
|
UVM_ASSERT(p2p_caps_params.p2pLink == UVM_LINK_TYPE_PCIE);
|
|
|
|
status = init_peer_access(gpu0, gpu1, &p2p_caps_params, peer_caps);
|
|
if (status != NV_OK)
|
|
goto cleanup;
|
|
|
|
return NV_OK;
|
|
|
|
cleanup:
|
|
disable_peer_access(gpu0, gpu1);
|
|
return status;
|
|
}
|
|
|
|
static NV_STATUS enable_nvlink_peer_access(uvm_gpu_t *gpu0,
|
|
uvm_gpu_t *gpu1,
|
|
UvmGpuP2PCapsParams *p2p_caps_params)
|
|
{
|
|
NV_STATUS status = NV_OK;
|
|
NvHandle p2p_handle;
|
|
uvm_gpu_peer_t *peer_caps;
|
|
|
|
UVM_ASSERT(gpu0);
|
|
UVM_ASSERT(gpu1);
|
|
uvm_assert_mutex_locked(&g_uvm_global.global_lock);
|
|
|
|
peer_caps = uvm_gpu_peer_caps(gpu0, gpu1);
|
|
UVM_ASSERT(peer_caps->ref_count == 0);
|
|
peer_caps->ref_count = 1;
|
|
|
|
if (!p2p_caps_params->indirectAccess) {
|
|
// Create P2P object for direct NVLink peers
|
|
status = create_p2p_object(gpu0, gpu1, &p2p_handle);
|
|
if (status != NV_OK) {
|
|
UVM_ERR_PRINT("failed to create a P2P object with error: %s, for GPU1:%s and GPU2:%s \n",
|
|
nvstatusToString(status),
|
|
uvm_gpu_name(gpu0),
|
|
uvm_gpu_name(gpu1));
|
|
return status;
|
|
}
|
|
|
|
UVM_ASSERT(p2p_handle != 0);
|
|
|
|
// Store the handle in the global table.
|
|
peer_caps->p2p_handle = p2p_handle;
|
|
|
|
// Update p2p caps after p2p object creation as it generates the peer
|
|
// ids
|
|
status = get_p2p_caps(gpu0, gpu1, p2p_caps_params);
|
|
if (status != NV_OK)
|
|
goto cleanup;
|
|
}
|
|
|
|
status = init_peer_access(gpu0, gpu1, p2p_caps_params, peer_caps);
|
|
if (status != NV_OK)
|
|
goto cleanup;
|
|
|
|
return NV_OK;
|
|
|
|
cleanup:
|
|
disable_peer_access(gpu0, gpu1);
|
|
return status;
|
|
}
|
|
|
|
static NV_STATUS discover_nvlink_peers(uvm_gpu_t *gpu)
|
|
{
|
|
NV_STATUS status = NV_OK;
|
|
uvm_gpu_t *other_gpu;
|
|
|
|
UVM_ASSERT(gpu);
|
|
uvm_assert_mutex_locked(&g_uvm_global.global_lock);
|
|
|
|
if (gpu->parent->smc.enabled)
|
|
return NV_OK;
|
|
|
|
for_each_global_gpu(other_gpu) {
|
|
UvmGpuP2PCapsParams p2p_caps_params;
|
|
|
|
if ((other_gpu == gpu) || other_gpu->parent->smc.enabled)
|
|
continue;
|
|
|
|
status = get_p2p_caps(gpu, other_gpu, &p2p_caps_params);
|
|
if (status != NV_OK)
|
|
goto cleanup;
|
|
|
|
// PCIe peers need to be explicitly enabled via UvmEnablePeerAccess
|
|
if (p2p_caps_params.p2pLink == UVM_LINK_TYPE_NONE || p2p_caps_params.p2pLink == UVM_LINK_TYPE_PCIE)
|
|
continue;
|
|
|
|
// Indirect peers are only supported when onlined as NUMA nodes, because
|
|
// we want to use vm_insert_page and dma_map_page.
|
|
if (p2p_caps_params.indirectAccess &&
|
|
(!gpu->parent->numa_info.enabled || !other_gpu->parent->numa_info.enabled))
|
|
continue;
|
|
|
|
status = enable_nvlink_peer_access(gpu, other_gpu, &p2p_caps_params);
|
|
if (status != NV_OK)
|
|
goto cleanup;
|
|
}
|
|
|
|
return NV_OK;
|
|
|
|
cleanup:
|
|
destroy_nvlink_peers(gpu);
|
|
|
|
return status;
|
|
}
|
|
|
|
static void destroy_nvlink_peers(uvm_gpu_t *gpu)
|
|
{
|
|
uvm_gpu_t *other_gpu;
|
|
|
|
UVM_ASSERT(gpu);
|
|
uvm_assert_mutex_locked(&g_uvm_global.global_lock);
|
|
|
|
if (gpu->parent->smc.enabled)
|
|
return;
|
|
|
|
for_each_global_gpu(other_gpu) {
|
|
uvm_gpu_peer_t *peer_caps;
|
|
|
|
if ((other_gpu == gpu) || other_gpu->parent->smc.enabled)
|
|
continue;
|
|
|
|
peer_caps = uvm_gpu_peer_caps(gpu, other_gpu);
|
|
|
|
// PCIe peers need to be explicitly destroyed via UvmDisablePeerAccess
|
|
if (peer_caps->link_type == UVM_GPU_LINK_INVALID || peer_caps->link_type == UVM_GPU_LINK_PCIE)
|
|
continue;
|
|
|
|
disable_peer_access(gpu, other_gpu);
|
|
}
|
|
}
|
|
|
|
NV_STATUS uvm_gpu_retain_pcie_peer_access(uvm_gpu_t *gpu0, uvm_gpu_t *gpu1)
|
|
{
|
|
NV_STATUS status = NV_OK;
|
|
uvm_gpu_peer_t *peer_caps;
|
|
|
|
UVM_ASSERT(gpu0);
|
|
UVM_ASSERT(gpu1);
|
|
uvm_assert_mutex_locked(&g_uvm_global.global_lock);
|
|
|
|
peer_caps = uvm_gpu_peer_caps(gpu0, gpu1);
|
|
|
|
// Insert an entry into global peer table, if not present.
|
|
if (peer_caps->link_type == UVM_GPU_LINK_INVALID) {
|
|
UVM_ASSERT(peer_caps->ref_count == 0);
|
|
|
|
status = enable_pcie_peer_access(gpu0, gpu1);
|
|
if (status != NV_OK)
|
|
return status;
|
|
}
|
|
else if (peer_caps->link_type != UVM_GPU_LINK_PCIE) {
|
|
return NV_ERR_INVALID_DEVICE;
|
|
}
|
|
|
|
// GPUs can't be destroyed until their peer pairings have also been
|
|
// destroyed.
|
|
uvm_gpu_retain(gpu0);
|
|
uvm_gpu_retain(gpu1);
|
|
|
|
peer_caps->ref_count++;
|
|
|
|
return status;
|
|
}
|
|
|
|
static void disable_peer_access(uvm_gpu_t *gpu0, uvm_gpu_t *gpu1)
|
|
{
|
|
uvm_gpu_peer_t *peer_caps;
|
|
NvHandle p2p_handle = 0;
|
|
|
|
UVM_ASSERT(gpu0);
|
|
UVM_ASSERT(gpu1);
|
|
|
|
// P2P is not supported under SMC partitioning
|
|
UVM_ASSERT(!gpu0->parent->smc.enabled);
|
|
UVM_ASSERT(!gpu1->parent->smc.enabled);
|
|
|
|
uvm_assert_mutex_locked(&g_uvm_global.global_lock);
|
|
|
|
peer_caps = uvm_gpu_peer_caps(gpu0, gpu1);
|
|
|
|
if (uvm_procfs_is_debug_enabled())
|
|
deinit_procfs_peer_cap_files(peer_caps);
|
|
|
|
p2p_handle = peer_caps->p2p_handle;
|
|
|
|
if (peer_caps->is_indirect_peer) {
|
|
uvm_pmm_gpu_indirect_peer_destroy(&gpu0->pmm, gpu1);
|
|
uvm_pmm_gpu_indirect_peer_destroy(&gpu1->pmm, gpu0);
|
|
}
|
|
else {
|
|
UVM_ASSERT(p2p_handle);
|
|
|
|
uvm_mmu_destroy_peer_identity_mappings(gpu0, gpu1);
|
|
uvm_mmu_destroy_peer_identity_mappings(gpu1, gpu0);
|
|
|
|
uvm_rm_locked_call_void(nvUvmInterfaceP2pObjectDestroy(uvm_gpu_session_handle(gpu0), p2p_handle));
|
|
|
|
UVM_ASSERT(uvm_gpu_get(gpu0->global_id) == gpu0);
|
|
UVM_ASSERT(uvm_gpu_get(gpu1->global_id) == gpu1);
|
|
|
|
uvm_spin_lock(&gpu0->peer_info.peer_gpus_lock);
|
|
uvm_processor_mask_clear(&gpu0->peer_info.peer_gpu_mask, gpu1->id);
|
|
gpu0->peer_info.peer_gpus[uvm_id_gpu_index(gpu1->id)] = NULL;
|
|
uvm_spin_unlock(&gpu0->peer_info.peer_gpus_lock);
|
|
|
|
uvm_spin_lock(&gpu1->peer_info.peer_gpus_lock);
|
|
uvm_processor_mask_clear(&gpu1->peer_info.peer_gpu_mask, gpu0->id);
|
|
gpu1->peer_info.peer_gpus[uvm_id_gpu_index(gpu0->id)] = NULL;
|
|
uvm_spin_unlock(&gpu1->peer_info.peer_gpus_lock);
|
|
}
|
|
|
|
// Flush the access counter buffer to avoid getting stale notifications for
|
|
// accesses to GPUs to which peer access is being disabled. This is also
|
|
// needed in the case of disabling automatic (NVLINK) peers on GPU
|
|
// unregister, because access counter processing might still be using GPU
|
|
// IDs queried from the peer table above which are about to be removed from
|
|
// the global table.
|
|
if (gpu0->parent->access_counters_supported)
|
|
uvm_gpu_access_counter_buffer_flush(gpu0);
|
|
if (gpu1->parent->access_counters_supported)
|
|
uvm_gpu_access_counter_buffer_flush(gpu1);
|
|
|
|
memset(peer_caps, 0, sizeof(*peer_caps));
|
|
}
|
|
|
|
void uvm_gpu_release_pcie_peer_access(uvm_gpu_t *gpu0, uvm_gpu_t *gpu1)
|
|
{
|
|
uvm_gpu_peer_t *peer_caps;
|
|
UVM_ASSERT(gpu0);
|
|
UVM_ASSERT(gpu1);
|
|
uvm_assert_mutex_locked(&g_uvm_global.global_lock);
|
|
|
|
peer_caps = uvm_gpu_peer_caps(gpu0, gpu1);
|
|
|
|
UVM_ASSERT(peer_caps->ref_count > 0);
|
|
UVM_ASSERT(peer_caps->link_type == UVM_GPU_LINK_PCIE);
|
|
peer_caps->ref_count--;
|
|
|
|
if (peer_caps->ref_count == 0)
|
|
disable_peer_access(gpu0, gpu1);
|
|
|
|
uvm_gpu_release_locked(gpu0);
|
|
uvm_gpu_release_locked(gpu1);
|
|
}
|
|
|
|
static uvm_aperture_t uvm_gpu_peer_caps_aperture(uvm_gpu_peer_t *peer_caps, uvm_gpu_t *local_gpu, uvm_gpu_t *remote_gpu)
|
|
{
|
|
size_t peer_index;
|
|
UVM_ASSERT(peer_caps->link_type != UVM_GPU_LINK_INVALID);
|
|
|
|
// Indirect peers are accessed as sysmem addresses
|
|
if (peer_caps->is_indirect_peer)
|
|
return UVM_APERTURE_SYS;
|
|
|
|
if (uvm_id_value(local_gpu->id) < uvm_id_value(remote_gpu->id))
|
|
peer_index = 0;
|
|
else
|
|
peer_index = 1;
|
|
|
|
return UVM_APERTURE_PEER(peer_caps->peer_ids[peer_index]);
|
|
}
|
|
|
|
uvm_aperture_t uvm_gpu_peer_aperture(uvm_gpu_t *local_gpu, uvm_gpu_t *remote_gpu)
|
|
{
|
|
uvm_gpu_peer_t *peer_caps = uvm_gpu_peer_caps(local_gpu, remote_gpu);
|
|
return uvm_gpu_peer_caps_aperture(peer_caps, local_gpu, remote_gpu);
|
|
}
|
|
|
|
uvm_aperture_t uvm_gpu_page_tree_init_location(const uvm_gpu_t *gpu)
|
|
{
|
|
// See comment in page_tree_set_location
|
|
return uvm_gpu_is_virt_mode_sriov_heavy(gpu)? UVM_APERTURE_VID : UVM_APERTURE_DEFAULT;
|
|
}
|
|
|
|
uvm_processor_id_t uvm_gpu_get_processor_id_by_address(uvm_gpu_t *gpu, uvm_gpu_phys_address_t addr)
|
|
{
|
|
uvm_processor_id_t id = UVM_ID_INVALID;
|
|
|
|
// TODO: Bug 1899622: On P9 systems with multiple CPU sockets, SYS aperture
|
|
// is also reported for accesses to remote GPUs connected to a different CPU
|
|
// NUMA domain. We will need to determine the actual processor id using the
|
|
// reported physical address.
|
|
if (addr.aperture == UVM_APERTURE_SYS)
|
|
return UVM_ID_CPU;
|
|
else if (addr.aperture == UVM_APERTURE_VID)
|
|
return gpu->id;
|
|
|
|
uvm_spin_lock(&gpu->peer_info.peer_gpus_lock);
|
|
|
|
for_each_gpu_id_in_mask(id, &gpu->peer_info.peer_gpu_mask) {
|
|
uvm_gpu_t *other_gpu = gpu->peer_info.peer_gpus[uvm_id_gpu_index(id)];
|
|
UVM_ASSERT(other_gpu);
|
|
|
|
if (uvm_gpus_are_nvswitch_connected(gpu, other_gpu)) {
|
|
// NVSWITCH connected systems use an extended physical address to
|
|
// map to peers. Find the physical memory 'slot' containing the
|
|
// given physical address to find the peer gpu that owns the
|
|
// physical address
|
|
NvU64 fabric_window_end = other_gpu->parent->nvswitch_info.fabric_memory_window_start +
|
|
other_gpu->mem_info.max_allocatable_address;
|
|
|
|
if (other_gpu->parent->nvswitch_info.fabric_memory_window_start <= addr.address &&
|
|
fabric_window_end >= addr.address)
|
|
break;
|
|
}
|
|
else if (uvm_gpu_peer_aperture(gpu, other_gpu) == addr.aperture) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
uvm_spin_unlock(&gpu->peer_info.peer_gpus_lock);
|
|
|
|
return id;
|
|
}
|
|
|
|
uvm_gpu_peer_t *uvm_gpu_index_peer_caps(const uvm_gpu_id_t gpu_id1, const uvm_gpu_id_t gpu_id2)
|
|
{
|
|
NvU32 table_index = uvm_gpu_peer_table_index(gpu_id1, gpu_id2);
|
|
return &g_uvm_global.peers[table_index];
|
|
}
|
|
|
|
static NvU64 instance_ptr_to_key(uvm_gpu_phys_address_t instance_ptr)
|
|
{
|
|
NvU64 key;
|
|
int is_sys = (instance_ptr.aperture == UVM_APERTURE_SYS);
|
|
|
|
// Instance pointers must be 4k aligned and they must have either VID or SYS
|
|
// apertures. Compress them as much as we can both to guarantee that the key
|
|
// fits within 64 bits, and to make the table as shallow as possible.
|
|
UVM_ASSERT(IS_ALIGNED(instance_ptr.address, UVM_PAGE_SIZE_4K));
|
|
UVM_ASSERT(instance_ptr.aperture == UVM_APERTURE_VID || instance_ptr.aperture == UVM_APERTURE_SYS);
|
|
|
|
key = (instance_ptr.address >> 11) | is_sys;
|
|
|
|
return key;
|
|
}
|
|
|
|
static NV_STATUS gpu_add_user_channel_subctx_info(uvm_gpu_t *gpu, uvm_user_channel_t *user_channel)
|
|
{
|
|
uvm_gpu_phys_address_t instance_ptr = user_channel->instance_ptr.addr;
|
|
NV_STATUS status = NV_OK;
|
|
uvm_rb_tree_node_t *channel_tree_node;
|
|
uvm_user_channel_subctx_info_t *channel_subctx_info;
|
|
uvm_user_channel_subctx_info_t *new_channel_subctx_info = NULL;
|
|
uvm_va_space_t *va_space = user_channel->gpu_va_space->va_space;
|
|
|
|
if (!user_channel->in_subctx)
|
|
return NV_OK;
|
|
|
|
// Pre-allocate a subcontext info descriptor out of the lock, in case we
|
|
// need to add a new entry to the tree
|
|
new_channel_subctx_info = uvm_kvmalloc_zero(sizeof(*new_channel_subctx_info));
|
|
|
|
// Don't check for the result of the allocation since it is only needed
|
|
// if the TSG has not been registered yet, and we do that under the lock
|
|
// below
|
|
if (new_channel_subctx_info) {
|
|
new_channel_subctx_info->subctxs =
|
|
uvm_kvmalloc_zero(sizeof(*new_channel_subctx_info->subctxs) * user_channel->tsg.max_subctx_count);
|
|
}
|
|
|
|
uvm_spin_lock(&gpu->parent->instance_ptr_table_lock);
|
|
|
|
// Check if the subcontext information for the channel already exists
|
|
channel_tree_node = uvm_rb_tree_find(&gpu->parent->tsg_table, user_channel->tsg.id);
|
|
|
|
if (!channel_tree_node) {
|
|
// We could not allocate the descriptor before taking the lock. Exiting
|
|
if (!new_channel_subctx_info || !new_channel_subctx_info->subctxs) {
|
|
status = NV_ERR_NO_MEMORY;
|
|
goto exit_unlock;
|
|
}
|
|
|
|
// Insert the new subcontext information descriptor
|
|
new_channel_subctx_info->node.key = user_channel->tsg.id;
|
|
status = uvm_rb_tree_insert(&gpu->parent->tsg_table, &new_channel_subctx_info->node);
|
|
UVM_ASSERT(status == NV_OK);
|
|
|
|
channel_subctx_info = new_channel_subctx_info;
|
|
channel_subctx_info->smc_engine_id = user_channel->smc_engine_id;
|
|
}
|
|
else {
|
|
channel_subctx_info = container_of(channel_tree_node, uvm_user_channel_subctx_info_t, node);
|
|
UVM_ASSERT(channel_subctx_info->smc_engine_id == user_channel->smc_engine_id);
|
|
}
|
|
|
|
user_channel->subctx_info = channel_subctx_info;
|
|
|
|
// Register the VA space of the channel subcontext info descriptor, or
|
|
// check that the existing one matches the channel's
|
|
if (channel_subctx_info->subctxs[user_channel->subctx_id].refcount++ > 0) {
|
|
UVM_ASSERT_MSG(channel_subctx_info->subctxs[user_channel->subctx_id].va_space == va_space,
|
|
"CH %u:%u instance_ptr {0x%llx:%s} SubCTX %u in TSG %u: expected VA space 0x%llx but got 0x%llx instead\n",
|
|
user_channel->hw_runlist_id,
|
|
user_channel->hw_channel_id,
|
|
instance_ptr.address,
|
|
uvm_aperture_string(instance_ptr.aperture),
|
|
user_channel->subctx_id,
|
|
user_channel->tsg.id,
|
|
(NvU64)va_space,
|
|
(NvU64)channel_subctx_info->subctxs[user_channel->subctx_id].va_space);
|
|
UVM_ASSERT_MSG(channel_subctx_info->subctxs[user_channel->subctx_id].va_space != NULL,
|
|
"CH %u:%u instance_ptr {0x%llx:%s} SubCTX %u in TSG %u: VA space is NULL\n",
|
|
user_channel->hw_runlist_id,
|
|
user_channel->hw_channel_id,
|
|
instance_ptr.address,
|
|
uvm_aperture_string(instance_ptr.aperture),
|
|
user_channel->subctx_id,
|
|
user_channel->tsg.id);
|
|
UVM_ASSERT_MSG(channel_subctx_info->total_refcount > 0,
|
|
"CH %u:%u instance_ptr {0x%llx:%s} SubCTX %u in TSG %u: TSG refcount is 0\n",
|
|
user_channel->hw_runlist_id,
|
|
user_channel->hw_channel_id,
|
|
instance_ptr.address,
|
|
uvm_aperture_string(instance_ptr.aperture),
|
|
user_channel->subctx_id,
|
|
user_channel->tsg.id);
|
|
}
|
|
else {
|
|
UVM_ASSERT_MSG(channel_subctx_info->subctxs[user_channel->subctx_id].va_space == NULL,
|
|
"CH %u:%u instance_ptr {0x%llx:%s} SubCTX %u in TSG %u: expected VA space NULL but got 0x%llx instead\n",
|
|
user_channel->hw_runlist_id,
|
|
user_channel->hw_channel_id,
|
|
instance_ptr.address,
|
|
uvm_aperture_string(instance_ptr.aperture),
|
|
user_channel->subctx_id,
|
|
user_channel->tsg.id,
|
|
(NvU64)channel_subctx_info->subctxs[user_channel->subctx_id].va_space);
|
|
|
|
channel_subctx_info->subctxs[user_channel->subctx_id].va_space = va_space;
|
|
}
|
|
|
|
++channel_subctx_info->total_refcount;
|
|
|
|
exit_unlock:
|
|
uvm_spin_unlock(&gpu->parent->instance_ptr_table_lock);
|
|
|
|
// Remove the pre-allocated per-TSG subctx information struct if there was
|
|
// some error or it was not used
|
|
if (status != NV_OK || user_channel->subctx_info != new_channel_subctx_info) {
|
|
if (new_channel_subctx_info)
|
|
uvm_kvfree(new_channel_subctx_info->subctxs);
|
|
|
|
uvm_kvfree(new_channel_subctx_info);
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
static void gpu_remove_user_channel_subctx_info_locked(uvm_gpu_t *gpu, uvm_user_channel_t *user_channel)
|
|
{
|
|
uvm_gpu_phys_address_t instance_ptr = user_channel->instance_ptr.addr;
|
|
uvm_va_space_t *va_space = user_channel->gpu_va_space->va_space;
|
|
|
|
uvm_assert_spinlock_locked(&gpu->parent->instance_ptr_table_lock);
|
|
|
|
// Channel subcontext info descriptor may not have been registered in
|
|
// tsg_table since this function is called in some teardown paths during
|
|
// channel creation
|
|
if (!user_channel->subctx_info)
|
|
return;
|
|
|
|
UVM_ASSERT_MSG(&user_channel->subctx_info->node ==
|
|
uvm_rb_tree_find(&gpu->parent->tsg_table, user_channel->subctx_info->node.key),
|
|
"CH %u:%u instance_ptr {0x%llx:%s} SubCTX %u in TSG %u: SubCTX not found in TSG table\n",
|
|
user_channel->hw_runlist_id,
|
|
user_channel->hw_channel_id,
|
|
instance_ptr.address,
|
|
uvm_aperture_string(instance_ptr.aperture),
|
|
user_channel->subctx_id,
|
|
user_channel->tsg.id);
|
|
|
|
UVM_ASSERT_MSG(user_channel->subctx_info->subctxs[user_channel->subctx_id].refcount > 0,
|
|
"CH %u:%u instance_ptr {0x%llx:%s} SubCTX %u in TSG %u: SubCTX refcount is 0\n",
|
|
user_channel->hw_runlist_id,
|
|
user_channel->hw_channel_id,
|
|
instance_ptr.address,
|
|
uvm_aperture_string(instance_ptr.aperture),
|
|
user_channel->subctx_id,
|
|
user_channel->tsg.id);
|
|
|
|
UVM_ASSERT_MSG(user_channel->subctx_info->subctxs[user_channel->subctx_id].va_space == va_space,
|
|
"CH %u:%u instance_ptr {0x%llx:%s} SubCTX %u in TSG %u: expected VA space 0x%llx but got 0x%llx instead\n",
|
|
user_channel->hw_runlist_id,
|
|
user_channel->hw_channel_id,
|
|
instance_ptr.address,
|
|
uvm_aperture_string(instance_ptr.aperture),
|
|
user_channel->subctx_id,
|
|
user_channel->tsg.id,
|
|
(NvU64)va_space,
|
|
(NvU64)user_channel->subctx_info->subctxs[user_channel->subctx_id].va_space);
|
|
|
|
UVM_ASSERT_MSG(user_channel->subctx_info->total_refcount > 0,
|
|
"CH %u:%u instance_ptr {0x%llx:%s} SubCTX %u in TSG %u: TSG refcount is 0\n",
|
|
user_channel->hw_runlist_id,
|
|
user_channel->hw_channel_id,
|
|
instance_ptr.address,
|
|
uvm_aperture_string(instance_ptr.aperture),
|
|
user_channel->subctx_id,
|
|
user_channel->tsg.id);
|
|
|
|
// Decrement VA space refcount. If it gets to zero, unregister the pointer
|
|
if (--user_channel->subctx_info->subctxs[user_channel->subctx_id].refcount == 0)
|
|
user_channel->subctx_info->subctxs[user_channel->subctx_id].va_space = NULL;
|
|
|
|
if (--user_channel->subctx_info->total_refcount == 0) {
|
|
uvm_rb_tree_remove(&gpu->parent->tsg_table, &user_channel->subctx_info->node);
|
|
uvm_kvfree(user_channel->subctx_info->subctxs);
|
|
uvm_kvfree(user_channel->subctx_info);
|
|
}
|
|
|
|
user_channel->subctx_info = NULL;
|
|
}
|
|
|
|
static void gpu_remove_user_channel_subctx_info(uvm_gpu_t *gpu, uvm_user_channel_t *user_channel)
|
|
{
|
|
uvm_spin_lock(&gpu->parent->instance_ptr_table_lock);
|
|
gpu_remove_user_channel_subctx_info_locked(gpu, user_channel);
|
|
uvm_spin_unlock(&gpu->parent->instance_ptr_table_lock);
|
|
}
|
|
|
|
static void gpu_add_user_channel_instance_ptr(uvm_gpu_t *gpu, uvm_user_channel_t *user_channel)
|
|
{
|
|
uvm_gpu_phys_address_t instance_ptr = user_channel->instance_ptr.addr;
|
|
NvU64 instance_ptr_key = instance_ptr_to_key(instance_ptr);
|
|
NV_STATUS status;
|
|
|
|
uvm_spin_lock(&gpu->parent->instance_ptr_table_lock);
|
|
|
|
// Insert the instance_ptr -> user_channel mapping
|
|
user_channel->instance_ptr.node.key = instance_ptr_key;
|
|
status = uvm_rb_tree_insert(&gpu->parent->instance_ptr_table, &user_channel->instance_ptr.node);
|
|
|
|
uvm_spin_unlock(&gpu->parent->instance_ptr_table_lock);
|
|
|
|
UVM_ASSERT_MSG(status == NV_OK, "CH %u:%u instance_ptr {0x%llx:%s} SubCTX %u in TSG %u: error %s\n",
|
|
user_channel->hw_runlist_id,
|
|
user_channel->hw_channel_id,
|
|
instance_ptr.address,
|
|
uvm_aperture_string(instance_ptr.aperture),
|
|
user_channel->subctx_id,
|
|
user_channel->tsg.id,
|
|
nvstatusToString(status));
|
|
}
|
|
|
|
static void gpu_remove_user_channel_instance_ptr_locked(uvm_gpu_t *gpu, uvm_user_channel_t *user_channel)
|
|
{
|
|
uvm_assert_spinlock_locked(&gpu->parent->instance_ptr_table_lock);
|
|
|
|
if (UVM_RB_TREE_EMPTY_NODE(&user_channel->instance_ptr.node))
|
|
return;
|
|
|
|
uvm_rb_tree_remove(&gpu->parent->instance_ptr_table, &user_channel->instance_ptr.node);
|
|
}
|
|
|
|
NV_STATUS uvm_gpu_add_user_channel(uvm_gpu_t *gpu, uvm_user_channel_t *user_channel)
|
|
{
|
|
uvm_va_space_t *va_space;
|
|
uvm_gpu_va_space_t *gpu_va_space = user_channel->gpu_va_space;
|
|
NV_STATUS status;
|
|
|
|
UVM_ASSERT(user_channel->rm_retained_channel);
|
|
UVM_ASSERT(gpu_va_space);
|
|
UVM_ASSERT(uvm_gpu_va_space_state(gpu_va_space) == UVM_GPU_VA_SPACE_STATE_ACTIVE);
|
|
va_space = gpu_va_space->va_space;
|
|
uvm_assert_rwsem_locked(&va_space->lock);
|
|
|
|
status = gpu_add_user_channel_subctx_info(gpu, user_channel);
|
|
if (status != NV_OK)
|
|
return status;
|
|
|
|
gpu_add_user_channel_instance_ptr(gpu, user_channel);
|
|
|
|
return NV_OK;
|
|
}
|
|
|
|
static uvm_user_channel_t *instance_ptr_to_user_channel(uvm_gpu_t *gpu, uvm_gpu_phys_address_t instance_ptr)
|
|
{
|
|
NvU64 key = instance_ptr_to_key(instance_ptr);
|
|
uvm_rb_tree_node_t *instance_node;
|
|
|
|
uvm_assert_spinlock_locked(&gpu->parent->instance_ptr_table_lock);
|
|
|
|
instance_node = uvm_rb_tree_find(&gpu->parent->instance_ptr_table, key);
|
|
if (!instance_node)
|
|
return NULL;
|
|
|
|
return get_user_channel(instance_node);
|
|
}
|
|
|
|
static uvm_va_space_t *user_channel_and_subctx_to_va_space(uvm_user_channel_t *user_channel, NvU32 subctx_id)
|
|
{
|
|
uvm_user_channel_subctx_info_t *channel_subctx_info;
|
|
|
|
UVM_ASSERT(user_channel);
|
|
UVM_ASSERT(user_channel->in_subctx);
|
|
UVM_ASSERT(user_channel->subctx_info);
|
|
|
|
uvm_assert_spinlock_locked(&user_channel->gpu->parent->instance_ptr_table_lock);
|
|
|
|
channel_subctx_info = user_channel->subctx_info;
|
|
|
|
UVM_ASSERT_MSG(subctx_id < user_channel->tsg.max_subctx_count,
|
|
"instance_ptr {0x%llx:%s} in TSG %u. Invalid SubCTX %u\n",
|
|
user_channel->instance_ptr.addr.address,
|
|
uvm_aperture_string(user_channel->instance_ptr.addr.aperture),
|
|
user_channel->tsg.id,
|
|
subctx_id);
|
|
UVM_ASSERT_MSG(channel_subctx_info->total_refcount > 0,
|
|
"instance_ptr {0x%llx:%s} in TSG %u: TSG refcount is 0\n",
|
|
user_channel->instance_ptr.addr.address,
|
|
uvm_aperture_string(user_channel->instance_ptr.addr.aperture),
|
|
user_channel->tsg.id);
|
|
|
|
// A subcontext's refcount can be zero if that subcontext is torn down
|
|
// uncleanly and work from that subcontext continues running with work from
|
|
// other subcontexts.
|
|
if (channel_subctx_info->subctxs[subctx_id].refcount == 0) {
|
|
UVM_ASSERT(channel_subctx_info->subctxs[subctx_id].va_space == NULL);
|
|
}
|
|
else {
|
|
UVM_ASSERT_MSG(channel_subctx_info->subctxs[subctx_id].va_space,
|
|
"instance_ptr {0x%llx:%s} in TSG %u: no VA space for SubCTX %u\n",
|
|
user_channel->instance_ptr.addr.address,
|
|
uvm_aperture_string(user_channel->instance_ptr.addr.aperture),
|
|
user_channel->tsg.id,
|
|
subctx_id);
|
|
}
|
|
|
|
return channel_subctx_info->subctxs[subctx_id].va_space;
|
|
}
|
|
|
|
NV_STATUS uvm_gpu_fault_entry_to_va_space(uvm_gpu_t *gpu,
|
|
uvm_fault_buffer_entry_t *fault,
|
|
uvm_va_space_t **out_va_space)
|
|
{
|
|
uvm_user_channel_t *user_channel;
|
|
NV_STATUS status = NV_OK;
|
|
|
|
*out_va_space = NULL;
|
|
|
|
uvm_spin_lock(&gpu->parent->instance_ptr_table_lock);
|
|
|
|
user_channel = instance_ptr_to_user_channel(gpu, fault->instance_ptr);
|
|
if (!user_channel) {
|
|
status = NV_ERR_INVALID_CHANNEL;
|
|
goto exit_unlock;
|
|
}
|
|
|
|
// Faults from HUB clients will always report VEID 0 even if the channel
|
|
// belongs a TSG with many subcontexts. Therefore, we cannot use the per-TSG
|
|
// subctx table and we need to directly return the channel's VA space
|
|
if (!user_channel->in_subctx || (fault->fault_source.client_type == UVM_FAULT_CLIENT_TYPE_HUB)) {
|
|
UVM_ASSERT_MSG(fault->fault_source.ve_id == 0,
|
|
"Fault packet contains SubCTX %u for channel not in subctx\n",
|
|
fault->fault_source.ve_id);
|
|
|
|
// We can safely access user_channel->gpu_va_space under the
|
|
// instance_ptr_table_lock since gpu_va_space is set to NULL after this
|
|
// function is called in uvm_user_channel_detach
|
|
UVM_ASSERT(uvm_gpu_va_space_state(user_channel->gpu_va_space) == UVM_GPU_VA_SPACE_STATE_ACTIVE);
|
|
*out_va_space = user_channel->gpu_va_space->va_space;
|
|
}
|
|
else {
|
|
NvU32 ve_id = fault->fault_source.ve_id;
|
|
|
|
// Compute the SMC engine-local VEID
|
|
UVM_ASSERT(ve_id >= user_channel->smc_engine_ve_id_offset);
|
|
|
|
ve_id -= user_channel->smc_engine_ve_id_offset;
|
|
|
|
*out_va_space = user_channel_and_subctx_to_va_space(user_channel, ve_id);
|
|
|
|
// Instance pointer is valid but the fault targets a non-existent
|
|
// subcontext.
|
|
if (!*out_va_space)
|
|
status = NV_ERR_PAGE_TABLE_NOT_AVAIL;
|
|
}
|
|
|
|
exit_unlock:
|
|
uvm_spin_unlock(&gpu->parent->instance_ptr_table_lock);
|
|
|
|
if (status == NV_OK)
|
|
UVM_ASSERT(uvm_va_space_initialized(*out_va_space) == NV_OK);
|
|
|
|
return status;
|
|
}
|
|
|
|
NV_STATUS uvm_gpu_access_counter_entry_to_va_space(uvm_gpu_t *gpu,
|
|
uvm_access_counter_buffer_entry_t *entry,
|
|
uvm_va_space_t **out_va_space)
|
|
{
|
|
uvm_user_channel_t *user_channel;
|
|
NV_STATUS status = NV_OK;
|
|
|
|
*out_va_space = NULL;
|
|
UVM_ASSERT(entry->address.is_virtual);
|
|
|
|
uvm_spin_lock(&gpu->parent->instance_ptr_table_lock);
|
|
|
|
user_channel = instance_ptr_to_user_channel(gpu, entry->virtual_info.instance_ptr);
|
|
if (!user_channel) {
|
|
status = NV_ERR_INVALID_CHANNEL;
|
|
goto exit_unlock;
|
|
}
|
|
|
|
if (!user_channel->in_subctx) {
|
|
UVM_ASSERT_MSG(entry->virtual_info.ve_id == 0,
|
|
"Access counter packet contains SubCTX %u for channel not in subctx\n",
|
|
entry->virtual_info.ve_id);
|
|
|
|
UVM_ASSERT(uvm_gpu_va_space_state(user_channel->gpu_va_space) == UVM_GPU_VA_SPACE_STATE_ACTIVE);
|
|
*out_va_space = user_channel->gpu_va_space->va_space;
|
|
}
|
|
else {
|
|
*out_va_space = user_channel_and_subctx_to_va_space(user_channel, entry->virtual_info.ve_id);
|
|
if (!*out_va_space)
|
|
status = NV_ERR_PAGE_TABLE_NOT_AVAIL;
|
|
}
|
|
|
|
exit_unlock:
|
|
uvm_spin_unlock(&gpu->parent->instance_ptr_table_lock);
|
|
|
|
if (status == NV_OK)
|
|
UVM_ASSERT(uvm_va_space_initialized(*out_va_space) == NV_OK);
|
|
|
|
return status;
|
|
}
|
|
|
|
void uvm_gpu_remove_user_channel(uvm_gpu_t *gpu, uvm_user_channel_t *user_channel)
|
|
{
|
|
uvm_va_space_t *va_space;
|
|
uvm_gpu_va_space_t *gpu_va_space = user_channel->gpu_va_space;
|
|
|
|
UVM_ASSERT(user_channel->rm_retained_channel);
|
|
UVM_ASSERT(gpu_va_space);
|
|
UVM_ASSERT(uvm_gpu_va_space_state(gpu_va_space) == UVM_GPU_VA_SPACE_STATE_ACTIVE);
|
|
va_space = gpu_va_space->va_space;
|
|
uvm_assert_rwsem_locked_write(&va_space->lock);
|
|
|
|
uvm_spin_lock(&gpu->parent->instance_ptr_table_lock);
|
|
gpu_remove_user_channel_subctx_info_locked(gpu, user_channel);
|
|
gpu_remove_user_channel_instance_ptr_locked(gpu, user_channel);
|
|
uvm_spin_unlock(&gpu->parent->instance_ptr_table_lock);
|
|
}
|
|
|
|
static NvU64 gpu_addr_to_dma_addr(uvm_parent_gpu_t *parent_gpu, NvU64 gpu_addr)
|
|
{
|
|
NvU64 dma_addr = gpu_addr;
|
|
UVM_ASSERT(dma_addr <= dma_addr + parent_gpu->dma_addressable_start);
|
|
|
|
if (parent_gpu->npu)
|
|
dma_addr = nv_expand_nvlink_addr(dma_addr);
|
|
|
|
dma_addr += parent_gpu->dma_addressable_start;
|
|
|
|
return dma_addr;
|
|
}
|
|
|
|
// The GPU has its NV_PFB_XV_UPPER_ADDR register set by RM to
|
|
// dma_addressable_start (in bifSetupDmaWindow_IMPL()) and hence when
|
|
// referencing sysmem from the GPU, dma_addressable_start should be
|
|
// subtracted from the DMA address we get from the OS.
|
|
static NvU64 dma_addr_to_gpu_addr(uvm_parent_gpu_t *parent_gpu, NvU64 dma_addr)
|
|
{
|
|
NvU64 gpu_addr = dma_addr - parent_gpu->dma_addressable_start;
|
|
UVM_ASSERT(dma_addr >= gpu_addr);
|
|
|
|
// See Bug 1920398 for background and details about NVLink DMA address
|
|
// transformations being applied here.
|
|
if (parent_gpu->npu)
|
|
gpu_addr = nv_compress_nvlink_addr(gpu_addr);
|
|
|
|
return gpu_addr;
|
|
}
|
|
|
|
void *uvm_gpu_dma_alloc_page(uvm_parent_gpu_t *parent_gpu, gfp_t gfp_flags, NvU64 *dma_address_out)
|
|
{
|
|
NvU64 dma_addr;
|
|
void *cpu_addr;
|
|
|
|
cpu_addr = dma_alloc_coherent(&parent_gpu->pci_dev->dev, PAGE_SIZE, &dma_addr, gfp_flags);
|
|
|
|
if (!cpu_addr)
|
|
return cpu_addr;
|
|
|
|
*dma_address_out = dma_addr_to_gpu_addr(parent_gpu, dma_addr);
|
|
atomic64_add(PAGE_SIZE, &parent_gpu->mapped_cpu_pages_size);
|
|
return cpu_addr;
|
|
}
|
|
|
|
void uvm_gpu_dma_free_page(uvm_parent_gpu_t *parent_gpu, void *va, NvU64 dma_address)
|
|
{
|
|
dma_address = gpu_addr_to_dma_addr(parent_gpu, dma_address);
|
|
dma_free_coherent(&parent_gpu->pci_dev->dev, PAGE_SIZE, va, dma_address);
|
|
atomic64_sub(PAGE_SIZE, &parent_gpu->mapped_cpu_pages_size);
|
|
}
|
|
|
|
NV_STATUS uvm_gpu_map_cpu_pages(uvm_gpu_t *gpu, struct page *page, size_t size, NvU64 *dma_address_out)
|
|
{
|
|
NvU64 dma_addr;
|
|
|
|
UVM_ASSERT(PAGE_ALIGNED(size));
|
|
|
|
dma_addr = dma_map_page(&gpu->parent->pci_dev->dev, page, 0, size, DMA_BIDIRECTIONAL);
|
|
if (dma_mapping_error(&gpu->parent->pci_dev->dev, dma_addr))
|
|
return NV_ERR_OPERATING_SYSTEM;
|
|
|
|
if (dma_addr < gpu->parent->dma_addressable_start ||
|
|
dma_addr + size - 1 > gpu->parent->dma_addressable_limit) {
|
|
dma_unmap_page(&gpu->parent->pci_dev->dev, dma_addr, size, DMA_BIDIRECTIONAL);
|
|
UVM_ERR_PRINT_RL("PCI mapped range [0x%llx, 0x%llx) not in the addressable range [0x%llx, 0x%llx), GPU %s\n",
|
|
dma_addr,
|
|
dma_addr + (NvU64)size,
|
|
gpu->parent->dma_addressable_start,
|
|
gpu->parent->dma_addressable_limit + 1,
|
|
uvm_gpu_name(gpu));
|
|
return NV_ERR_INVALID_ADDRESS;
|
|
}
|
|
|
|
atomic64_add(size, &gpu->parent->mapped_cpu_pages_size);
|
|
*dma_address_out = dma_addr_to_gpu_addr(gpu->parent, dma_addr);
|
|
|
|
return NV_OK;
|
|
}
|
|
|
|
void uvm_gpu_unmap_cpu_pages(uvm_gpu_t *gpu, NvU64 dma_address, size_t size)
|
|
{
|
|
UVM_ASSERT(PAGE_ALIGNED(size));
|
|
|
|
dma_address = gpu_addr_to_dma_addr(gpu->parent, dma_address);
|
|
dma_unmap_page(&gpu->parent->pci_dev->dev, dma_address, size, DMA_BIDIRECTIONAL);
|
|
atomic64_sub(size, &gpu->parent->mapped_cpu_pages_size);
|
|
}
|
|
|
|
// This function implements the UvmRegisterGpu API call, as described in uvm.h.
|
|
// Notes:
|
|
//
|
|
// 1. The UVM VA space has a 1-to-1 relationship with an open instance of
|
|
// /dev/nvidia-uvm. That, in turn, has a 1-to-1 relationship with a process,
|
|
// because the user-level UVM code (os-user-linux.c, for example) enforces an
|
|
// "open /dev/nvidia-uvm only once per process" policy. So a UVM VA space is
|
|
// very close to a process's VA space.
|
|
//
|
|
// If that user space code fails or is not used, then the relationship is no
|
|
// longer 1-to-1. That situation requires that this code should avoid crashing,
|
|
// leaking resources, exhibiting security holes, etc, but it does not have to
|
|
// provide correct UVM API behavior. Correct UVM API behavior requires doing
|
|
// the right things in user space before calling into the kernel.
|
|
//
|
|
// 2. The uvm_api*() routines are invoked directly from the top-level ioctl
|
|
// handler. They are considered "API routing routines", because they are
|
|
// responsible for providing the behavior that is described in the UVM
|
|
// user-to-kernel API documentation, in uvm.h.
|
|
//
|
|
// 3. A GPU VA space, which you'll see in other parts of the driver,
|
|
// is something different: there may be more than one
|
|
// GPU VA space within a process, and therefore within a UVM VA space.
|
|
//
|
|
NV_STATUS uvm_api_register_gpu(UVM_REGISTER_GPU_PARAMS *params, struct file *filp)
|
|
{
|
|
uvm_va_space_t *va_space = uvm_va_space_get(filp);
|
|
uvm_rm_user_object_t user_rm_va_space = {
|
|
.rm_control_fd = params->rmCtrlFd,
|
|
.user_client = params->hClient,
|
|
.user_object = params->hSmcPartRef,
|
|
};
|
|
|
|
return uvm_va_space_register_gpu(va_space,
|
|
¶ms->gpu_uuid,
|
|
&user_rm_va_space,
|
|
¶ms->numaEnabled,
|
|
¶ms->numaNodeId);
|
|
}
|
|
|
|
NV_STATUS uvm_api_unregister_gpu(UVM_UNREGISTER_GPU_PARAMS *params, struct file *filp)
|
|
{
|
|
uvm_va_space_t *va_space = uvm_va_space_get(filp);
|
|
|
|
return uvm_va_space_unregister_gpu(va_space, ¶ms->gpu_uuid);
|
|
}
|
|
|
|
NV_STATUS uvm_api_register_gpu_va_space(UVM_REGISTER_GPU_VASPACE_PARAMS *params, struct file *filp)
|
|
{
|
|
uvm_va_space_t *va_space = uvm_va_space_get(filp);
|
|
uvm_rm_user_object_t user_rm_va_space = {
|
|
.rm_control_fd = params->rmCtrlFd,
|
|
.user_client = params->hClient,
|
|
.user_object = params->hVaSpace
|
|
};
|
|
return uvm_va_space_register_gpu_va_space(va_space, &user_rm_va_space, ¶ms->gpuUuid);
|
|
}
|
|
|
|
NV_STATUS uvm_api_unregister_gpu_va_space(UVM_UNREGISTER_GPU_VASPACE_PARAMS *params, struct file *filp)
|
|
{
|
|
uvm_va_space_t *va_space = uvm_va_space_get(filp);
|
|
return uvm_va_space_unregister_gpu_va_space(va_space, ¶ms->gpuUuid);
|
|
}
|
|
|
|
NV_STATUS uvm_api_pageable_mem_access_on_gpu(UVM_PAGEABLE_MEM_ACCESS_ON_GPU_PARAMS *params, struct file *filp)
|
|
{
|
|
uvm_va_space_t *va_space = uvm_va_space_get(filp);
|
|
uvm_gpu_t *gpu;
|
|
|
|
uvm_va_space_down_read(va_space);
|
|
gpu = uvm_va_space_get_gpu_by_uuid(va_space, ¶ms->gpu_uuid);
|
|
|
|
if (!gpu) {
|
|
uvm_va_space_up_read(va_space);
|
|
return NV_ERR_INVALID_DEVICE;
|
|
}
|
|
|
|
if (uvm_va_space_pageable_mem_access_supported(va_space) && gpu->parent->replayable_faults_supported)
|
|
params->pageableMemAccess = NV_TRUE;
|
|
else
|
|
params->pageableMemAccess = NV_FALSE;
|
|
|
|
uvm_va_space_up_read(va_space);
|
|
return NV_OK;
|
|
}
|
|
|
|
NV_STATUS uvm_test_set_prefetch_filtering(UVM_TEST_SET_PREFETCH_FILTERING_PARAMS *params, struct file *filp)
|
|
{
|
|
uvm_va_space_t *va_space = uvm_va_space_get(filp);
|
|
uvm_gpu_t *gpu = NULL;
|
|
NV_STATUS status = NV_OK;
|
|
|
|
uvm_mutex_lock(&g_uvm_global.global_lock);
|
|
|
|
uvm_va_space_down_read(va_space);
|
|
|
|
gpu = uvm_va_space_get_gpu_by_uuid(va_space, ¶ms->gpu_uuid);
|
|
|
|
if (!gpu) {
|
|
status = NV_ERR_INVALID_DEVICE;
|
|
goto done;
|
|
}
|
|
|
|
if (!gpu->parent->isr.replayable_faults.handling || !gpu->parent->prefetch_fault_supported) {
|
|
status = NV_ERR_INVALID_DEVICE;
|
|
goto done;
|
|
}
|
|
|
|
switch (params->filtering_mode) {
|
|
case UVM_TEST_PREFETCH_FILTERING_MODE_FILTER_ALL:
|
|
uvm_gpu_disable_prefetch_faults(gpu->parent);
|
|
break;
|
|
case UVM_TEST_PREFETCH_FILTERING_MODE_FILTER_NONE:
|
|
uvm_gpu_enable_prefetch_faults(gpu->parent);
|
|
break;
|
|
default:
|
|
status = NV_ERR_INVALID_ARGUMENT;
|
|
break;
|
|
}
|
|
|
|
done:
|
|
uvm_va_space_up_read(va_space);
|
|
|
|
uvm_mutex_unlock(&g_uvm_global.global_lock);
|
|
return status;
|
|
}
|
|
|
|
NV_STATUS uvm_test_get_gpu_time(UVM_TEST_GET_GPU_TIME_PARAMS *params, struct file *filp)
|
|
{
|
|
uvm_va_space_t *va_space = uvm_va_space_get(filp);
|
|
uvm_gpu_t *gpu = NULL;
|
|
NV_STATUS status = NV_OK;
|
|
|
|
uvm_va_space_down_read(va_space);
|
|
|
|
gpu = uvm_va_space_get_gpu_by_uuid(va_space, ¶ms->gpu_uuid);
|
|
|
|
if (gpu)
|
|
params->timestamp_ns = gpu->parent->host_hal->get_time(gpu);
|
|
else
|
|
status = NV_ERR_INVALID_DEVICE;
|
|
|
|
uvm_va_space_up_read(va_space);
|
|
|
|
return status;
|
|
}
|