mirror of
https://github.com/NVIDIA/open-gpu-kernel-modules.git
synced 2024-12-13 15:08:59 +01:00
439 lines
13 KiB
C
439 lines
13 KiB
C
/*******************************************************************************
|
|
Copyright (c) 2015-2019 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_tracker.h"
|
|
#include "uvm_push.h"
|
|
#include "uvm_channel.h"
|
|
#include "uvm_kvmalloc.h"
|
|
#include "uvm_gpu.h"
|
|
#include "uvm_global.h"
|
|
#include "uvm_common.h"
|
|
#include "uvm_linux.h"
|
|
|
|
static bool tracker_is_using_static_entries(uvm_tracker_t *tracker)
|
|
{
|
|
return tracker->max_size == ARRAY_SIZE(tracker->static_entries);
|
|
}
|
|
|
|
static void free_entries(uvm_tracker_t *tracker)
|
|
{
|
|
if (tracker_is_using_static_entries(tracker))
|
|
return;
|
|
uvm_kvfree(tracker->dynamic_entries);
|
|
}
|
|
|
|
uvm_tracker_entry_t *uvm_tracker_get_entries(uvm_tracker_t *tracker)
|
|
{
|
|
if (tracker_is_using_static_entries(tracker)) {
|
|
return tracker->static_entries;
|
|
}
|
|
else {
|
|
UVM_ASSERT(tracker->dynamic_entries != NULL);
|
|
return tracker->dynamic_entries;
|
|
}
|
|
}
|
|
|
|
static uvm_tracker_entry_t *get_new_entry(uvm_tracker_t *tracker)
|
|
{
|
|
NV_STATUS status = uvm_tracker_reserve(tracker, 1);
|
|
if (status != NV_OK)
|
|
return NULL;
|
|
UVM_ASSERT(tracker->size < tracker->max_size);
|
|
|
|
return &uvm_tracker_get_entries(tracker)[tracker->size++];
|
|
}
|
|
|
|
NV_STATUS uvm_tracker_init_from(uvm_tracker_t *dst, uvm_tracker_t *src)
|
|
{
|
|
NV_STATUS status;
|
|
uvm_tracker_init(dst);
|
|
status = uvm_tracker_overwrite(dst, src);
|
|
if (status != NV_OK) {
|
|
uvm_tracker_deinit(dst);
|
|
uvm_tracker_init(dst);
|
|
}
|
|
return status;
|
|
}
|
|
|
|
void uvm_tracker_deinit(uvm_tracker_t *tracker)
|
|
{
|
|
free_entries(tracker);
|
|
memset(tracker, 0, sizeof(*tracker));
|
|
}
|
|
|
|
NV_STATUS uvm_tracker_overwrite(uvm_tracker_t *dst, uvm_tracker_t *src)
|
|
{
|
|
NV_STATUS status;
|
|
|
|
uvm_tracker_clear(dst);
|
|
|
|
status = uvm_tracker_reserve(dst, src->size);
|
|
if (status != NV_OK)
|
|
return status;
|
|
|
|
dst->size = src->size;
|
|
memcpy(uvm_tracker_get_entries(dst),
|
|
uvm_tracker_get_entries(src),
|
|
src->size * sizeof(*uvm_tracker_get_entries(dst)));
|
|
|
|
return NV_OK;
|
|
}
|
|
|
|
NV_STATUS uvm_tracker_reserve(uvm_tracker_t *tracker, NvU32 min_free_entries)
|
|
{
|
|
if (tracker->size + min_free_entries > tracker->max_size) {
|
|
// Special case the first resize to jump from 1 all the way to 8.
|
|
// This is based on a guess that if a tracker needs more than 1
|
|
// entry it likely needs much more.
|
|
// TODO: Bug 1764961: Verify that guess.
|
|
NvU32 new_max_size = max((NvU32)8, (NvU32)roundup_pow_of_two(tracker->size + min_free_entries));
|
|
uvm_tracker_entry_t *new_entries;
|
|
|
|
if (tracker_is_using_static_entries(tracker)) {
|
|
new_entries = uvm_kvmalloc(sizeof(*new_entries) * new_max_size);
|
|
if (new_entries)
|
|
memcpy(new_entries, tracker->static_entries, sizeof(*new_entries) * tracker->size);
|
|
} else {
|
|
new_entries = uvm_kvrealloc(tracker->dynamic_entries, sizeof(*new_entries) * new_max_size);
|
|
}
|
|
if (!new_entries)
|
|
return NV_ERR_NO_MEMORY;
|
|
tracker->dynamic_entries = new_entries;
|
|
tracker->max_size = new_max_size;
|
|
}
|
|
UVM_ASSERT(tracker->size + min_free_entries <= tracker->max_size);
|
|
return NV_OK;
|
|
}
|
|
|
|
NV_STATUS uvm_tracker_add_push(uvm_tracker_t *tracker, uvm_push_t *push)
|
|
{
|
|
uvm_tracker_entry_t entry;
|
|
|
|
uvm_push_get_tracker_entry(push, &entry);
|
|
|
|
return uvm_tracker_add_entry(tracker, &entry);
|
|
}
|
|
|
|
NV_STATUS uvm_tracker_add_entry(uvm_tracker_t *tracker, uvm_tracker_entry_t *new_entry)
|
|
{
|
|
uvm_tracker_entry_t *tracker_entry;
|
|
|
|
for_each_tracker_entry(tracker_entry, tracker) {
|
|
if (tracker_entry->channel == new_entry->channel) {
|
|
tracker_entry->value = max(tracker_entry->value, new_entry->value);
|
|
return NV_OK;
|
|
}
|
|
}
|
|
|
|
tracker_entry = get_new_entry(tracker);
|
|
if (tracker_entry == NULL)
|
|
return NV_ERR_NO_MEMORY;
|
|
|
|
*tracker_entry = *new_entry;
|
|
|
|
return NV_OK;
|
|
}
|
|
|
|
void uvm_tracker_overwrite_with_entry(uvm_tracker_t *tracker, uvm_tracker_entry_t *entry)
|
|
{
|
|
NV_STATUS status;
|
|
|
|
uvm_tracker_clear(tracker);
|
|
|
|
// An empty tracker always has space for at least one entry so this cannot
|
|
// fail.
|
|
status = uvm_tracker_add_entry(tracker, entry);
|
|
UVM_ASSERT(status == NV_OK);
|
|
}
|
|
|
|
void uvm_tracker_overwrite_with_push(uvm_tracker_t *tracker, uvm_push_t *push)
|
|
{
|
|
uvm_tracker_entry_t entry;
|
|
|
|
uvm_push_get_tracker_entry(push, &entry);
|
|
|
|
uvm_tracker_overwrite_with_entry(tracker, &entry);
|
|
}
|
|
|
|
static NV_STATUS reserve_for_entries_from_tracker(uvm_tracker_t *dst, uvm_tracker_t *src)
|
|
{
|
|
NvU32 needed_free_entries = 0;
|
|
uvm_tracker_entry_t *src_entry, *dst_entry;
|
|
|
|
for_each_tracker_entry(src_entry, src) {
|
|
bool found = false;
|
|
for_each_tracker_entry(dst_entry, dst) {
|
|
if (dst_entry->channel == src_entry->channel) {
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
if (!found)
|
|
needed_free_entries++;
|
|
}
|
|
|
|
return uvm_tracker_reserve(dst, needed_free_entries);
|
|
}
|
|
|
|
NV_STATUS uvm_tracker_add_tracker(uvm_tracker_t *dst, uvm_tracker_t *src)
|
|
{
|
|
NV_STATUS status;
|
|
uvm_tracker_entry_t *src_entry;
|
|
|
|
if (src == dst)
|
|
return NV_OK;
|
|
|
|
status = uvm_tracker_reserve(dst, src->size);
|
|
if (status == NV_ERR_NO_MEMORY) {
|
|
uvm_tracker_remove_completed(dst);
|
|
uvm_tracker_remove_completed(src);
|
|
status = reserve_for_entries_from_tracker(dst, src);
|
|
}
|
|
if (status != NV_OK) {
|
|
return status;
|
|
}
|
|
|
|
for_each_tracker_entry(src_entry, src) {
|
|
status = uvm_tracker_add_entry(dst, src_entry);
|
|
UVM_ASSERT_MSG(status == NV_OK, "Expected success with reserved memory but got error %d\n", status);
|
|
}
|
|
|
|
return NV_OK;
|
|
}
|
|
|
|
NV_STATUS uvm_tracker_overwrite_safe(uvm_tracker_t *dst, uvm_tracker_t *src)
|
|
{
|
|
NV_STATUS status = uvm_tracker_overwrite(dst, src);
|
|
if (status == NV_ERR_NO_MEMORY) {
|
|
UVM_DBG_PRINT_RL("Failed to overwrite tracker, waiting\n");
|
|
status = uvm_tracker_wait(src);
|
|
}
|
|
return status;
|
|
}
|
|
|
|
NV_STATUS uvm_tracker_add_push_safe(uvm_tracker_t *tracker, uvm_push_t *push)
|
|
{
|
|
NV_STATUS status = uvm_tracker_add_push(tracker, push);
|
|
if (status == NV_ERR_NO_MEMORY) {
|
|
UVM_DBG_PRINT_RL("Failed to add push to tracker, waiting\n");
|
|
status = uvm_push_wait(push);
|
|
}
|
|
return status;
|
|
}
|
|
|
|
NV_STATUS uvm_tracker_add_entry_safe(uvm_tracker_t *tracker, uvm_tracker_entry_t *new_entry)
|
|
{
|
|
NV_STATUS status = uvm_tracker_add_entry(tracker, new_entry);
|
|
if (status == NV_ERR_NO_MEMORY) {
|
|
UVM_DBG_PRINT_RL("Failed to add entry to tracker, waiting\n");
|
|
status = uvm_tracker_wait_for_entry(new_entry);
|
|
}
|
|
return status;
|
|
}
|
|
|
|
NV_STATUS uvm_tracker_add_tracker_safe(uvm_tracker_t *dst, uvm_tracker_t *src)
|
|
{
|
|
NV_STATUS status = uvm_tracker_add_tracker(dst, src);
|
|
if (status == NV_ERR_NO_MEMORY) {
|
|
UVM_DBG_PRINT_RL("Failed to add tracker to tracker, waiting\n");
|
|
status = uvm_tracker_wait(src);
|
|
}
|
|
return status;
|
|
}
|
|
|
|
bool uvm_tracker_is_entry_completed(uvm_tracker_entry_t *tracker_entry)
|
|
{
|
|
if (!tracker_entry->channel)
|
|
return true;
|
|
|
|
return uvm_channel_is_value_completed(tracker_entry->channel, tracker_entry->value);
|
|
}
|
|
|
|
static void uvm_tracker_entry_print_pending_pushes(uvm_tracker_entry_t *entry)
|
|
{
|
|
uvm_channel_t *channel = entry->channel;
|
|
uvm_gpu_t *gpu = uvm_channel_get_gpu(channel);
|
|
|
|
UVM_DBG_PRINT("Tracker entry for value %llu (sema VA 0x%llx) channel %s GPU %s\n",
|
|
entry->value,
|
|
uvm_channel_tracking_semaphore_get_gpu_va(channel),
|
|
channel->name,
|
|
uvm_gpu_name(gpu));
|
|
|
|
uvm_channel_print_pending_pushes(channel);
|
|
}
|
|
|
|
static void uvm_tracker_print_pending_pushes(uvm_tracker_t *tracker)
|
|
{
|
|
uvm_tracker_entry_t *entry;
|
|
for_each_tracker_entry(entry, tracker)
|
|
uvm_tracker_entry_print_pending_pushes(entry);
|
|
}
|
|
|
|
static NV_STATUS wait_for_entry_with_spin(uvm_tracker_entry_t *tracker_entry, uvm_spin_loop_t *spin)
|
|
{
|
|
NV_STATUS status = NV_OK;
|
|
|
|
while (!uvm_tracker_is_entry_completed(tracker_entry) && status == NV_OK) {
|
|
if (UVM_SPIN_LOOP(spin) == NV_ERR_TIMEOUT_RETRY)
|
|
uvm_tracker_entry_print_pending_pushes(tracker_entry);
|
|
|
|
status = uvm_channel_check_errors(tracker_entry->channel);
|
|
if (status == NV_OK)
|
|
status = uvm_global_get_status();
|
|
}
|
|
|
|
if (status != NV_OK) {
|
|
UVM_ASSERT(status == uvm_global_get_status());
|
|
tracker_entry->channel = NULL;
|
|
tracker_entry->value = 0;
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
NV_STATUS uvm_tracker_wait_for_entry(uvm_tracker_entry_t *tracker_entry)
|
|
{
|
|
uvm_spin_loop_t spin;
|
|
uvm_spin_loop_init(&spin);
|
|
return wait_for_entry_with_spin(tracker_entry, &spin);
|
|
}
|
|
|
|
NV_STATUS uvm_tracker_wait(uvm_tracker_t *tracker)
|
|
{
|
|
NV_STATUS status = NV_OK;
|
|
uvm_spin_loop_t spin;
|
|
|
|
uvm_spin_loop_init(&spin);
|
|
while (!uvm_tracker_is_completed(tracker) && status == NV_OK) {
|
|
if (UVM_SPIN_LOOP(&spin) == NV_ERR_TIMEOUT_RETRY)
|
|
uvm_tracker_print_pending_pushes(tracker);
|
|
|
|
status = uvm_tracker_check_errors(tracker);
|
|
}
|
|
|
|
if (status != NV_OK) {
|
|
UVM_ASSERT(status == uvm_global_get_status());
|
|
|
|
// Just clear the tracker without printing anything extra. If one of the
|
|
// entries from this tracker caused a channel error,
|
|
// uvm_tracker_check_errors() would have already printed it. And if we
|
|
// hit a global error for some other reason, we don't want to spam the
|
|
// log with all other pending entries.
|
|
//
|
|
// See the comment for uvm_tracker_wait() on why the entries are cleared.
|
|
uvm_tracker_clear(tracker);
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
NV_STATUS uvm_tracker_wait_for_other_gpus(uvm_tracker_t *tracker, uvm_gpu_t *gpu)
|
|
{
|
|
NV_STATUS status = NV_OK;
|
|
uvm_tracker_entry_t *entry;
|
|
uvm_spin_loop_t spin;
|
|
|
|
uvm_spin_loop_init(&spin);
|
|
|
|
for_each_tracker_entry(entry, tracker) {
|
|
if (uvm_tracker_entry_gpu(entry) == gpu)
|
|
continue;
|
|
|
|
status = wait_for_entry_with_spin(entry, &spin);
|
|
if (status != NV_OK)
|
|
break;
|
|
}
|
|
|
|
if (status == NV_OK) {
|
|
uvm_tracker_remove_completed(tracker);
|
|
}
|
|
else {
|
|
UVM_ASSERT(status == uvm_global_get_status());
|
|
uvm_tracker_clear(tracker);
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
NV_STATUS uvm_tracker_check_errors(uvm_tracker_t *tracker)
|
|
{
|
|
uvm_tracker_entry_t *tracker_entry;
|
|
NV_STATUS status = uvm_global_get_status();
|
|
|
|
if (status != NV_OK)
|
|
return status;
|
|
|
|
for_each_tracker_entry(tracker_entry, tracker) {
|
|
status = uvm_channel_check_errors(tracker_entry->channel);
|
|
if (status != NV_OK)
|
|
return status;
|
|
}
|
|
|
|
return NV_OK;
|
|
}
|
|
|
|
NV_STATUS uvm_tracker_query(uvm_tracker_t *tracker)
|
|
{
|
|
NV_STATUS status;
|
|
bool completed = uvm_tracker_is_completed(tracker);
|
|
|
|
status = uvm_tracker_check_errors(tracker);
|
|
if (status != NV_OK)
|
|
return status;
|
|
|
|
return completed ? NV_OK : NV_WARN_MORE_PROCESSING_REQUIRED;
|
|
}
|
|
|
|
void uvm_tracker_remove_completed(uvm_tracker_t *tracker)
|
|
{
|
|
NvU32 i = 0;
|
|
|
|
uvm_tracker_entry_t *entries = uvm_tracker_get_entries(tracker);
|
|
|
|
// Keep removing completed entries until we run out of entries
|
|
while (i < tracker->size) {
|
|
if (uvm_tracker_is_entry_completed(&entries[i])) {
|
|
--tracker->size;
|
|
if (i != tracker->size)
|
|
entries[i] = entries[tracker->size];
|
|
}
|
|
else {
|
|
++i;
|
|
}
|
|
}
|
|
}
|
|
|
|
bool uvm_tracker_is_completed(uvm_tracker_t *tracker)
|
|
{
|
|
uvm_tracker_remove_completed(tracker);
|
|
|
|
return tracker->size == 0;
|
|
}
|
|
|
|
uvm_gpu_t *uvm_tracker_entry_gpu(uvm_tracker_entry_t *entry)
|
|
{
|
|
return uvm_channel_get_gpu(entry->channel);
|
|
}
|
|
|