Andy Ritger 758b4ee818
525.53
2022-11-10 08:39:33 -08:00

254 lines
11 KiB
C

/*******************************************************************************
Copyright (c) 2018 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.
*******************************************************************************/
#ifndef __UVM_MIGRATE_PAGEABLE_H__
#define __UVM_MIGRATE_PAGEABLE_H__
#include "uvm_common.h"
#include "uvm_linux.h"
#include "uvm_populate_pageable.h"
#include "uvm_forward_decl.h"
#include "uvm_processors.h"
typedef struct
{
uvm_va_space_t *va_space;
struct mm_struct *mm;
unsigned long start;
unsigned long length;
uvm_processor_id_t dst_id;
// dst_node_id may be clobbered by uvm_migrate_pageable().
int dst_node_id;
uvm_populate_permissions_t populate_permissions;
bool touch : 1;
bool skip_mapped : 1;
NvU64 *user_space_start;
NvU64 *user_space_length;
} uvm_migrate_args_t;
#if defined(CONFIG_MIGRATE_VMA_HELPER)
#define UVM_MIGRATE_VMA_SUPPORTED 1
#else
#if defined(CONFIG_DEVICE_PRIVATE) && defined(NV_MIGRATE_VMA_SETUP_PRESENT)
#define UVM_MIGRATE_VMA_SUPPORTED 1
#endif
#endif
#ifdef UVM_MIGRATE_VMA_SUPPORTED
#include <linux/migrate.h>
// The calls to migrate_vma are capped at 32MB to set an upper bound on the
// amount of metadata that needs to be allocated for the operation. This number
// was chosen because performance seems to plateau at this size.
#define UVM_MIGRATE_VMA_MAX_SIZE (32UL * 1024 * 1024)
#define UVM_MIGRATE_VMA_MAX_PAGES (UVM_MIGRATE_VMA_MAX_SIZE >> PAGE_SHIFT)
typedef struct
{
// Input parameters
uvm_migrate_args_t *uvm_migrate_args;
// Output parameters
//
// Error code. This only signals errors in internal UVM operations.
// Pages that failed allocation or could not be populated are communicated
// using the fields below.
NV_STATUS status;
// Mask of pages that couldn't be made resident on the destination because
// (a) they are backed with data but pages are not populated (e.g. in swap),
// (b) pages are not backed with any data yet but were not populated
// due to the vma not being READ_WRITE, as it would not charge the pages to
// the process properly.
// (c) pages are already mapped and such pages were requested to not be
// migrated via skip_mapped.
// (d) pages which couldn't be migrated by the kernel.
DECLARE_BITMAP(populate_pages_mask, UVM_MIGRATE_VMA_MAX_PAGES);
// Mask of pages that failed allocation on the destination
DECLARE_BITMAP(allocation_failed_mask, UVM_MIGRATE_VMA_MAX_PAGES);
// Mask of pages which are already resident at the destination.
DECLARE_BITMAP(dst_resident_pages_mask, UVM_MIGRATE_VMA_MAX_PAGES);
// Global state managed by the caller
//
// These are scratch masks that can be used by the migrate_vma caller to
// save output page masks and orchestrate the migrate_vma
// retries/population calls if needed.
DECLARE_BITMAP(scratch1_mask, UVM_MIGRATE_VMA_MAX_PAGES);
DECLARE_BITMAP(scratch2_mask, UVM_MIGRATE_VMA_MAX_PAGES);
// Arrays used by migrate_vma to store the src/dst pfns
unsigned long dst_pfn_array[UVM_MIGRATE_VMA_MAX_PAGES];
unsigned long src_pfn_array[UVM_MIGRATE_VMA_MAX_PAGES];
// Internal state
//
uvm_tracker_t tracker;
struct {
// Array of page IOMMU mappings created during allocate_and_copy.
// Required when using SYS aperture. They are freed in
// finalize_and_map. Also keep an array with the GPUs for which the
// mapping was created.
NvU64 addrs[UVM_MIGRATE_VMA_MAX_PAGES];
uvm_gpu_t *addrs_gpus[UVM_MIGRATE_VMA_MAX_PAGES];
// Mask of pages with entries in the dma address arrays above
DECLARE_BITMAP(page_mask, UVM_MIGRATE_VMA_MAX_PAGES);
// Number of pages for which IOMMU mapping were created
unsigned long num_pages;
} dma;
// Processors where pages are resident before calling migrate_vma
uvm_processor_mask_t src_processors;
// Array of per-processor page masks with the pages that are resident
// before calling migrate_vma.
struct {
DECLARE_BITMAP(page_mask, UVM_MIGRATE_VMA_MAX_PAGES);
} processors[UVM_ID_MAX_PROCESSORS];
// Number of pages in the migrate_vma call
unsigned long num_pages;
// Number of pages that are directly populated on the destination
unsigned long num_populate_anon_pages;
} migrate_vma_state_t;
#if defined(CONFIG_MIGRATE_VMA_HELPER)
struct migrate_vma {
struct vm_area_struct *vma;
unsigned long *dst;
unsigned long *src;
unsigned long start;
unsigned long end;
};
void uvm_migrate_vma_alloc_and_copy_helper(struct vm_area_struct *vma,
const unsigned long *src,
unsigned long *dst,
unsigned long start,
unsigned long end,
void *private);
void uvm_migrate_vma_finalize_and_map_helper(struct vm_area_struct *vma,
const unsigned long *src,
const unsigned long *dst,
unsigned long start,
unsigned long end,
void *private);
#else
void uvm_migrate_vma_alloc_and_copy(struct migrate_vma *args, migrate_vma_state_t *state);
void uvm_migrate_vma_finalize_and_map(struct migrate_vma *args, migrate_vma_state_t *state);
#endif // CONFIG_MIGRATE_VMA_HELPER
// Populates the given VA range and tries to migrate all the pages to dst_id. If
// the destination processor is the CPU, the NUMA node in dst_node_id is used.
// The input VA range must be fully backed by VMAs. This function relies on
// migrate_vma, which was added in Linux 4.14. If skip_mapped is set to true,
// then already mapped pages will not be migrated. For kernels that do not
// provide migrate_vma, this function populates the memory using get_user_pages
// and returns NV_WARN_NOTHING_TO_DO to complete the migration in user space for
// API calls made from userspace. Kernel callers are expected to handle this
// error according to their respective usecases. user_space_start and
// user_space_length will contain the full input range. skip_mapped is ignored
// for such kernels. If the destination is the CPU and dst_node_id is full,
// NV_ERR_MORE_PROCESSING_REQUIRED is returned and user-space will call
// UVM_MIGRATE with the next preferred CPU node (if more are available),
// starting at the address specified by user_space_start. If the destination is
// a GPU and a page could not be populated, return NV_ERR_NO_MEMORY. Otherwise,
// return NV_OK. This is fine because UvmMigrate/UvmMigrateAsync only guarantee
// that the memory is populated somewhere in the system, not that pages moved to
// the requested processor.
//
// migrate_vma does not support file-backed vmas yet. If a file-backed vma is
// found, populates the pages corresponding to the vma with get_user_pages() and
// returns NV_WARN_NOTHING_TO_DO. The caller is expected to handle this error.
// API calls will fall back to user-mode to complete the migration. Kernel
// callers are expected to handle this error according to the respective
// usecases. If NV_WARN_NOTHING_TO_DO is returned, user_space_start and
// user_space_length will contain the intersection of the vma address range and
// [start:start + length].
//
// If a user-mode fallback is required but current->mm != uvm_migrate_args->mm,
// NV_ERR_NOT_SUPPORTED is returned since user mode can't perform such a
// migration. If the caller is a kernel thread, such as the GPU BH, this check
// is ignored since the caller is not expected to take such a fallback.
//
// Also, if no GPUs have been registered in the VA space, populates the pages
// corresponding to the first vma in the requested region using get_user_pages()
// and returns NV_WARN_NOTHING_TO_DO to fall back to user space for the
// userspace API callers to complete the whole migration. Kernel callers are
// expected to handle this error according to their respective usecases.
//
// If touch is true, a touch will be attempted on all pages in the requested
// range. All pages are only guaranteed to have been touched if
// NV_WARN_NOTHING_TO_DO or NV_OK is returned.
//
// Locking: mmap_lock must be held in read or write mode
NV_STATUS uvm_migrate_pageable(uvm_migrate_args_t *uvm_migrate_args);
NV_STATUS uvm_migrate_pageable_init(void);
void uvm_migrate_pageable_exit(void);
#else // UVM_MIGRATE_VMA_SUPPORTED
static NV_STATUS uvm_migrate_pageable(uvm_migrate_args_t *uvm_migrate_args)
{
NV_STATUS status;
if (current->mm != uvm_migrate_args->mm && !(current->flags & PF_KTHREAD))
return NV_ERR_NOT_SUPPORTED;
status = uvm_populate_pageable(uvm_migrate_args->mm,
uvm_migrate_args->start,
uvm_migrate_args->length,
0,
uvm_migrate_args->touch,
uvm_migrate_args->populate_permissions);
if (status != NV_OK)
return status;
*(uvm_migrate_args->user_space_start) = uvm_migrate_args->start;
*(uvm_migrate_args->user_space_length) = uvm_migrate_args->length;
return NV_WARN_NOTHING_TO_DO;
}
static NV_STATUS uvm_migrate_pageable_init(void)
{
return NV_OK;
}
static void uvm_migrate_pageable_exit(void)
{
}
#endif // UVM_MIGRATE_VMA_SUPPORTED
#endif