/* * SPDX-FileCopyrightText: Copyright (c) 2000-2018 NVIDIA CORPORATION & AFFILIATES. All rights reserved. * SPDX-License-Identifier: MIT * * 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. */ #define __NO_VERSION__ #define NV_DEFINE_REGISTRY_KEY_TABLE #include "os-interface.h" #include "nv-linux.h" #include "nv-reg.h" #include "nv-gpu-info.h" /*! * @brief This function parses the PCI BDF identifier string and returns the * Domain, Bus, Device and function components from the PCI BDF string. * * This parser is highly adaptable and hence allows PCI BDF string in following * 3 formats. * * 1) bus:slot : Domain and function defaults to 0. * 2) domain:bus:slot : Function defaults to 0. * 3) domain:bus:slot.func : Complete PCI dev id string. * * @param[in] pci_dev_str String containing the BDF to be parsed. * @param[out] pci_domain Pointer where pci_domain is to be returned. * @param[out] pci_bus Pointer where pci_bus is to be returned. * @param[out] pci_slot Pointer where pci_slot is to be returned. * @param[out] pci_func Pointer where pci_func is to be returned. * * @return NV_TRUE if succeeds, or NV_FALSE otherwise. */ static NV_STATUS pci_str_to_bdf(char *pci_dev_str, NvU32 *pci_domain, NvU32 *pci_bus, NvU32 *pci_slot, NvU32 *pci_func) { char *option_string = NULL; char *token, *string; NvU32 domain, bus, slot; NV_STATUS status = NV_OK; // // remove_spaces() allocates memory, hence we need to keep a pointer // to the original string for freeing at end of function. // if ((option_string = rm_remove_spaces(pci_dev_str)) == NULL) { // memory allocation failed, returning return NV_ERR_GENERIC; } string = option_string; if (!strlen(string) || !pci_domain || !pci_bus || !pci_slot || !pci_func) { status = NV_ERR_INVALID_ARGUMENT; goto done; } if ((token = strsep(&string, ".")) != NULL) { // PCI device can have maximum 8 functions only. if ((string != NULL) && (!(*string >= '0' && *string <= '7') || (strlen(string) > 1))) { nv_printf(NV_DBG_ERRORS, "NVRM: Invalid PCI function in token %s\n", pci_dev_str); status = NV_ERR_INVALID_ARGUMENT; goto done; } else if (string == NULL) { *pci_func = 0; } else { *pci_func = (NvU32)(*string - '0'); } domain = simple_strtoul(token, &string, 16); if ((string == NULL) || (*string != ':') || (*(string + 1) == '\0')) { nv_printf(NV_DBG_ERRORS, "NVRM: Invalid PCI domain/bus in token %s\n", pci_dev_str); status = NV_ERR_INVALID_ARGUMENT; goto done; } token = string; bus = simple_strtoul((token + 1), &string, 16); if (string == NULL) { nv_printf(NV_DBG_ERRORS, "NVRM: Invalid PCI bus/slot in token %s\n", pci_dev_str); status = NV_ERR_INVALID_ARGUMENT; goto done; } if (*string != '\0') { if ((*string != ':') || (*(string + 1) == '\0')) { nv_printf(NV_DBG_ERRORS, "NVRM: Invalid PCI slot in token %s\n", pci_dev_str); status = NV_ERR_INVALID_ARGUMENT; goto done; } token = string; slot = (NvU32)simple_strtoul(token + 1, &string, 16); if ((slot == 0) && ((token + 1) == string)) { nv_printf(NV_DBG_ERRORS, "NVRM: Invalid PCI slot in token %s\n", pci_dev_str); status = NV_ERR_INVALID_ARGUMENT; goto done; } *pci_domain = domain; *pci_bus = bus; *pci_slot = slot; } else { *pci_slot = bus; *pci_bus = domain; *pci_domain = 0; } status = NV_OK; } else { status = NV_ERR_INVALID_ARGUMENT; } done: // Freeing the memory allocated by remove_spaces(). os_free_mem(option_string); return status; } /*! * @brief This function parses the registry keys per GPU device. It accepts a * semicolon separated list of key=value pairs. The first key value pair MUST be * "pci=DDDD:BB:DD.F;" where DDDD is Domain, BB is Bus Id, DD is device slot * number and F is the Function. This PCI BDF is used to identify which GPU to * assign the registry keys that follows next. * If a GPU corresponding to the value specified in "pci=DDDD:BB:DD.F;" is NOT * found, then all the registry keys that follows are skipped, until we find next * valid pci identified "pci=DDDD:BB:DD.F;". Following are the valid formats for * the value of the "pci" string: * 1) bus:slot : Domain and function defaults to 0. * 2) domain:bus:slot : Function defaults to 0. * 3) domain:bus:slot.func : Complete PCI dev id string. * * * @param[in] sp pointer to nvidia_stack_t struct. * * @return NV_OK if succeeds, or NV_STATUS error code otherwise. */ NV_STATUS nv_parse_per_device_option_string(nvidia_stack_t *sp) { NV_STATUS status = NV_OK; char *option_string = NULL; char *ptr, *token; char *name, *value; NvU32 data, domain, bus, slot, func; nv_linux_state_t *nvl = NULL; nv_state_t *nv = NULL; if (NVreg_RegistryDwordsPerDevice != NULL) { if ((option_string = rm_remove_spaces(NVreg_RegistryDwordsPerDevice)) == NULL) { return NV_ERR_GENERIC; } ptr = option_string; while ((token = strsep(&ptr, ";")) != NULL) { if (!(name = strsep(&token, "=")) || !strlen(name)) { continue; } if (!(value = strsep(&token, "=")) || !strlen(value)) { continue; } if (strsep(&token, "=") != NULL) { continue; } // If this key is "pci", then value is pci_dev id string // which needs special parsing as it is NOT a dword. if (strcmp(name, NV_REG_PCI_DEVICE_BDF) == 0) { status = pci_str_to_bdf(value, &domain, &bus, &slot, &func); // Check if PCI_DEV id string was in a valid format or NOT. if (NV_OK != status) { // lets reset cached pci dev nv = NULL; } else { nvl = find_pci(domain, bus, slot, func); // // If NO GPU found corresponding to this GPU, then reset // cached state. This helps ignore the following registry // keys until valid PCI BDF is found in the commandline. // if (!nvl) { nv = NULL; } else { nv = NV_STATE_PTR(nvl); } } continue; } // // Check if cached pci_dev string in the commandline is in valid // format, else we will skip all the successive registry entries // ( pairs) until a valid PCI_DEV string is encountered // in the commandline. // if (!nv) continue; data = (NvU32)simple_strtoul(value, NULL, 0); rm_write_registry_dword(sp, nv, name, data); } os_free_mem(option_string); } return status; } /* * Compare given string UUID with the GpuBlacklist or ExcludedGpus registry * parameter string and return whether the UUID is in the GPU exclusion list */ NvBool nv_is_uuid_in_gpu_exclusion_list(const char *uuid) { const char *input; char *list; char *ptr; char *token; // // When both NVreg_GpuBlacklist and NVreg_ExcludedGpus are defined // NVreg_ExcludedGpus takes precedence. // if (NVreg_ExcludedGpus != NULL) input = NVreg_ExcludedGpus; else if (NVreg_GpuBlacklist != NULL) input = NVreg_GpuBlacklist; else return NV_FALSE; if ((list = rm_remove_spaces(input)) == NULL) return NV_FALSE; ptr = list; while ((token = strsep(&ptr, ",")) != NULL) { if (strcmp(token, uuid) == 0) { os_free_mem(list); return NV_TRUE; } } os_free_mem(list); return NV_FALSE; } NV_STATUS NV_API_CALL os_registry_init(void) { nv_parm_t *entry; unsigned int i; nvidia_stack_t *sp = NULL; if (nv_kmem_cache_alloc_stack(&sp) != 0) { return NV_ERR_NO_MEMORY; } if (NVreg_RmNvlinkBandwidth != NULL) { rm_write_registry_string(sp, NULL, "RmNvlinkBandwidth", NVreg_RmNvlinkBandwidth, strlen(NVreg_RmNvlinkBandwidth)); } if (NVreg_RmMsg != NULL) { rm_write_registry_string(sp, NULL, "RmMsg", NVreg_RmMsg, strlen(NVreg_RmMsg)); } rm_parse_option_string(sp, NVreg_RegistryDwords); for (i = 0; (entry = &nv_parms[i])->name != NULL; i++) { rm_write_registry_dword(sp, NULL, entry->name, *entry->data); } nv_kmem_cache_free_stack(sp); return NV_OK; }