2022-05-09 22:18:59 +02:00
|
|
|
/*******************************************************************************
|
2023-02-28 20:12:44 +01:00
|
|
|
Copyright (c) 2016-2023 NVIDIA Corporation
|
2022-05-09 22:18:59 +02:00
|
|
|
|
|
|
|
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.
|
|
|
|
|
|
|
|
*******************************************************************************/
|
|
|
|
|
|
|
|
#ifndef _UVM_HMM_H_
|
|
|
|
#define _UVM_HMM_H_
|
|
|
|
|
|
|
|
#include "nvtypes.h"
|
|
|
|
#include "uvm_forward_decl.h"
|
|
|
|
#include "uvm_va_block_types.h"
|
|
|
|
#include "uvm_va_policy.h"
|
|
|
|
#include "uvm_linux.h"
|
|
|
|
#include "uvm_range_tree.h"
|
|
|
|
#include "uvm_lock.h"
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
// This stores pointers to uvm_va_block_t for HMM blocks.
|
|
|
|
uvm_range_tree_t blocks;
|
|
|
|
uvm_mutex_t blocks_lock;
|
|
|
|
} uvm_hmm_va_space_t;
|
|
|
|
|
|
|
|
#if UVM_IS_CONFIG_HMM()
|
2023-02-28 20:12:44 +01:00
|
|
|
|
2022-05-09 22:18:59 +02:00
|
|
|
// Tells whether HMM is enabled for the given va_space.
|
|
|
|
// If it is not enabled, all of the functions below are no-ops.
|
|
|
|
bool uvm_hmm_is_enabled(uvm_va_space_t *va_space);
|
|
|
|
|
|
|
|
// Self-explanatory name: reports if HMM is enabled system-wide.
|
|
|
|
bool uvm_hmm_is_enabled_system_wide(void);
|
|
|
|
|
|
|
|
// Initialize HMM for the given the va_space.
|
2023-08-22 15:09:37 +02:00
|
|
|
void uvm_hmm_va_space_initialize(uvm_va_space_t *va_space);
|
2022-05-09 22:18:59 +02:00
|
|
|
|
|
|
|
// Destroy any HMM state for the given the va_space.
|
|
|
|
// Locking: va_space lock must be held in write mode.
|
|
|
|
void uvm_hmm_va_space_destroy(uvm_va_space_t *va_space);
|
|
|
|
|
2023-02-28 20:12:44 +01:00
|
|
|
// Unmap all page tables in this VA space which map memory owned by this
|
|
|
|
// GPU. Any memory still resident on this GPU will be evicted to system
|
|
|
|
// memory. Note that 'mm' can be NULL (e.g., when closing the UVM file)
|
|
|
|
// in which case any GPU memory is simply freed.
|
|
|
|
// Locking: if mm is not NULL, the caller must hold mm->mmap_lock in at
|
|
|
|
// least read mode and the va_space lock must be held in write mode.
|
|
|
|
void uvm_hmm_unregister_gpu(uvm_va_space_t *va_space, uvm_gpu_t *gpu, struct mm_struct *mm);
|
|
|
|
|
|
|
|
// Destroy the VA space's mappings on the GPU, if it has any.
|
|
|
|
// Locking: if mm is not NULL, the caller must hold mm->mmap_lock in at
|
|
|
|
// least read mode and the va_space lock must be held in write mode.
|
|
|
|
void uvm_hmm_remove_gpu_va_space(uvm_va_space_t *va_space,
|
|
|
|
uvm_gpu_va_space_t *gpu_va_space,
|
|
|
|
struct mm_struct *mm);
|
|
|
|
|
2022-05-09 22:18:59 +02:00
|
|
|
// Find an existing HMM va_block.
|
|
|
|
// This function can be called without having retained and locked the mm,
|
|
|
|
// but in that case, the only allowed operations on the returned block are
|
|
|
|
// locking the block, reading its state, and performing eviction. GPU fault
|
|
|
|
// handling and user-initiated migrations are not allowed.
|
|
|
|
// Return values are the same as uvm_va_block_find().
|
|
|
|
// Locking: This must be called with va_space lock held in at least read
|
|
|
|
// mode.
|
|
|
|
NV_STATUS uvm_hmm_va_block_find(uvm_va_space_t *va_space,
|
|
|
|
NvU64 addr,
|
|
|
|
uvm_va_block_t **va_block_ptr);
|
|
|
|
|
|
|
|
// Find or create a new HMM va_block.
|
|
|
|
//
|
|
|
|
// Return NV_ERR_INVALID_ADDRESS if there is no VMA associated with the
|
|
|
|
// address 'addr' or the VMA does not have at least PROT_READ permission.
|
2022-11-10 17:39:33 +01:00
|
|
|
// The caller is also responsible for checking that there is no UVM
|
|
|
|
// va_range covering the given address before calling this function.
|
2023-08-22 15:09:37 +02:00
|
|
|
// The VMA is returned in vma_out if it's not NULL.
|
2022-05-09 22:18:59 +02:00
|
|
|
// Locking: This function must be called with mm retained and locked for
|
|
|
|
// at least read and the va_space lock at least for read.
|
|
|
|
NV_STATUS uvm_hmm_va_block_find_create(uvm_va_space_t *va_space,
|
|
|
|
NvU64 addr,
|
2023-08-22 15:09:37 +02:00
|
|
|
struct vm_area_struct **vma_out,
|
2022-05-09 22:18:59 +02:00
|
|
|
uvm_va_block_t **va_block_ptr);
|
|
|
|
|
2023-08-22 15:09:37 +02:00
|
|
|
// Find the VMA for the given address and return it in vma_out. Return
|
|
|
|
// NV_ERR_INVALID_ADDRESS if mm is NULL or there is no VMA associated with
|
|
|
|
// the address 'addr' or the VMA does not have at least PROT_READ
|
|
|
|
// permission.
|
2022-11-10 17:39:33 +01:00
|
|
|
// Locking: This function must be called with mm retained and locked for
|
|
|
|
// at least read or mm equal to NULL.
|
2023-08-22 15:09:37 +02:00
|
|
|
NV_STATUS uvm_hmm_find_vma(struct mm_struct *mm, struct vm_area_struct **vma_out, NvU64 addr);
|
2022-11-10 17:39:33 +01:00
|
|
|
|
2023-08-22 15:09:37 +02:00
|
|
|
// If va_block is a HMM va_block, check that vma is not NULL and covers the
|
|
|
|
// given region. This always returns true and is intended to only be used
|
|
|
|
// with UVM_ASSERT().
|
2022-11-10 17:39:33 +01:00
|
|
|
// Locking: This function must be called with the va_block lock held and if
|
2023-08-22 15:09:37 +02:00
|
|
|
// va_block is a HMM block, va_space->va_space_mm.mm->mmap_lock must be
|
|
|
|
// retained and locked for at least read.
|
2023-02-28 20:12:44 +01:00
|
|
|
bool uvm_hmm_check_context_vma_is_valid(uvm_va_block_t *va_block,
|
2023-08-22 15:09:37 +02:00
|
|
|
struct vm_area_struct *vma,
|
2023-02-28 20:12:44 +01:00
|
|
|
uvm_va_block_region_t region);
|
2022-11-10 17:39:33 +01:00
|
|
|
|
2023-05-30 19:11:36 +02:00
|
|
|
// Initialize the HMM portion of the service_context.
|
|
|
|
// This should be called one time before any retry loops calling
|
|
|
|
// uvm_va_block_service_locked().
|
|
|
|
void uvm_hmm_service_context_init(uvm_service_block_context_t *service_context);
|
|
|
|
|
|
|
|
// Begin a migration critical section. When calling into the kernel it is
|
|
|
|
// sometimes necessary to drop the va_block lock. This function returns
|
|
|
|
// NV_OK when no other thread has started a migration critical section.
|
|
|
|
// Otherwise, it returns NV_ERR_BUSY_RETRY and threads should then retry
|
|
|
|
// this function to begin a critical section.
|
|
|
|
// Locking: va_block lock must not be held.
|
|
|
|
NV_STATUS uvm_hmm_migrate_begin(uvm_va_block_t *va_block);
|
|
|
|
|
|
|
|
// Same as uvm_hmm_migrate_begin() but waits if required before beginning a
|
|
|
|
// critical section.
|
|
|
|
void uvm_hmm_migrate_begin_wait(uvm_va_block_t *va_block);
|
|
|
|
|
|
|
|
// Finish a migration critical section.
|
|
|
|
void uvm_hmm_migrate_finish(uvm_va_block_t *va_block);
|
|
|
|
|
2022-11-10 17:39:33 +01:00
|
|
|
// Find or create a HMM va_block and mark it so the next va_block split
|
|
|
|
// will fail for testing purposes.
|
|
|
|
// Locking: This function must be called with mm retained and locked for
|
|
|
|
// at least read and the va_space lock at least for read.
|
|
|
|
NV_STATUS uvm_hmm_test_va_block_inject_split_error(uvm_va_space_t *va_space, NvU64 addr);
|
|
|
|
|
2022-05-09 22:18:59 +02:00
|
|
|
// Reclaim any HMM va_blocks that overlap the given range.
|
2022-11-10 17:39:33 +01:00
|
|
|
// Note that 'end' is inclusive. If mm is NULL, any HMM va_block in the
|
|
|
|
// range will be reclaimed which assumes that the mm is being torn down
|
|
|
|
// and was not retained.
|
2022-05-09 22:18:59 +02:00
|
|
|
// Return values:
|
|
|
|
// NV_ERR_NO_MEMORY: Reclaim required a block split, which failed.
|
|
|
|
// NV_OK: There were no HMM blocks in the range, or all HMM
|
|
|
|
// blocks in the range were successfully reclaimed.
|
|
|
|
// Locking: If mm is not NULL, it must equal va_space_mm.mm, the caller
|
2022-11-10 17:39:33 +01:00
|
|
|
// must retain it with uvm_va_space_mm_or_current_retain() or be sure that
|
|
|
|
// mm->mm_users is not zero, and it must be locked for at least read mode.
|
|
|
|
// Also, the va_space lock must be held in write mode.
|
2022-05-09 22:18:59 +02:00
|
|
|
// TODO: Bug 3372166: add asynchronous va_block reclaim.
|
|
|
|
NV_STATUS uvm_hmm_va_block_reclaim(uvm_va_space_t *va_space,
|
|
|
|
struct mm_struct *mm,
|
|
|
|
NvU64 start,
|
|
|
|
NvU64 end);
|
|
|
|
|
2022-11-10 17:39:33 +01:00
|
|
|
// This is called to update the va_space tree of HMM va_blocks after an
|
|
|
|
// existing va_block is split.
|
|
|
|
// Locking: the va_space lock must be held in write mode.
|
|
|
|
void uvm_hmm_va_block_split_tree(uvm_va_block_t *existing_va_block, uvm_va_block_t *new_block);
|
|
|
|
|
2022-05-09 22:18:59 +02:00
|
|
|
// Find a HMM policy range that needs to be split. The callback function
|
|
|
|
// 'split_needed_cb' returns true if the policy range needs to be split.
|
|
|
|
// If a policy range is split, the existing range is truncated to
|
|
|
|
// [existing_start, addr - 1] and a new policy node with the same policy
|
|
|
|
// values as the existing node is created covering [addr, existing_end].
|
|
|
|
// Before: [----------- existing ------------]
|
|
|
|
// After: [---- existing ----][---- new ----]
|
|
|
|
// ^addr
|
|
|
|
// Locking: the va_space must be write locked.
|
|
|
|
NV_STATUS uvm_hmm_split_as_needed(uvm_va_space_t *va_space,
|
|
|
|
NvU64 addr,
|
|
|
|
uvm_va_policy_is_split_needed_t split_needed_cb,
|
|
|
|
void *data);
|
|
|
|
|
|
|
|
// Set the preferred location policy for the given range.
|
|
|
|
// Note that 'last_address' is inclusive.
|
|
|
|
// Locking: the va_space->va_space_mm.mm mmap_lock must be locked
|
|
|
|
// and the va_space lock must be held in write mode.
|
|
|
|
NV_STATUS uvm_hmm_set_preferred_location(uvm_va_space_t *va_space,
|
|
|
|
uvm_processor_id_t preferred_location,
|
|
|
|
NvU64 base,
|
2023-02-28 20:12:44 +01:00
|
|
|
NvU64 last_address,
|
|
|
|
uvm_tracker_t *out_tracker);
|
2022-05-09 22:18:59 +02:00
|
|
|
|
|
|
|
// Set the accessed by policy for the given range. This also tries to
|
|
|
|
// map the range. Note that 'last_address' is inclusive.
|
|
|
|
// Locking: the va_space->va_space_mm.mm mmap_lock must be locked
|
|
|
|
// and the va_space lock must be held in write mode.
|
|
|
|
NV_STATUS uvm_hmm_set_accessed_by(uvm_va_space_t *va_space,
|
|
|
|
uvm_processor_id_t processor_id,
|
|
|
|
bool set_bit,
|
|
|
|
NvU64 base,
|
2023-02-28 20:12:44 +01:00
|
|
|
NvU64 last_address,
|
|
|
|
uvm_tracker_t *out_tracker);
|
|
|
|
|
|
|
|
// Deferred work item to reestablish accessed by mappings after eviction. On
|
|
|
|
// GPUs with access counters enabled, the evicted GPU will also get remote
|
|
|
|
// mappings.
|
|
|
|
// Locking: the va_space->va_space_mm.mm mmap_lock must be locked
|
|
|
|
// and the va_space lock must be held in at least read mode.
|
|
|
|
void uvm_hmm_block_add_eviction_mappings(uvm_va_space_t *va_space,
|
|
|
|
uvm_va_block_t *va_block,
|
|
|
|
uvm_va_block_context_t *block_context);
|
2022-05-09 22:18:59 +02:00
|
|
|
|
|
|
|
// Set the read duplication policy for the given range.
|
|
|
|
// Note that 'last_address' is inclusive.
|
|
|
|
// Locking: the va_space->va_space_mm.mm mmap_lock must be write locked
|
|
|
|
// and the va_space lock must be held in write mode.
|
2022-11-10 17:39:33 +01:00
|
|
|
// TODO: Bug 3660922: need to implement HMM read duplication support.
|
2022-05-09 22:18:59 +02:00
|
|
|
static NV_STATUS uvm_hmm_set_read_duplication(uvm_va_space_t *va_space,
|
|
|
|
uvm_read_duplication_policy_t new_policy,
|
|
|
|
NvU64 base,
|
|
|
|
NvU64 last_address)
|
|
|
|
{
|
|
|
|
if (!uvm_hmm_is_enabled(va_space))
|
|
|
|
return NV_ERR_INVALID_ADDRESS;
|
|
|
|
return NV_OK;
|
|
|
|
}
|
|
|
|
|
2023-08-22 15:09:37 +02:00
|
|
|
// This function returns the policy covering the given address 'addr' and
|
|
|
|
// assigns the ending address '*endp' to the minimum of va_block->end,
|
|
|
|
// vma->vm_end - 1, and the ending address of the policy range. Locking:
|
|
|
|
// This function must be called with vma->vm_mm retained and locked for at
|
|
|
|
// least read and the va_block and va_space lock held.
|
|
|
|
const uvm_va_policy_t *uvm_hmm_find_policy_end(uvm_va_block_t *va_block,
|
|
|
|
struct vm_area_struct *vma,
|
|
|
|
unsigned long addr,
|
|
|
|
NvU64 *endp);
|
|
|
|
|
|
|
|
// This function finds the VMA for the page index 'page_index' and returns
|
|
|
|
// it in vma_out which must not be NULL. Returns the policy covering the
|
|
|
|
// given address, and sets the ending page range '*outerp' to the minimum of
|
|
|
|
// *outerp, vma->vm_end - 1, the ending address of the policy range, and
|
|
|
|
// va_block->end.
|
|
|
|
// Return NV_ERR_INVALID_ADDRESS if no VMA is found; otherwise sets *vma
|
|
|
|
// and returns NV_OK.
|
|
|
|
// Locking: This function must be called with mm retained and locked for at
|
|
|
|
// least read and the va_block and va_space lock held.
|
2022-05-09 22:18:59 +02:00
|
|
|
NV_STATUS uvm_hmm_find_policy_vma_and_outer(uvm_va_block_t *va_block,
|
2023-08-22 15:09:37 +02:00
|
|
|
struct vm_area_struct **vma,
|
2022-05-09 22:18:59 +02:00
|
|
|
uvm_page_index_t page_index,
|
2023-08-22 15:09:37 +02:00
|
|
|
const uvm_va_policy_t **policy,
|
2022-05-09 22:18:59 +02:00
|
|
|
uvm_page_index_t *outerp);
|
|
|
|
|
2022-10-10 23:59:24 +02:00
|
|
|
// Clear thrashing policy information from all HMM va_blocks.
|
|
|
|
// Locking: va_space lock must be held in write mode.
|
|
|
|
NV_STATUS uvm_hmm_clear_thrashing_policy(uvm_va_space_t *va_space);
|
|
|
|
|
2022-11-10 17:39:33 +01:00
|
|
|
// Return the expanded region around 'address' limited to the intersection
|
|
|
|
// of va_block start/end, vma start/end, and policy start/end.
|
2023-08-22 15:09:37 +02:00
|
|
|
// Locking: the caller must hold va_space->va_space_mm.mm->mmap_lock in at
|
|
|
|
// least read mode, the va_space lock must be held in at least read mode,
|
|
|
|
// and the va_block lock held.
|
2022-11-10 17:39:33 +01:00
|
|
|
uvm_va_block_region_t uvm_hmm_get_prefetch_region(uvm_va_block_t *va_block,
|
2023-08-22 15:09:37 +02:00
|
|
|
struct vm_area_struct *vma,
|
|
|
|
const uvm_va_policy_t *policy,
|
2022-11-10 17:39:33 +01:00
|
|
|
NvU64 address);
|
|
|
|
|
|
|
|
// Return the logical protection allowed of a HMM va_block for the page at
|
2023-08-22 15:09:37 +02:00
|
|
|
// the given address within the vma which must be valid. This is usually
|
|
|
|
// obtained from uvm_hmm_va_block_find_create()).
|
|
|
|
// Locking: the caller must hold va_space->va_space_mm.mm mmap_lock in at
|
|
|
|
// least read mode.
|
2022-11-10 17:39:33 +01:00
|
|
|
uvm_prot_t uvm_hmm_compute_logical_prot(uvm_va_block_t *va_block,
|
2023-08-22 15:09:37 +02:00
|
|
|
struct vm_area_struct *vma,
|
2022-11-10 17:39:33 +01:00
|
|
|
NvU64 addr);
|
|
|
|
|
2023-02-28 20:12:44 +01:00
|
|
|
// This is called to service a GPU fault.
|
|
|
|
// Locking: the va_space->va_space_mm.mm mmap_lock must be locked,
|
|
|
|
// the va_space read lock must be held, and the va_block lock held.
|
|
|
|
NV_STATUS uvm_hmm_va_block_service_locked(uvm_processor_id_t processor_id,
|
|
|
|
uvm_processor_id_t new_residency,
|
|
|
|
uvm_va_block_t *va_block,
|
|
|
|
uvm_va_block_retry_t *va_block_retry,
|
|
|
|
uvm_service_block_context_t *service_context);
|
|
|
|
|
|
|
|
// This is called to migrate a region within a HMM va_block.
|
2023-08-22 15:09:37 +02:00
|
|
|
// va_block_context must not be NULL and va_block_context->hmm.vma
|
|
|
|
// must be valid.
|
|
|
|
// Locking: the va_space->va_space_mm.mm must be retained, mmap_lock must be
|
2023-02-28 20:12:44 +01:00
|
|
|
// locked, and the va_block lock held.
|
|
|
|
NV_STATUS uvm_hmm_va_block_migrate_locked(uvm_va_block_t *va_block,
|
|
|
|
uvm_va_block_retry_t *va_block_retry,
|
|
|
|
uvm_va_block_context_t *va_block_context,
|
|
|
|
uvm_processor_id_t dest_id,
|
|
|
|
uvm_va_block_region_t region,
|
|
|
|
uvm_make_resident_cause_t cause);
|
|
|
|
|
|
|
|
// This is called to migrate an address range of HMM allocations via
|
|
|
|
// UvmMigrate().
|
|
|
|
//
|
|
|
|
// va_block_context must not be NULL. The caller is not required to set
|
2023-08-22 15:09:37 +02:00
|
|
|
// va_block_context->hmm.vma.
|
2023-02-28 20:12:44 +01:00
|
|
|
//
|
|
|
|
// Locking: the va_space->va_space_mm.mm mmap_lock must be locked and
|
|
|
|
// the va_space read lock must be held.
|
|
|
|
NV_STATUS uvm_hmm_migrate_ranges(uvm_va_space_t *va_space,
|
|
|
|
uvm_va_block_context_t *va_block_context,
|
|
|
|
NvU64 base,
|
|
|
|
NvU64 length,
|
|
|
|
uvm_processor_id_t dest_id,
|
|
|
|
uvm_migrate_mode_t mode,
|
|
|
|
uvm_tracker_t *out_tracker);
|
|
|
|
|
2023-05-30 19:11:36 +02:00
|
|
|
// Evicts all va_blocks in the va_space to the CPU. Unlike the
|
|
|
|
// other va_block eviction functions this is based on virtual
|
|
|
|
// address and therefore takes mmap_lock for read.
|
|
|
|
void uvm_hmm_evict_va_blocks(uvm_va_space_t *va_space);
|
|
|
|
|
2023-02-28 20:12:44 +01:00
|
|
|
// This sets the va_block_context->hmm.src_pfns[] to the ZONE_DEVICE private
|
|
|
|
// PFN for the GPU chunk memory.
|
|
|
|
NV_STATUS uvm_hmm_va_block_evict_chunk_prep(uvm_va_block_t *va_block,
|
|
|
|
uvm_va_block_context_t *va_block_context,
|
|
|
|
uvm_gpu_chunk_t *gpu_chunk,
|
|
|
|
uvm_va_block_region_t chunk_region);
|
|
|
|
|
|
|
|
// Migrate pages to system memory for the given page mask.
|
|
|
|
// Note that the mmap lock is not held and there is no MM retained.
|
|
|
|
// This must be called after uvm_hmm_va_block_evict_chunk_prep() has
|
|
|
|
// initialized va_block_context->hmm.src_pfns[] for the source GPU physical
|
|
|
|
// PFNs being migrated. Note that the input mask 'pages_to_evict' can be
|
|
|
|
// modified. If any of the evicted pages has the accessed by policy set,
|
|
|
|
// then record that by setting out_accessed_by_set.
|
|
|
|
// Locking: the va_block lock must be locked.
|
|
|
|
NV_STATUS uvm_hmm_va_block_evict_chunks(uvm_va_block_t *va_block,
|
|
|
|
uvm_va_block_context_t *va_block_context,
|
|
|
|
const uvm_page_mask_t *pages_to_evict,
|
|
|
|
uvm_va_block_region_t region,
|
|
|
|
bool *out_accessed_by_set);
|
|
|
|
|
|
|
|
// Migrate pages from the given GPU to system memory for the given page
|
|
|
|
// mask and region. va_block_context must not be NULL.
|
|
|
|
// Note that the mmap lock is not held and there is no MM retained.
|
|
|
|
// Locking: the va_block lock must be locked.
|
|
|
|
NV_STATUS uvm_hmm_va_block_evict_pages_from_gpu(uvm_va_block_t *va_block,
|
|
|
|
uvm_gpu_t *gpu,
|
|
|
|
uvm_va_block_context_t *va_block_context,
|
|
|
|
const uvm_page_mask_t *pages_to_evict,
|
|
|
|
uvm_va_block_region_t region);
|
|
|
|
|
2023-05-30 19:11:36 +02:00
|
|
|
// Migrate a GPU device-private page to system memory. This is
|
|
|
|
// called to remove CPU page table references to device private
|
|
|
|
// struct pages for the given GPU after all other references in
|
|
|
|
// va_blocks have been released and the GPU is in the process of
|
|
|
|
// being removed/torn down. Note that there is no mm, VMA,
|
|
|
|
// va_block or any user channel activity on this GPU.
|
|
|
|
NV_STATUS uvm_hmm_pmm_gpu_evict_pfn(unsigned long pfn);
|
2023-02-28 20:12:44 +01:00
|
|
|
|
|
|
|
// This returns what would be the intersection of va_block start/end and
|
|
|
|
// VMA start/end-1 for the given 'lookup_address' if
|
|
|
|
// uvm_hmm_va_block_find_create() was called.
|
|
|
|
// Locking: the caller must hold mm->mmap_lock in at least read mode and
|
|
|
|
// the va_space lock must be held in at least read mode.
|
|
|
|
NV_STATUS uvm_hmm_va_block_range_bounds(uvm_va_space_t *va_space,
|
|
|
|
struct mm_struct *mm,
|
|
|
|
NvU64 lookup_address,
|
|
|
|
NvU64 *startp,
|
|
|
|
NvU64 *endp,
|
|
|
|
UVM_TEST_VA_RESIDENCY_INFO_PARAMS *params);
|
|
|
|
|
|
|
|
// This updates the HMM va_block CPU residency information for a single
|
|
|
|
// page at 'lookup_address' by calling hmm_range_fault(). If 'populate' is
|
|
|
|
// true, the CPU page will be faulted in read/write or read-only
|
|
|
|
// (depending on the permission of the underlying VMA at lookup_address).
|
|
|
|
// Locking: the caller must hold mm->mmap_lock in at least read mode and
|
|
|
|
// the va_space lock must be held in at least read mode.
|
|
|
|
NV_STATUS uvm_hmm_va_block_update_residency_info(uvm_va_block_t *va_block,
|
|
|
|
struct mm_struct *mm,
|
|
|
|
NvU64 lookup_address,
|
|
|
|
bool populate);
|
2022-11-10 17:39:33 +01:00
|
|
|
|
|
|
|
NV_STATUS uvm_test_split_invalidate_delay(UVM_TEST_SPLIT_INVALIDATE_DELAY_PARAMS *params,
|
|
|
|
struct file *filp);
|
|
|
|
|
|
|
|
NV_STATUS uvm_hmm_va_range_info(uvm_va_space_t *va_space,
|
|
|
|
struct mm_struct *mm,
|
|
|
|
UVM_TEST_VA_RANGE_INFO_PARAMS *params);
|
|
|
|
|
|
|
|
// Return true if GPU fault new residency location should be system memory.
|
|
|
|
// va_block_context must not be NULL and va_block_context->hmm.vma must be
|
|
|
|
// valid (this is usually set by uvm_hmm_va_block_find_create()).
|
|
|
|
// TODO: Bug 3660968: Remove this hack as soon as HMM migration is
|
|
|
|
// implemented for VMAs other than anonymous memory.
|
|
|
|
// Locking: the va_block lock must be held. If the va_block is a HMM
|
|
|
|
// va_block, the va_block_context->mm must be retained and locked for least
|
|
|
|
// read.
|
|
|
|
bool uvm_hmm_must_use_sysmem(uvm_va_block_t *va_block,
|
|
|
|
uvm_va_block_context_t *va_block_context);
|
|
|
|
|
2022-05-09 22:18:59 +02:00
|
|
|
#else // UVM_IS_CONFIG_HMM()
|
|
|
|
|
|
|
|
static bool uvm_hmm_is_enabled(uvm_va_space_t *va_space)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool uvm_hmm_is_enabled_system_wide(void)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-08-22 15:09:37 +02:00
|
|
|
static void uvm_hmm_va_space_initialize(uvm_va_space_t *va_space)
|
2022-05-09 22:18:59 +02:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2023-02-28 20:12:44 +01:00
|
|
|
static void uvm_hmm_va_space_destroy(uvm_va_space_t *va_space)
|
2022-05-09 22:18:59 +02:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2023-02-28 20:12:44 +01:00
|
|
|
static void uvm_hmm_unregister_gpu(uvm_va_space_t *va_space, uvm_gpu_t *gpu, struct mm_struct *mm)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static void uvm_hmm_remove_gpu_va_space(uvm_va_space_t *va_space,
|
|
|
|
uvm_gpu_va_space_t *gpu_va_space,
|
|
|
|
struct mm_struct *mm)
|
2022-05-09 22:18:59 +02:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static NV_STATUS uvm_hmm_va_block_find(uvm_va_space_t *va_space,
|
|
|
|
NvU64 addr,
|
|
|
|
uvm_va_block_t **va_block_ptr)
|
|
|
|
{
|
|
|
|
return NV_ERR_INVALID_ADDRESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
static NV_STATUS uvm_hmm_va_block_find_create(uvm_va_space_t *va_space,
|
|
|
|
NvU64 addr,
|
2023-08-22 15:09:37 +02:00
|
|
|
struct vm_area_struct **vma,
|
2022-05-09 22:18:59 +02:00
|
|
|
uvm_va_block_t **va_block_ptr)
|
|
|
|
{
|
|
|
|
return NV_ERR_INVALID_ADDRESS;
|
|
|
|
}
|
|
|
|
|
2023-08-22 15:09:37 +02:00
|
|
|
static NV_STATUS uvm_hmm_find_vma(struct mm_struct *mm, struct vm_area_struct **vma, NvU64 addr)
|
2022-11-10 17:39:33 +01:00
|
|
|
{
|
|
|
|
return NV_OK;
|
|
|
|
}
|
|
|
|
|
2023-02-28 20:12:44 +01:00
|
|
|
static bool uvm_hmm_check_context_vma_is_valid(uvm_va_block_t *va_block,
|
2023-08-22 15:09:37 +02:00
|
|
|
struct vm_area_struct *vma,
|
2023-02-28 20:12:44 +01:00
|
|
|
uvm_va_block_region_t region)
|
2022-11-10 17:39:33 +01:00
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-05-30 19:11:36 +02:00
|
|
|
static void uvm_hmm_service_context_init(uvm_service_block_context_t *service_context)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static NV_STATUS uvm_hmm_migrate_begin(uvm_va_block_t *va_block)
|
|
|
|
{
|
|
|
|
return NV_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void uvm_hmm_migrate_begin_wait(uvm_va_block_t *va_block)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static void uvm_hmm_migrate_finish(uvm_va_block_t *va_block)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2022-11-10 17:39:33 +01:00
|
|
|
static NV_STATUS uvm_hmm_test_va_block_inject_split_error(uvm_va_space_t *va_space, NvU64 addr)
|
|
|
|
{
|
|
|
|
return NV_ERR_INVALID_ADDRESS;
|
|
|
|
}
|
|
|
|
|
2022-05-09 22:18:59 +02:00
|
|
|
static NV_STATUS uvm_hmm_va_block_reclaim(uvm_va_space_t *va_space,
|
|
|
|
struct mm_struct *mm,
|
|
|
|
NvU64 start,
|
|
|
|
NvU64 end)
|
|
|
|
{
|
|
|
|
return NV_OK;
|
|
|
|
}
|
|
|
|
|
2022-11-10 17:39:33 +01:00
|
|
|
static void uvm_hmm_va_block_split_tree(uvm_va_block_t *existing_va_block, uvm_va_block_t *new_block)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2022-05-09 22:18:59 +02:00
|
|
|
static NV_STATUS uvm_hmm_split_as_needed(uvm_va_space_t *va_space,
|
|
|
|
NvU64 addr,
|
|
|
|
uvm_va_policy_is_split_needed_t split_needed_cb,
|
|
|
|
void *data)
|
|
|
|
{
|
|
|
|
return NV_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static NV_STATUS uvm_hmm_set_preferred_location(uvm_va_space_t *va_space,
|
|
|
|
uvm_processor_id_t preferred_location,
|
|
|
|
NvU64 base,
|
2023-02-28 20:12:44 +01:00
|
|
|
NvU64 last_address,
|
|
|
|
uvm_tracker_t *out_tracker)
|
2022-05-09 22:18:59 +02:00
|
|
|
{
|
|
|
|
return NV_ERR_INVALID_ADDRESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
static NV_STATUS uvm_hmm_set_accessed_by(uvm_va_space_t *va_space,
|
|
|
|
uvm_processor_id_t processor_id,
|
|
|
|
bool set_bit,
|
|
|
|
NvU64 base,
|
2023-02-28 20:12:44 +01:00
|
|
|
NvU64 last_address,
|
|
|
|
uvm_tracker_t *out_tracker)
|
2022-05-09 22:18:59 +02:00
|
|
|
{
|
|
|
|
return NV_ERR_INVALID_ADDRESS;
|
|
|
|
}
|
|
|
|
|
2023-02-28 20:12:44 +01:00
|
|
|
static void uvm_hmm_block_add_eviction_mappings(uvm_va_space_t *va_space,
|
|
|
|
uvm_va_block_t *va_block,
|
|
|
|
uvm_va_block_context_t *block_context)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2022-05-09 22:18:59 +02:00
|
|
|
static NV_STATUS uvm_hmm_set_read_duplication(uvm_va_space_t *va_space,
|
|
|
|
uvm_read_duplication_policy_t new_policy,
|
|
|
|
NvU64 base,
|
|
|
|
NvU64 last_address)
|
|
|
|
{
|
|
|
|
return NV_ERR_INVALID_ADDRESS;
|
|
|
|
}
|
|
|
|
|
2023-08-22 15:09:37 +02:00
|
|
|
static const uvm_va_policy_t *uvm_hmm_find_policy_end(uvm_va_block_t *va_block,
|
|
|
|
struct vm_area_struct *vma,
|
|
|
|
unsigned long addr,
|
|
|
|
NvU64 *endp)
|
2022-05-09 22:18:59 +02:00
|
|
|
{
|
2023-08-22 15:09:37 +02:00
|
|
|
UVM_ASSERT(0);
|
|
|
|
return NULL;
|
2022-05-09 22:18:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static NV_STATUS uvm_hmm_find_policy_vma_and_outer(uvm_va_block_t *va_block,
|
2023-08-22 15:09:37 +02:00
|
|
|
struct vm_area_struct **vma,
|
2022-05-09 22:18:59 +02:00
|
|
|
uvm_page_index_t page_index,
|
2023-08-22 15:09:37 +02:00
|
|
|
const uvm_va_policy_t **policy,
|
2022-05-09 22:18:59 +02:00
|
|
|
uvm_page_index_t *outerp)
|
|
|
|
{
|
|
|
|
return NV_OK;
|
|
|
|
}
|
|
|
|
|
2022-10-10 23:59:24 +02:00
|
|
|
static NV_STATUS uvm_hmm_clear_thrashing_policy(uvm_va_space_t *va_space)
|
|
|
|
{
|
|
|
|
return NV_OK;
|
|
|
|
}
|
|
|
|
|
2022-11-10 17:39:33 +01:00
|
|
|
static uvm_va_block_region_t uvm_hmm_get_prefetch_region(uvm_va_block_t *va_block,
|
2023-08-22 15:09:37 +02:00
|
|
|
struct vm_area_struct *vma,
|
|
|
|
const uvm_va_policy_t *policy,
|
2022-11-10 17:39:33 +01:00
|
|
|
NvU64 address)
|
|
|
|
{
|
|
|
|
return (uvm_va_block_region_t){};
|
|
|
|
}
|
|
|
|
|
|
|
|
static uvm_prot_t uvm_hmm_compute_logical_prot(uvm_va_block_t *va_block,
|
2023-08-22 15:09:37 +02:00
|
|
|
struct vm_area_struct *vma,
|
2022-11-10 17:39:33 +01:00
|
|
|
NvU64 addr)
|
|
|
|
{
|
|
|
|
return UVM_PROT_NONE;
|
|
|
|
}
|
|
|
|
|
2023-02-28 20:12:44 +01:00
|
|
|
static NV_STATUS uvm_hmm_va_block_service_locked(uvm_processor_id_t processor_id,
|
|
|
|
uvm_processor_id_t new_residency,
|
|
|
|
uvm_va_block_t *va_block,
|
|
|
|
uvm_va_block_retry_t *va_block_retry,
|
|
|
|
uvm_service_block_context_t *service_context)
|
2022-11-10 17:39:33 +01:00
|
|
|
{
|
2023-02-28 20:12:44 +01:00
|
|
|
return NV_ERR_INVALID_ADDRESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
static NV_STATUS uvm_hmm_va_block_migrate_locked(uvm_va_block_t *va_block,
|
|
|
|
uvm_va_block_retry_t *va_block_retry,
|
|
|
|
uvm_va_block_context_t *va_block_context,
|
|
|
|
uvm_processor_id_t dest_id,
|
|
|
|
uvm_va_block_region_t region,
|
|
|
|
uvm_make_resident_cause_t cause)
|
|
|
|
{
|
|
|
|
return NV_ERR_INVALID_ADDRESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
static NV_STATUS uvm_hmm_migrate_ranges(uvm_va_space_t *va_space,
|
|
|
|
uvm_va_block_context_t *va_block_context,
|
|
|
|
NvU64 base,
|
|
|
|
NvU64 length,
|
|
|
|
uvm_processor_id_t dest_id,
|
|
|
|
uvm_migrate_mode_t mode,
|
|
|
|
uvm_tracker_t *out_tracker)
|
|
|
|
{
|
|
|
|
return NV_ERR_INVALID_ADDRESS;
|
|
|
|
}
|
|
|
|
|
2023-05-30 19:11:36 +02:00
|
|
|
static void uvm_hmm_evict_va_blocks(uvm_va_space_t *va_space)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2023-02-28 20:12:44 +01:00
|
|
|
static NV_STATUS uvm_hmm_va_block_evict_chunk_prep(uvm_va_block_t *va_block,
|
|
|
|
uvm_va_block_context_t *va_block_context,
|
|
|
|
uvm_gpu_chunk_t *gpu_chunk,
|
|
|
|
uvm_va_block_region_t chunk_region)
|
|
|
|
{
|
|
|
|
return NV_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static NV_STATUS uvm_hmm_va_block_evict_chunks(uvm_va_block_t *va_block,
|
|
|
|
uvm_va_block_context_t *va_block_context,
|
|
|
|
const uvm_page_mask_t *pages_to_evict,
|
|
|
|
uvm_va_block_region_t region,
|
|
|
|
bool *out_accessed_by_set)
|
|
|
|
{
|
|
|
|
return NV_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static NV_STATUS uvm_hmm_va_block_evict_pages_from_gpu(uvm_va_block_t *va_block,
|
|
|
|
uvm_gpu_t *gpu,
|
|
|
|
uvm_va_block_context_t *va_block_context,
|
|
|
|
const uvm_page_mask_t *pages_to_evict,
|
|
|
|
uvm_va_block_region_t region)
|
|
|
|
{
|
|
|
|
return NV_OK;
|
|
|
|
}
|
|
|
|
|
2023-05-30 19:11:36 +02:00
|
|
|
static NV_STATUS uvm_hmm_pmm_gpu_evict_pfn(unsigned long pfn)
|
2023-02-28 20:12:44 +01:00
|
|
|
{
|
|
|
|
return NV_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static NV_STATUS uvm_hmm_va_block_range_bounds(uvm_va_space_t *va_space,
|
|
|
|
struct mm_struct *mm,
|
|
|
|
NvU64 lookup_address,
|
|
|
|
NvU64 *startp,
|
|
|
|
NvU64 *endp,
|
|
|
|
UVM_TEST_VA_RESIDENCY_INFO_PARAMS *params)
|
|
|
|
{
|
|
|
|
return NV_ERR_INVALID_ADDRESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
static NV_STATUS uvm_hmm_va_block_update_residency_info(uvm_va_block_t *va_block,
|
|
|
|
struct mm_struct *mm,
|
|
|
|
NvU64 lookup_address,
|
|
|
|
bool populate)
|
|
|
|
{
|
|
|
|
return NV_ERR_INVALID_ADDRESS;
|
2022-11-10 17:39:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static NV_STATUS uvm_test_split_invalidate_delay(UVM_TEST_SPLIT_INVALIDATE_DELAY_PARAMS *params,
|
|
|
|
struct file *filp)
|
|
|
|
{
|
|
|
|
return NV_ERR_INVALID_STATE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static NV_STATUS uvm_hmm_va_range_info(uvm_va_space_t *va_space,
|
|
|
|
struct mm_struct *mm,
|
|
|
|
UVM_TEST_VA_RANGE_INFO_PARAMS *params)
|
|
|
|
{
|
|
|
|
return NV_ERR_INVALID_ADDRESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool uvm_hmm_must_use_sysmem(uvm_va_block_t *va_block,
|
|
|
|
uvm_va_block_context_t *va_block_context)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-05-09 22:18:59 +02:00
|
|
|
#endif // UVM_IS_CONFIG_HMM()
|
|
|
|
|
|
|
|
#endif // _UVM_HMM_H_
|