/******************************************************************************* Copyright (c) 2015-2023 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 "uvm_channel.h" #include "uvm_global.h" #include "uvm_hal.h" #include "uvm_kvmalloc.h" #include "uvm_push.h" #include "uvm_test.h" #include "uvm_tracker.h" #include "uvm_va_space.h" #include "uvm_rm_mem.h" #include "uvm_mem.h" #include "uvm_gpu.h" #define CE_TEST_MEM_SIZE (2 * 1024 * 1024) #define CE_TEST_MEM_END_SIZE 32 #define CE_TEST_MEM_BEGIN_SIZE 32 #define CE_TEST_MEM_MIDDLE_SIZE (CE_TEST_MEM_SIZE - CE_TEST_MEM_BEGIN_SIZE - CE_TEST_MEM_END_SIZE) #define CE_TEST_MEM_MIDDLE_OFFSET (CE_TEST_MEM_BEGIN_SIZE) #define CE_TEST_MEM_END_OFFSET (CE_TEST_MEM_SIZE - CE_TEST_MEM_BEGIN_SIZE) #define CE_TEST_MEM_COUNT 5 static NV_STATUS test_non_pipelined(uvm_gpu_t *gpu) { NvU32 i; NV_STATUS status; uvm_rm_mem_t *mem[CE_TEST_MEM_COUNT] = { NULL }; uvm_rm_mem_t *host_mem = NULL; NvU32 *host_ptr; NvU64 host_mem_gpu_va, mem_gpu_va; NvU64 dst_va; NvU64 src_va; uvm_push_t push; bool is_proxy; // TODO: Bug 3839176: the test is waived on Confidential Computing because // it assumes that GPU can access system memory without using encryption. if (g_uvm_global.conf_computing_enabled) return NV_OK; status = uvm_rm_mem_alloc_and_map_cpu(gpu, UVM_RM_MEM_TYPE_SYS, CE_TEST_MEM_SIZE, 0, &host_mem); TEST_CHECK_GOTO(status == NV_OK, done); host_ptr = (NvU32 *)uvm_rm_mem_get_cpu_va(host_mem); memset(host_ptr, 0, CE_TEST_MEM_SIZE); for (i = 0; i < CE_TEST_MEM_COUNT; ++i) { status = uvm_rm_mem_alloc(gpu, UVM_RM_MEM_TYPE_GPU, CE_TEST_MEM_SIZE, 0, &mem[i]); TEST_CHECK_GOTO(status == NV_OK, done); } status = uvm_push_begin(gpu->channel_manager, UVM_CHANNEL_TYPE_GPU_INTERNAL, &push, "Non-pipelined test"); TEST_CHECK_GOTO(status == NV_OK, done); is_proxy = uvm_channel_is_proxy(push.channel); host_mem_gpu_va = uvm_rm_mem_get_gpu_va(host_mem, gpu, is_proxy).address; // All of the following CE transfers are done from a single (L)CE and // disabling pipelining is enough to order them when needed. Only push_end // needs a MEMBAR SYS to order everything with the CPU. // Initialize to a bad value for (i = 0; i < CE_TEST_MEM_COUNT; ++i) { mem_gpu_va = uvm_rm_mem_get_gpu_va(mem[i], gpu, is_proxy).address; uvm_push_set_flag(&push, UVM_PUSH_FLAG_CE_NEXT_PIPELINED); uvm_push_set_flag(&push, UVM_PUSH_FLAG_NEXT_MEMBAR_NONE); gpu->parent->ce_hal->memset_v_4(&push, mem_gpu_va, 1337 + i, CE_TEST_MEM_SIZE); } // Set the first buffer to 1 uvm_push_set_flag(&push, UVM_PUSH_FLAG_NEXT_MEMBAR_NONE); mem_gpu_va = uvm_rm_mem_get_gpu_va(mem[0], gpu, is_proxy).address; gpu->parent->ce_hal->memset_v_4(&push, mem_gpu_va, 1, CE_TEST_MEM_SIZE); for (i = 0; i < CE_TEST_MEM_COUNT; ++i) { NvU32 dst = i + 1; if (dst == CE_TEST_MEM_COUNT) dst_va = host_mem_gpu_va; else dst_va = uvm_rm_mem_get_gpu_va(mem[dst], gpu, is_proxy).address; src_va = uvm_rm_mem_get_gpu_va(mem[i], gpu, is_proxy).address; // The first memcpy needs to be non-pipelined as otherwise the previous // memset/memcpy to the source may not be done yet. // Alternate the order of copying the beginning and the end if (i % 2 == 0) { uvm_push_set_flag(&push, UVM_PUSH_FLAG_NEXT_MEMBAR_NONE); gpu->parent->ce_hal->memcopy_v_to_v(&push, dst_va + CE_TEST_MEM_END_OFFSET, src_va + CE_TEST_MEM_END_OFFSET, CE_TEST_MEM_END_SIZE); uvm_push_set_flag(&push, UVM_PUSH_FLAG_NEXT_MEMBAR_NONE); uvm_push_set_flag(&push, UVM_PUSH_FLAG_CE_NEXT_PIPELINED); gpu->parent->ce_hal->memcopy_v_to_v(&push, dst_va + CE_TEST_MEM_MIDDLE_OFFSET, src_va + CE_TEST_MEM_MIDDLE_OFFSET, CE_TEST_MEM_MIDDLE_SIZE); uvm_push_set_flag(&push, UVM_PUSH_FLAG_NEXT_MEMBAR_NONE); uvm_push_set_flag(&push, UVM_PUSH_FLAG_CE_NEXT_PIPELINED); gpu->parent->ce_hal->memcopy_v_to_v(&push, dst_va, src_va, CE_TEST_MEM_BEGIN_SIZE); } else { uvm_push_set_flag(&push, UVM_PUSH_FLAG_NEXT_MEMBAR_NONE); gpu->parent->ce_hal->memcopy_v_to_v(&push, dst_va, src_va, CE_TEST_MEM_BEGIN_SIZE); uvm_push_set_flag(&push, UVM_PUSH_FLAG_NEXT_MEMBAR_NONE); uvm_push_set_flag(&push, UVM_PUSH_FLAG_CE_NEXT_PIPELINED); gpu->parent->ce_hal->memcopy_v_to_v(&push, dst_va + CE_TEST_MEM_MIDDLE_OFFSET, src_va + CE_TEST_MEM_MIDDLE_OFFSET, CE_TEST_MEM_MIDDLE_SIZE); uvm_push_set_flag(&push, UVM_PUSH_FLAG_NEXT_MEMBAR_NONE); uvm_push_set_flag(&push, UVM_PUSH_FLAG_CE_NEXT_PIPELINED); gpu->parent->ce_hal->memcopy_v_to_v(&push, dst_va + CE_TEST_MEM_END_OFFSET, src_va + CE_TEST_MEM_END_OFFSET, CE_TEST_MEM_END_SIZE); } } status = uvm_push_end_and_wait(&push); TEST_CHECK_GOTO(status == NV_OK, done); for (i = 0; i < CE_TEST_MEM_SIZE / sizeof(NvU32); ++i) { if (host_ptr[i] != 1) { UVM_TEST_PRINT("host_ptr[%u] = %u instead of 1\n", i, host_ptr[i]); status = NV_ERR_INVALID_STATE; goto done; } } done: for (i = 0; i < CE_TEST_MEM_COUNT; ++i) { uvm_rm_mem_free(mem[i]); } uvm_rm_mem_free(host_mem); return status; } #define REDUCTIONS 32 static NV_STATUS test_membar(uvm_gpu_t *gpu) { NvU32 i; NV_STATUS status; uvm_rm_mem_t *host_mem = NULL; NvU32 *host_ptr; NvU64 host_mem_gpu_va; uvm_push_t push; NvU32 value; // TODO: Bug 3839176: the test is waived on Confidential Computing because // it assumes that GPU can access system memory without using encryption. if (g_uvm_global.conf_computing_enabled) return NV_OK; status = uvm_rm_mem_alloc_and_map_cpu(gpu, UVM_RM_MEM_TYPE_SYS, sizeof(NvU32), 0, &host_mem); TEST_CHECK_GOTO(status == NV_OK, done); host_ptr = (NvU32 *)uvm_rm_mem_get_cpu_va(host_mem); *host_ptr = 0; status = uvm_push_begin(gpu->channel_manager, UVM_CHANNEL_TYPE_GPU_TO_CPU, &push, "Membar test"); TEST_CHECK_GOTO(status == NV_OK, done); host_mem_gpu_va = uvm_rm_mem_get_gpu_va(host_mem, gpu, uvm_channel_is_proxy(push.channel)).address; for (i = 0; i < REDUCTIONS; ++i) { uvm_push_set_flag(&push, UVM_PUSH_FLAG_NEXT_MEMBAR_NONE); gpu->parent->ce_hal->semaphore_reduction_inc(&push, host_mem_gpu_va, REDUCTIONS); } // Without a sys membar the channel tracking semaphore can and does complete // before all the reductions. status = uvm_push_end_and_wait(&push); TEST_CHECK_GOTO(status == NV_OK, done); value = *host_ptr; if (value != REDUCTIONS) { UVM_TEST_PRINT("Value = %u instead of %u, GPU %s\n", value, REDUCTIONS, uvm_gpu_name(gpu)); status = NV_ERR_INVALID_STATE; goto done; } done: uvm_rm_mem_free(host_mem); return status; } static void push_memset(uvm_push_t *push, uvm_gpu_address_t dst, NvU64 value, size_t element_size, size_t size) { switch (element_size) { case 1: uvm_push_get_gpu(push)->parent->ce_hal->memset_1(push, dst, (NvU8)value, size); break; case 4: uvm_push_get_gpu(push)->parent->ce_hal->memset_4(push, dst, (NvU32)value, size); break; case 8: uvm_push_get_gpu(push)->parent->ce_hal->memset_8(push, dst, value, size); break; default: UVM_ASSERT(0); } } static NV_STATUS test_unaligned_memset(uvm_gpu_t *gpu, uvm_gpu_address_t gpu_verif_addr, NvU8 *cpu_verif_addr, size_t size, size_t element_size, size_t offset) { uvm_push_t push; NV_STATUS status; size_t i; NvU64 value64 = (offset + 2) * (1ull << 32) + (offset + 1); NvU64 test_value, expected_value = 0; uvm_gpu_address_t dst; // Copy a single element at an unaligned position and make sure it doesn't // clobber anything else TEST_CHECK_RET(gpu_verif_addr.address % element_size == 0); TEST_CHECK_RET(offset + element_size <= size); dst = gpu_verif_addr; dst.address += offset; memset(cpu_verif_addr, (NvU8)(~value64), size); status = uvm_push_begin(gpu->channel_manager, UVM_CHANNEL_TYPE_GPU_INTERNAL, &push, "memset_%zu offset %zu", element_size, offset); TEST_CHECK_RET(status == NV_OK); push_memset(&push, dst, value64, element_size, element_size); status = uvm_push_end_and_wait(&push); TEST_CHECK_RET(status == NV_OK); // Make sure all bytes of element are present test_value = 0; memcpy(&test_value, cpu_verif_addr + offset, element_size); switch (element_size) { case 1: expected_value = (NvU8)value64; break; case 4: expected_value = (NvU32)value64; break; case 8: expected_value = value64; break; default: UVM_ASSERT(0); } if (test_value != expected_value) { UVM_TEST_PRINT("memset_%zu offset %zu failed, written value is 0x%llx instead of 0x%llx\n", element_size, offset, test_value, expected_value); return NV_ERR_INVALID_STATE; } // Make sure all other bytes are unchanged for (i = 0; i < size; i++) { if (i >= offset && i < offset + element_size) continue; if (cpu_verif_addr[i] != (NvU8)(~value64)) { UVM_TEST_PRINT("memset_%zu offset %zu failed, immutable byte %zu changed value from 0x%x to 0x%x\n", element_size, offset, i, (NvU8)(~value64), cpu_verif_addr[i]); return NV_ERR_INVALID_STATE; } } return NV_OK; } static NV_STATUS test_memcpy_and_memset_inner(uvm_gpu_t *gpu, uvm_gpu_address_t dst, uvm_gpu_address_t src, size_t size, size_t element_size, uvm_gpu_address_t gpu_verif_addr, void *cpu_verif_addr, int test_iteration) { uvm_push_t push; size_t i; const char *src_type = src.is_virtual ? "virtual" : "physical"; const char *src_loc = src.aperture == UVM_APERTURE_SYS ? "sysmem" : "vidmem"; const char *dst_type = dst.is_virtual ? "virtual" : "physical"; const char *dst_loc = dst.aperture == UVM_APERTURE_SYS ? "sysmem" : "vidmem"; NvU64 value64 = (test_iteration + 2) * (1ull << 32) + (test_iteration + 1); NvU64 test_value = 0, expected_value = 0; TEST_NV_CHECK_RET(uvm_push_begin(gpu->channel_manager, UVM_CHANNEL_TYPE_GPU_INTERNAL, &push, "Memset %s %s (0x%llx) and memcopy to %s %s (0x%llx), iter %d", src_type, src_loc, src.address, dst_type, dst_loc, dst.address, test_iteration)); // Waive if any of the input addresses is physical but the channel does not // support physical addressing if (!uvm_channel_is_privileged(push.channel) && (!dst.is_virtual || !src.is_virtual)) { TEST_NV_CHECK_RET(uvm_push_end_and_wait(&push)); return NV_OK; } // The input virtual addresses exist in UVM's internal address space, not // the proxy address space if (uvm_channel_is_proxy(push.channel)) { TEST_NV_CHECK_RET(uvm_push_end_and_wait(&push)); return NV_ERR_INVALID_STATE; } // If physical accesses aren't supported, silently convert to virtual to // test the flat mapping. TEST_CHECK_RET(gpu_verif_addr.is_virtual); if (!src.is_virtual) src = uvm_gpu_address_copy(gpu, uvm_gpu_phys_address(src.aperture, src.address)); if (!dst.is_virtual) dst = uvm_gpu_address_copy(gpu, uvm_gpu_phys_address(dst.aperture, dst.address)); // Memset src with the appropriate element size, then memcpy to dst and from // dst to the verif location (physical sysmem). push_memset(&push, src, value64, element_size, size); gpu->parent->ce_hal->memcopy(&push, dst, src, size); gpu->parent->ce_hal->memcopy(&push, gpu_verif_addr, dst, size); TEST_NV_CHECK_RET(uvm_push_end_and_wait(&push)); for (i = 0; i < size / element_size; i++) { switch (element_size) { case 1: expected_value = (NvU8)value64; test_value = ((NvU8 *)cpu_verif_addr)[i]; break; case 4: expected_value = (NvU32)value64; test_value = ((NvU32 *)cpu_verif_addr)[i]; break; case 8: expected_value = value64; test_value = ((NvU64 *)cpu_verif_addr)[i]; break; default: UVM_ASSERT(0); } if (test_value != expected_value) { UVM_TEST_PRINT("memset_%zu of %s %s and memcpy into %s %s failed, value[%zu] = 0x%llx instead of 0x%llx\n", element_size, src_type, src_loc, dst_type, dst_loc, i, test_value, expected_value); return NV_ERR_INVALID_STATE; } } return NV_OK; } static NV_STATUS test_memcpy_and_memset(uvm_gpu_t *gpu) { NV_STATUS status = NV_OK; bool is_proxy_va_space = false; uvm_gpu_address_t gpu_verif_addr; void *cpu_verif_addr; uvm_mem_t *verif_mem = NULL; uvm_mem_t *sys_uvm_mem = NULL; uvm_mem_t *gpu_uvm_mem = NULL; uvm_rm_mem_t *sys_rm_mem = NULL; uvm_rm_mem_t *gpu_rm_mem = NULL; uvm_gpu_address_t gpu_addresses[4] = {0}; size_t size = gpu->big_page.internal_size; static const size_t element_sizes[] = {1, 4, 8}; const size_t iterations = 4; size_t i, j, k, s; uvm_mem_alloc_params_t mem_params = {0}; if (g_uvm_global.conf_computing_enabled) TEST_NV_CHECK_GOTO(uvm_mem_alloc_sysmem_dma_and_map_cpu_kernel(size, gpu, current->mm, &verif_mem), done); else TEST_NV_CHECK_GOTO(uvm_mem_alloc_sysmem_and_map_cpu_kernel(size, current->mm, &verif_mem), done); TEST_NV_CHECK_GOTO(uvm_mem_map_gpu_kernel(verif_mem, gpu), done); gpu_verif_addr = uvm_mem_gpu_address_virtual_kernel(verif_mem, gpu); cpu_verif_addr = uvm_mem_get_cpu_addr_kernel(verif_mem); for (i = 0; i < iterations; ++i) { for (s = 0; s < ARRAY_SIZE(element_sizes); s++) { TEST_NV_CHECK_GOTO(test_unaligned_memset(gpu, gpu_verif_addr, cpu_verif_addr, size, element_sizes[s], i), done); } } // Virtual address (in UVM's internal address space) backed by sysmem TEST_NV_CHECK_GOTO(uvm_rm_mem_alloc(gpu, UVM_RM_MEM_TYPE_SYS, size, 0, &sys_rm_mem), done); gpu_addresses[0] = uvm_rm_mem_get_gpu_va(sys_rm_mem, gpu, is_proxy_va_space); if (g_uvm_global.conf_computing_enabled) { for (i = 0; i < iterations; ++i) { for (s = 0; s < ARRAY_SIZE(element_sizes); s++) { TEST_NV_CHECK_GOTO(test_memcpy_and_memset_inner(gpu, gpu_addresses[0], gpu_addresses[0], size, element_sizes[s], gpu_verif_addr, cpu_verif_addr, i), done); } } // Because gpu_verif_addr is in sysmem, when the Confidential // Computing feature is enabled, only the previous cases are valid. // TODO: Bug 3839176: the test partially waived on Confidential // Computing because it assumes that GPU can access system memory // without using encryption. goto done; } // Using a page size equal to the allocation size ensures that the UVM // memories about to be allocated are physically contiguous. And since the // size is a valid GPU page size, the memories can be virtually mapped on // the GPU if needed. mem_params.size = size; mem_params.page_size = size; mem_params.mm = current->mm; // Physical address in sysmem TEST_NV_CHECK_GOTO(uvm_mem_alloc(&mem_params, &sys_uvm_mem), done); TEST_NV_CHECK_GOTO(uvm_mem_map_gpu_phys(sys_uvm_mem, gpu), done); gpu_addresses[1] = uvm_mem_gpu_address_physical(sys_uvm_mem, gpu, 0, size); // Physical address in vidmem mem_params.backing_gpu = gpu; TEST_NV_CHECK_GOTO(uvm_mem_alloc(&mem_params, &gpu_uvm_mem), done); gpu_addresses[2] = uvm_mem_gpu_address_physical(gpu_uvm_mem, gpu, 0, size); // Virtual address (in UVM's internal address space) backed by vidmem TEST_NV_CHECK_GOTO(uvm_rm_mem_alloc(gpu, UVM_RM_MEM_TYPE_GPU, size, 0, &gpu_rm_mem), done); gpu_addresses[3] = uvm_rm_mem_get_gpu_va(gpu_rm_mem, gpu, is_proxy_va_space); for (i = 0; i < iterations; ++i) { for (j = 0; j < ARRAY_SIZE(gpu_addresses); ++j) { for (k = 0; k < ARRAY_SIZE(gpu_addresses); ++k) { for (s = 0; s < ARRAY_SIZE(element_sizes); s++) { TEST_NV_CHECK_GOTO(test_memcpy_and_memset_inner(gpu, gpu_addresses[k], gpu_addresses[j], size, element_sizes[s], gpu_verif_addr, cpu_verif_addr, i), done); } } } } done: uvm_rm_mem_free(sys_rm_mem); uvm_rm_mem_free(gpu_rm_mem); uvm_mem_free(gpu_uvm_mem); uvm_mem_free(sys_uvm_mem); uvm_mem_free(verif_mem); return status; } static NV_STATUS test_semaphore_alloc_sem(uvm_gpu_t *gpu, size_t size, uvm_mem_t **mem_out) { NvU64 gpu_va; NV_STATUS status = NV_OK; uvm_mem_t *mem = NULL; TEST_NV_CHECK_RET(uvm_mem_alloc_sysmem_and_map_cpu_kernel(size, current->mm, &mem)); TEST_NV_CHECK_GOTO(uvm_mem_map_gpu_kernel(mem, gpu), error); gpu_va = uvm_mem_get_gpu_va_kernel(mem, gpu); // This semaphore resides in the uvm_mem region, i.e., it has the GPU VA // MSbit set. The intent is to validate semaphore operations when the // semaphore's VA is in the high-end of the GPU effective virtual address // space spectrum, i.e., its VA upper-bit is set. TEST_CHECK_GOTO(gpu_va & (1ULL << (gpu->address_space_tree.hal->num_va_bits() - 1)), error); *mem_out = mem; return NV_OK; error: uvm_mem_free(mem); return status; } // test_semaphore_reduction_inc is similar in concept to test_membar(). It uses // uvm_mem (instead of uvm_rm_mem) as the semaphore, i.e., it assumes that the // CE HAL has been validated, since uvm_mem needs the CE memset/memcopy to be // operational as a pre-requisite for GPU PTE writes. The purpose of // test_semaphore_reduction_inc is to validate the reduction inc operation on // semaphores with their VA's upper-bit set. static NV_STATUS test_semaphore_reduction_inc(uvm_gpu_t *gpu) { NV_STATUS status; uvm_push_t push; uvm_mem_t *mem; NvU64 gpu_va; NvU32 i; NvU32 *host_ptr = NULL; NvU32 value; // Semaphore reduction needs 1 word (4 bytes). const size_t size = sizeof(NvU32); // TODO: Bug 3839176: the test is waived on Confidential Computing because // it assumes that GPU can access system memory without using encryption. if (g_uvm_global.conf_computing_enabled) return NV_OK; status = test_semaphore_alloc_sem(gpu, size, &mem); TEST_CHECK_RET(status == NV_OK); // Initialize the counter of reductions. host_ptr = uvm_mem_get_cpu_addr_kernel(mem); TEST_CHECK_GOTO(host_ptr != NULL, done); *host_ptr = 0; gpu_va = uvm_mem_get_gpu_va_kernel(mem, gpu); status = uvm_push_begin(gpu->channel_manager, UVM_CHANNEL_TYPE_GPU_INTERNAL, &push, "semaphore_reduction_inc test"); TEST_CHECK_GOTO(status == NV_OK, done); for (i = 0; i < REDUCTIONS; i++) { uvm_push_set_flag(&push, UVM_PUSH_FLAG_NEXT_MEMBAR_NONE); gpu->parent->ce_hal->semaphore_reduction_inc(&push, gpu_va, REDUCTIONS); } status = uvm_push_end_and_wait(&push); TEST_CHECK_GOTO(status == NV_OK, done); value = *host_ptr; if (value != REDUCTIONS) { UVM_TEST_PRINT("Value = %u instead of %u, GPU %s\n", value, REDUCTIONS, uvm_gpu_name(gpu)); status = NV_ERR_INVALID_STATE; goto done; } done: uvm_mem_free(mem); return status; } static NV_STATUS test_semaphore_release(uvm_gpu_t *gpu) { NV_STATUS status; uvm_push_t push; uvm_mem_t *mem; NvU64 gpu_va; NvU32 value; NvU32 *host_ptr = NULL; NvU32 payload = 0xA5A55A5A; // Semaphore release needs 1 word (4 bytes). const size_t size = sizeof(NvU32); // TODO: Bug 3839176: the test is waived on Confidential Computing because // it assumes that GPU can access system memory without using encryption. if (g_uvm_global.conf_computing_enabled) return NV_OK; status = test_semaphore_alloc_sem(gpu, size, &mem); TEST_CHECK_RET(status == NV_OK); // Initialize the payload. host_ptr = uvm_mem_get_cpu_addr_kernel(mem); TEST_CHECK_GOTO(host_ptr != NULL, done); *host_ptr = 0; gpu_va = uvm_mem_get_gpu_va_kernel(mem, gpu); status = uvm_push_begin(gpu->channel_manager, UVM_CHANNEL_TYPE_GPU_INTERNAL, &push, "semaphore_release test"); TEST_CHECK_GOTO(status == NV_OK, done); gpu->parent->ce_hal->semaphore_release(&push, gpu_va, payload); status = uvm_push_end_and_wait(&push); TEST_CHECK_GOTO(status == NV_OK, done); value = *host_ptr; if (value != payload) { UVM_TEST_PRINT("Semaphore payload = %u instead of %u, GPU %s\n", value, payload, uvm_gpu_name(gpu)); status = NV_ERR_INVALID_STATE; goto done; } done: uvm_mem_free(mem); return status; } static NV_STATUS test_semaphore_timestamp(uvm_gpu_t *gpu) { NV_STATUS status; uvm_push_t push; uvm_mem_t *mem; NvU64 gpu_va; NvU32 i; NvU64 *timestamp; NvU64 last_timestamp = 0; // 2 iterations: // 1: compare retrieved timestamp with 0; // 2: compare retrieved timestamp with previous timestamp (obtained in 1). const NvU32 iterations = 2; // The semaphore is 4 words long (16 bytes). const size_t size = 16; // TODO: Bug 3839176: the test is waived on Confidential Computing because // it assumes that GPU can access system memory without using encryption. if (g_uvm_global.conf_computing_enabled) return NV_OK; status = test_semaphore_alloc_sem(gpu, size, &mem); TEST_CHECK_RET(status == NV_OK); timestamp = uvm_mem_get_cpu_addr_kernel(mem); TEST_CHECK_GOTO(timestamp != NULL, done); memset(timestamp, 0, size); // Shift the timestamp pointer to where the semaphore timestamp info is. timestamp += 1; gpu_va = uvm_mem_get_gpu_va_kernel(mem, gpu); for (i = 0; i < iterations; i++) { status = uvm_push_begin(gpu->channel_manager, UVM_CHANNEL_TYPE_GPU_INTERNAL, &push, "semaphore_timestamp test, iter: %u", i); TEST_CHECK_GOTO(status == NV_OK, done); gpu->parent->ce_hal->semaphore_timestamp(&push, gpu_va); status = uvm_push_end_and_wait(&push); TEST_CHECK_GOTO(status == NV_OK, done); TEST_CHECK_GOTO(*timestamp != 0, done); TEST_CHECK_GOTO(*timestamp >= last_timestamp, done); last_timestamp = *timestamp; } done: uvm_mem_free(mem); return status; } static bool mem_match(uvm_mem_t *mem1, uvm_mem_t *mem2, size_t size) { void *mem1_addr; void *mem2_addr; UVM_ASSERT(uvm_mem_is_sysmem(mem1)); UVM_ASSERT(uvm_mem_is_sysmem(mem2)); UVM_ASSERT(mem1->size >= size); UVM_ASSERT(mem2->size >= size); mem1_addr = uvm_mem_get_cpu_addr_kernel(mem1); mem2_addr = uvm_mem_get_cpu_addr_kernel(mem2); return !memcmp(mem1_addr, mem2_addr, size); } static NV_STATUS zero_vidmem(uvm_mem_t *mem) { uvm_push_t push; uvm_gpu_address_t gpu_address; uvm_gpu_t *gpu = mem->backing_gpu; UVM_ASSERT(uvm_mem_is_vidmem(mem)); TEST_NV_CHECK_RET(uvm_push_begin(gpu->channel_manager, UVM_CHANNEL_TYPE_GPU_INTERNAL, &push, "zero vidmem")); gpu_address = uvm_mem_gpu_address_virtual_kernel(mem, gpu); gpu->parent->ce_hal->memset_1(&push, gpu_address, 0, mem->size); TEST_NV_CHECK_RET(uvm_push_end_and_wait(&push)); return NV_OK; } static void write_range_cpu(uvm_mem_t *mem, NvU64 base_val) { NvU64 *mem_cpu_va; unsigned i; UVM_ASSERT(uvm_mem_is_sysmem(mem)); UVM_ASSERT(IS_ALIGNED(mem->size, sizeof(*mem_cpu_va))); mem_cpu_va = (NvU64 *) uvm_mem_get_cpu_addr_kernel(mem); for (i = 0; i < (mem->size / sizeof(*mem_cpu_va)); i++) mem_cpu_va[i] = base_val++; } static NV_STATUS alloc_vidmem_protected(uvm_gpu_t *gpu, uvm_mem_t **mem, size_t size) { NV_STATUS status; UVM_ASSERT(mem); *mem = NULL; TEST_NV_CHECK_RET(uvm_mem_alloc_vidmem(size, gpu, mem)); TEST_NV_CHECK_GOTO(uvm_mem_map_gpu_kernel(*mem, gpu), err); TEST_NV_CHECK_GOTO(zero_vidmem(*mem), err); return NV_OK; err: uvm_mem_free(*mem); return status; } static NV_STATUS alloc_sysmem_unprotected(uvm_gpu_t *gpu, uvm_mem_t **mem, size_t size) { NV_STATUS status; UVM_ASSERT(mem); *mem = NULL; TEST_NV_CHECK_RET(uvm_mem_alloc_sysmem_dma(size, gpu, NULL, mem)); TEST_NV_CHECK_GOTO(uvm_mem_map_cpu_kernel(*mem), err); TEST_NV_CHECK_GOTO(uvm_mem_map_gpu_kernel(*mem, gpu), err); memset(uvm_mem_get_cpu_addr_kernel(*mem), 0, (*mem)->size); return NV_OK; err: uvm_mem_free(*mem); return status; } static void cpu_encrypt(uvm_channel_t *channel, uvm_mem_t *dst_mem, uvm_mem_t *src_mem, uvm_mem_t *auth_tag_mem, size_t size, NvU32 copy_size) { size_t offset = 0; char *src_plain = (char *) uvm_mem_get_cpu_addr_kernel(src_mem); char *dst_cipher = (char *) uvm_mem_get_cpu_addr_kernel(dst_mem); char *auth_tag_buffer = (char *) uvm_mem_get_cpu_addr_kernel(auth_tag_mem); while (offset < size) { uvm_conf_computing_cpu_encrypt(channel, dst_cipher, src_plain, NULL, copy_size, auth_tag_buffer); offset += copy_size; dst_cipher += copy_size; src_plain += copy_size; auth_tag_buffer += UVM_CONF_COMPUTING_AUTH_TAG_SIZE; } } static void cpu_acquire_encryption_ivs(uvm_channel_t *channel, size_t size, NvU32 copy_size, UvmCslIv *ivs) { size_t offset = 0; int i = 0; for (; offset < size; offset += copy_size) uvm_conf_computing_acquire_encryption_iv(channel, &ivs[i++]); } static void cpu_encrypt_rev(uvm_channel_t *channel, uvm_mem_t *dst_mem, uvm_mem_t *src_mem, uvm_mem_t *auth_tag_mem, size_t size, NvU32 copy_size, UvmCslIv *encrypt_iv) { char *src_plain = (char *) uvm_mem_get_cpu_addr_kernel(src_mem); char *dst_cipher = (char *) uvm_mem_get_cpu_addr_kernel(dst_mem); char *auth_tag_buffer = (char *) uvm_mem_get_cpu_addr_kernel(auth_tag_mem); int i; // CPU encrypt order is the opposite of the GPU decrypt order for (i = (size / copy_size) - 1; i >= 0; i--) { uvm_conf_computing_cpu_encrypt(channel, dst_cipher + i * copy_size, src_plain + i * copy_size, encrypt_iv + i, copy_size, auth_tag_buffer + i * UVM_CONF_COMPUTING_AUTH_TAG_SIZE); } } static NV_STATUS cpu_decrypt_in_order(uvm_channel_t *channel, uvm_mem_t *dst_mem, uvm_mem_t *src_mem, const UvmCslIv *decrypt_iv, NvU32 key_version, uvm_mem_t *auth_tag_mem, size_t size, NvU32 copy_size) { size_t i; char *dst_plain = (char *) uvm_mem_get_cpu_addr_kernel(dst_mem); char *src_cipher = (char *) uvm_mem_get_cpu_addr_kernel(src_mem); char *auth_tag_buffer = (char *) uvm_mem_get_cpu_addr_kernel(auth_tag_mem); for (i = 0; i < size / copy_size; i++) { TEST_NV_CHECK_RET(uvm_conf_computing_cpu_decrypt(channel, dst_plain + i * copy_size, src_cipher + i * copy_size, decrypt_iv + i, key_version, copy_size, auth_tag_buffer + i * UVM_CONF_COMPUTING_AUTH_TAG_SIZE)); } return NV_OK; } static NV_STATUS cpu_decrypt_out_of_order(uvm_channel_t *channel, uvm_mem_t *dst_mem, uvm_mem_t *src_mem, const UvmCslIv *decrypt_iv, NvU32 key_version, uvm_mem_t *auth_tag_mem, size_t size, NvU32 copy_size) { int i; char *dst_plain = (char *) uvm_mem_get_cpu_addr_kernel(dst_mem); char *src_cipher = (char *) uvm_mem_get_cpu_addr_kernel(src_mem); char *auth_tag_buffer = (char *) uvm_mem_get_cpu_addr_kernel(auth_tag_mem); UVM_ASSERT((size / copy_size) <= INT_MAX); // CPU decrypt order is the opposite of the GPU decrypt order for (i = (size / copy_size) - 1; i >= 0; i--) { TEST_NV_CHECK_RET(uvm_conf_computing_cpu_decrypt(channel, dst_plain + i * copy_size, src_cipher + i * copy_size, decrypt_iv + i, key_version, copy_size, auth_tag_buffer + i * UVM_CONF_COMPUTING_AUTH_TAG_SIZE)); } return NV_OK; } // GPU address to use as source or destination in CE decrypt/encrypt operations. // If the uvm_mem backing storage is contiguous in the [offset, offset + size) // interval, the physical address gets priority over the virtual counterpart. static uvm_gpu_address_t gpu_address(uvm_mem_t *mem, uvm_gpu_t *gpu, NvU64 offset, NvU32 size) { uvm_gpu_address_t gpu_virtual_address; if (uvm_mem_is_physically_contiguous(mem, offset, size)) return uvm_mem_gpu_address_physical(mem, gpu, offset, size); gpu_virtual_address = uvm_mem_gpu_address_virtual_kernel(mem, gpu); gpu_virtual_address.address += offset; return gpu_virtual_address; } // Automatically get the correct address for the authentication tag. The // addressing mode of the tag should match that of the reference address // (destination pointer for GPU encrypt, source pointer for GPU encrypt) static uvm_gpu_address_t auth_tag_gpu_address(uvm_mem_t *auth_tag_mem, uvm_gpu_t *gpu, size_t offset, uvm_gpu_address_t reference) { uvm_gpu_address_t auth_tag_gpu_address; if (!reference.is_virtual) return uvm_mem_gpu_address_physical(auth_tag_mem, gpu, offset, UVM_CONF_COMPUTING_AUTH_TAG_SIZE); auth_tag_gpu_address = uvm_mem_gpu_address_virtual_kernel(auth_tag_mem, gpu); auth_tag_gpu_address.address += offset; return auth_tag_gpu_address; } // Note: no membar is issued in any of the GPU transfers (encryptions) static void gpu_encrypt(uvm_push_t *push, uvm_mem_t *dst_mem, uvm_mem_t *src_mem, uvm_mem_t *auth_tag_mem, UvmCslIv *decrypt_iv, size_t size, NvU32 copy_size) { size_t i; size_t num_iterations = size / copy_size; uvm_gpu_t *gpu = uvm_push_get_gpu(push); for (i = 0; i < num_iterations; i++) { uvm_gpu_address_t dst_cipher = gpu_address(dst_mem, gpu, i * copy_size, copy_size); uvm_gpu_address_t src_plain = gpu_address(src_mem, gpu, i * copy_size, copy_size); uvm_gpu_address_t auth_tag = auth_tag_gpu_address(auth_tag_mem, gpu, i * UVM_CONF_COMPUTING_AUTH_TAG_SIZE, dst_cipher); uvm_conf_computing_log_gpu_encryption(push->channel, copy_size, decrypt_iv); if (i > 0) uvm_push_set_flag(push, UVM_PUSH_FLAG_CE_NEXT_PIPELINED); uvm_push_set_flag(push, UVM_PUSH_FLAG_NEXT_MEMBAR_NONE); gpu->parent->ce_hal->encrypt(push, dst_cipher, src_plain, copy_size, auth_tag); decrypt_iv++; } } // Note: no membar is issued in any of the GPU transfers (decryptions) static void gpu_decrypt(uvm_push_t *push, uvm_mem_t *dst_mem, uvm_mem_t *src_mem, uvm_mem_t *auth_tag_mem, size_t size, NvU32 copy_size) { size_t i; size_t num_iterations = size / copy_size; uvm_gpu_t *gpu = uvm_push_get_gpu(push); for (i = 0; i < num_iterations; i++) { uvm_gpu_address_t dst_plain = gpu_address(dst_mem, gpu, i * copy_size, copy_size); uvm_gpu_address_t src_cipher = gpu_address(src_mem, gpu, i * copy_size, copy_size); uvm_gpu_address_t auth_tag = auth_tag_gpu_address(auth_tag_mem, gpu, i * UVM_CONF_COMPUTING_AUTH_TAG_SIZE, src_cipher); if (i > 0) uvm_push_set_flag(push, UVM_PUSH_FLAG_CE_NEXT_PIPELINED); uvm_push_set_flag(push, UVM_PUSH_FLAG_NEXT_MEMBAR_NONE); gpu->parent->ce_hal->decrypt(push, dst_plain, src_cipher, copy_size, auth_tag); } } static NV_STATUS test_cpu_to_gpu_roundtrip(uvm_gpu_t *gpu, uvm_channel_type_t decrypt_channel_type, uvm_channel_type_t encrypt_channel_type, size_t size, NvU32 copy_size, bool decrypt_in_order, bool encrypt_in_order) { uvm_push_t push; NvU64 init_value; NV_STATUS status = NV_OK; uvm_mem_t *src_plain = NULL; uvm_mem_t *src_cipher = NULL; uvm_mem_t *dst_cipher = NULL; uvm_mem_t *dst_plain_gpu = NULL; uvm_mem_t *dst_plain = NULL; uvm_mem_t *auth_tag_mem = NULL; size_t auth_tag_buffer_size = (size / copy_size) * UVM_CONF_COMPUTING_AUTH_TAG_SIZE; UvmCslIv *decrypt_iv = NULL; UvmCslIv *encrypt_iv = NULL; NvU32 key_version; uvm_tracker_t tracker; size_t src_plain_size; TEST_CHECK_RET(copy_size <= size); TEST_CHECK_RET(IS_ALIGNED(size, copy_size)); uvm_tracker_init(&tracker); decrypt_iv = uvm_kvmalloc_zero((size / copy_size) * sizeof(UvmCslIv)); if (!decrypt_iv) { status = NV_ERR_NO_MEMORY; goto out; } encrypt_iv = uvm_kvmalloc_zero((size / copy_size) * sizeof(UvmCslIv)); if (!encrypt_iv) { status = NV_ERR_NO_MEMORY; goto out; } TEST_NV_CHECK_GOTO(alloc_sysmem_unprotected(gpu, &src_cipher, size), out); TEST_NV_CHECK_GOTO(alloc_vidmem_protected(gpu, &dst_plain_gpu, size), out); TEST_NV_CHECK_GOTO(alloc_sysmem_unprotected(gpu, &dst_cipher, size), out); TEST_NV_CHECK_GOTO(alloc_sysmem_unprotected(gpu, &dst_plain, size), out); TEST_NV_CHECK_GOTO(alloc_sysmem_unprotected(gpu, &auth_tag_mem, auth_tag_buffer_size), out); // The plaintext CPU buffer size should fit the initialization value src_plain_size = UVM_ALIGN_UP(size, sizeof(init_value)); TEST_NV_CHECK_GOTO(alloc_sysmem_unprotected(gpu, &src_plain, src_plain_size), out); // Initialize the plaintext CPU buffer using a value that uniquely // identifies the given inputs TEST_CHECK_GOTO((((NvU64) size) < (1ULL << 63)), out); init_value = ((NvU64) decrypt_in_order << 63) | ((NvU64) size) | ((NvU64) copy_size); write_range_cpu(src_plain, init_value); TEST_NV_CHECK_GOTO(uvm_push_begin(gpu->channel_manager, decrypt_channel_type, &push, "CPU > GPU decrypt"), out); // CPU (decrypted) > CPU (encrypted), using CPU, if in-order // acquire IVs if not in-order if (encrypt_in_order) cpu_encrypt(push.channel, src_cipher, src_plain, auth_tag_mem, size, copy_size); else cpu_acquire_encryption_ivs(push.channel, size, copy_size, encrypt_iv); // CPU (encrypted) > GPU (decrypted), using GPU gpu_decrypt(&push, dst_plain_gpu, src_cipher, auth_tag_mem, size, copy_size); // Use acquired IVs to encrypt in reverse order if (!encrypt_in_order) cpu_encrypt_rev(push.channel, src_cipher, src_plain, auth_tag_mem, size, copy_size, encrypt_iv); uvm_push_end(&push); TEST_NV_CHECK_GOTO(uvm_tracker_add_push(&tracker, &push), out); // GPU (decrypted) > CPU (encrypted), using GPU TEST_NV_CHECK_GOTO(uvm_push_begin_acquire(gpu->channel_manager, encrypt_channel_type, &tracker, &push, "GPU > CPU encrypt"), out); gpu_encrypt(&push, dst_cipher, dst_plain_gpu, auth_tag_mem, decrypt_iv, size, copy_size); // There shouldn't be any key rotation between the end of the push and the // CPU decryption(s), but it is more robust against test changes to force // decryption to use the saved key. key_version = uvm_channel_pool_key_version(push.channel->pool); TEST_NV_CHECK_GOTO(uvm_push_end_and_wait(&push), out); TEST_CHECK_GOTO(!mem_match(src_plain, src_cipher, size), out); TEST_CHECK_GOTO(!mem_match(dst_cipher, src_plain, size), out); // CPU (encrypted) > CPU (decrypted), using CPU if (decrypt_in_order) { TEST_NV_CHECK_GOTO(cpu_decrypt_in_order(push.channel, dst_plain, dst_cipher, decrypt_iv, key_version, auth_tag_mem, size, copy_size), out); } else { TEST_NV_CHECK_GOTO(cpu_decrypt_out_of_order(push.channel, dst_plain, dst_cipher, decrypt_iv, key_version, auth_tag_mem, size, copy_size), out); } TEST_CHECK_GOTO(mem_match(src_plain, dst_plain, size), out); out: uvm_mem_free(auth_tag_mem); uvm_mem_free(dst_plain); uvm_mem_free(dst_plain_gpu); uvm_mem_free(dst_cipher); uvm_mem_free(src_cipher); uvm_mem_free(src_plain); uvm_tracker_deinit(&tracker); uvm_kvfree(decrypt_iv); uvm_kvfree(encrypt_iv); return status; } static NV_STATUS test_encryption_decryption(uvm_gpu_t *gpu, uvm_channel_type_t decrypt_channel_type, uvm_channel_type_t encrypt_channel_type) { bool cpu_decrypt_in_order = true; bool cpu_encrypt_in_order = true; size_t size[] = {UVM_PAGE_SIZE_4K, UVM_PAGE_SIZE_4K * 2, UVM_PAGE_SIZE_2M}; size_t copy_size[] = {UVM_PAGE_SIZE_4K, UVM_PAGE_SIZE_64K, UVM_PAGE_SIZE_2M}; unsigned i; struct { bool encrypt_in_order; bool decrypt_in_order; } orders[] = {{true, true}, {true, false}, {false, true}, {false, false}}; struct { size_t size; NvU32 copy_size; } small_sizes[] = {{1, 1}, {3, 1}, {8, 1}, {2, 2}, {8, 4}, {UVM_PAGE_SIZE_4K - 8, 8}, {UVM_PAGE_SIZE_4K + 8, 8}}; // Only Confidential Computing uses CE encryption/decryption if (!g_uvm_global.conf_computing_enabled) return NV_OK; // Use a size, and copy size, that are not a multiple of common page sizes. for (i = 0; i < ARRAY_SIZE(small_sizes); ++i) { // Skip tests that need large pushbuffer on WLC. Secure work launch // needs to do at least one decrypt operation so tests that only need // one operation work ok. Tests using more operations might overflow // UVM_MAX_WLC_PUSH_SIZE. if (encrypt_channel_type == UVM_CHANNEL_TYPE_WLC && (small_sizes[i].size / small_sizes[i].copy_size > 1)) continue; TEST_NV_CHECK_RET(test_cpu_to_gpu_roundtrip(gpu, decrypt_channel_type, encrypt_channel_type, small_sizes[i].size, small_sizes[i].copy_size, cpu_decrypt_in_order, cpu_encrypt_in_order)); } // Use sizes, and copy sizes, that are a multiple of common page sizes. // This is the most typical usage of encrypt/decrypt in the UVM driver. for (i = 0; i < ARRAY_SIZE(orders); ++i) { unsigned j; cpu_encrypt_in_order = orders[i].encrypt_in_order; cpu_decrypt_in_order = orders[i].decrypt_in_order; for (j = 0; j < ARRAY_SIZE(size); ++j) { unsigned k; for (k = 0; k < ARRAY_SIZE(copy_size); ++k) { if (copy_size[k] > size[j]) continue; // Skip tests that need large pushbuffer on WLC. Secure work // launch needs to do at least one decrypt operation so tests // that only need one operation work ok. Tests using more // operations might overflow UVM_MAX_WLC_PUSH_SIZE. if (encrypt_channel_type == UVM_CHANNEL_TYPE_WLC && (size[j] / copy_size[k] > 1)) continue; // There is no difference between in-order and out-of-order // decryption when encrypting once. if ((copy_size[k] == size[j]) && !cpu_decrypt_in_order) continue; TEST_NV_CHECK_RET(test_cpu_to_gpu_roundtrip(gpu, decrypt_channel_type, encrypt_channel_type, size[j], copy_size[k], cpu_decrypt_in_order, cpu_encrypt_in_order)); } } } return NV_OK; } static NV_STATUS test_ce(uvm_va_space_t *va_space, bool skipTimestampTest) { uvm_gpu_t *gpu; for_each_va_space_gpu(gpu, va_space) { TEST_NV_CHECK_RET(test_non_pipelined(gpu)); TEST_NV_CHECK_RET(test_membar(gpu)); TEST_NV_CHECK_RET(test_memcpy_and_memset(gpu)); TEST_NV_CHECK_RET(test_semaphore_reduction_inc(gpu)); TEST_NV_CHECK_RET(test_semaphore_release(gpu)); if (!skipTimestampTest) TEST_NV_CHECK_RET(test_semaphore_timestamp(gpu)); TEST_NV_CHECK_RET(test_encryption_decryption(gpu, UVM_CHANNEL_TYPE_CPU_TO_GPU, UVM_CHANNEL_TYPE_GPU_TO_CPU)); TEST_NV_CHECK_RET(test_encryption_decryption(gpu, UVM_CHANNEL_TYPE_WLC, UVM_CHANNEL_TYPE_WLC)); } return NV_OK; } NV_STATUS uvm_test_ce_sanity(UVM_TEST_CE_SANITY_PARAMS *params, struct file *filp) { NV_STATUS status; uvm_va_space_t *va_space = uvm_va_space_get(filp); uvm_va_space_down_read_rm(va_space); status = test_ce(va_space, params->skipTimestampTest); if (status != NV_OK) goto done; done: uvm_va_space_up_read_rm(va_space); return status; }