mirror of
https://github.com/NVIDIA/open-gpu-kernel-modules.git
synced 2025-01-20 15:52:10 +01:00
508 lines
20 KiB
Python
Executable File
508 lines
20 KiB
Python
Executable File
#!/usr/bin/env python3
|
|
|
|
# SPDX-FileCopyrightText: Copyright (c) 2023 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.
|
|
|
|
# Converts OpenRM binhex-encoded images to Nouveau-compatible binary blobs
|
|
# See nouveau_firmware_layout.ods for documentation on the file format
|
|
|
|
import sys
|
|
import os
|
|
import argparse
|
|
import shutil
|
|
import re
|
|
import gzip
|
|
import struct
|
|
|
|
class MyException(Exception):
|
|
pass
|
|
|
|
def round_up_to_base(x, base = 10):
|
|
return x + (base - x) % base
|
|
|
|
def getbytes(filename, array):
|
|
"""Extract the bytes for the given array in the given file.
|
|
|
|
:param filename: the file to parse
|
|
:param array: the name of the array to parse
|
|
:returns: byte array
|
|
|
|
This function scans the file for the array and returns a bytearray of
|
|
its contents, uncompressing the data if it is tagged as compressed.
|
|
|
|
This function assumes that each array is immediately preceded with a comment
|
|
section that specifies whether the array is compressed and how many bytes of
|
|
data there should be. Example:
|
|
|
|
#if defined(BINDATA_INCLUDE_DATA)
|
|
//
|
|
// FUNCTION: ksec2GetBinArchiveSecurescrubUcode_AD10X("header_prod")
|
|
// FILE NAME: kernel/inc/securescrub/bin/ad10x/g_securescrubuc_sec2_ad10x_boot_from_hs_prod.h
|
|
// FILE TYPE: TEXT
|
|
// VAR NAME: securescrub_ucode_header_ad10x_boot_from_hs
|
|
// COMPRESSION: YES
|
|
// COMPLEX_STRUCT: NO
|
|
// DATA SIZE (bytes): 36
|
|
// COMPRESSED SIZE (bytes): 27
|
|
//
|
|
static BINDATA_CONST NvU8 ksec2BinArchiveSecurescrubUcode_AD10X_header_prod_data[] =
|
|
{
|
|
0x63, 0x60, 0x00, 0x02, 0x46, 0x20, 0x96, 0x02, 0x62, 0x66, 0x08, 0x13, 0x4c, 0x48, 0x42, 0x69,
|
|
0x20, 0x00, 0x00, 0x30, 0x39, 0x0a, 0xfc, 0x24, 0x00, 0x00, 0x00,
|
|
};
|
|
#endif // defined(BINDATA_INCLUDE_DATA)
|
|
"""
|
|
|
|
with open(filename) as f:
|
|
for line in f:
|
|
if "COMPRESSION: NO" in line:
|
|
compressed = False
|
|
if "COMPRESSION: YES" in line:
|
|
compressed = True
|
|
m = re.search("DATA SIZE \(bytes\): (\d+)", line)
|
|
if m:
|
|
data_size = int(m.group(1))
|
|
m = re.search("COMPRESSED SIZE \(bytes\): (\d+)", line)
|
|
if m:
|
|
compressed_size = int(m.group(1))
|
|
if "static BINDATA_CONST NvU8 " + array in line:
|
|
break
|
|
else:
|
|
raise MyException(f"array {array} not found in {filename}")
|
|
|
|
output = b''
|
|
for line in f:
|
|
if "};" in line:
|
|
break
|
|
bytes = [int(b, 16) for b in re.findall('0x[0-9a-f][0-9a-f]', line)]
|
|
if len(bytes) > 0:
|
|
output += struct.pack(f"{len(bytes)}B", *bytes)
|
|
|
|
if len(output) == 0:
|
|
raise MyException(f"no data found for {array}")
|
|
|
|
if compressed:
|
|
if len(output) != compressed_size:
|
|
raise MyException(f"compressed array {array} in {filename} should be {compressed_size} bytes but is actually {len(output)}.")
|
|
gzipheader = struct.pack("<4BL2B", 0x1f, 0x8b, 8, 0, 0, 0, 3)
|
|
output = gzip.decompress(gzipheader + output)
|
|
if len(output) != data_size:
|
|
raise MyException(f"array {array} in {filename} decompressed to {len(output)} bytes but should have been {data_size} bytes.")
|
|
return output
|
|
else:
|
|
if len(output) != data_size:
|
|
raise MyException(f"array {array} in {filename} should be {compressed_size} bytes but is actually {len(output)}.")
|
|
return output
|
|
|
|
# GSP bootloader
|
|
def bootloader(gpu, type):
|
|
global outputpath
|
|
global version
|
|
|
|
GPU=gpu.upper()
|
|
filename = f"src/nvidia/generated/g_bindata_kgspGetBinArchiveGspRmBoot_{GPU}.c"
|
|
|
|
print(f"Creating nvidia/{gpu}/gsp/bootloader-{version}.bin")
|
|
os.makedirs(f"{outputpath}/nvidia/{gpu}/gsp/", exist_ok = True)
|
|
|
|
with open(f"{outputpath}/nvidia/{gpu}/gsp/bootloader-{version}.bin", "wb") as f:
|
|
# Extract the actual bootloader firmware
|
|
array = f"kgspBinArchiveGspRmBoot_{GPU}_ucode_image{type}data"
|
|
firmware = getbytes(filename, array)
|
|
firmware_size = len(firmware)
|
|
|
|
# Extract the descriptor (RM_RISCV_UCODE_DESC)
|
|
array = f"kgspBinArchiveGspRmBoot_{GPU}_ucode_desc{type}data"
|
|
descriptor = getbytes(filename, array)
|
|
descriptor_size = len(descriptor)
|
|
|
|
# First, add the nvfw_bin_hdr header
|
|
total_size = round_up_to_base(24 + firmware_size + descriptor_size, 256)
|
|
firmware_offset = 24 + descriptor_size
|
|
f.write(struct.pack("<6L", 0x10de, 1, total_size, 24, firmware_offset, firmware_size))
|
|
|
|
# Second, add the descriptor
|
|
f.write(descriptor)
|
|
|
|
# Finally, the actual bootloader image
|
|
f.write(firmware)
|
|
|
|
# GSP Booter load and unload
|
|
def booter(gpu, load, sigsize):
|
|
global outputpath
|
|
global version
|
|
|
|
GPU = gpu.upper()
|
|
LOAD = load.capitalize()
|
|
|
|
filename = f"src/nvidia/generated/g_bindata_kgspGetBinArchiveBooter{LOAD}Ucode_{GPU}.c"
|
|
|
|
print(f"Creating nvidia/{gpu}/gsp/booter_{load}-{version}.bin")
|
|
os.makedirs(f"{outputpath}/nvidia/{gpu}/gsp/", exist_ok = True)
|
|
|
|
with open(f"{outputpath}/nvidia/{gpu}/gsp/booter_{load}-{version}.bin", "wb") as f:
|
|
# Extract the actual scrubber firmware
|
|
array = f"kgspBinArchiveBooter{LOAD}Ucode_{GPU}_image_prod_data"
|
|
firmware = getbytes(filename, array)
|
|
firmware_size = len(firmware)
|
|
|
|
# Extract the signatures
|
|
array = f"kgspBinArchiveBooter{LOAD}Ucode_{GPU}_sig_prod_data"
|
|
signatures = getbytes(filename, array)
|
|
signatures_size = len(signatures)
|
|
if signatures_size % sigsize:
|
|
raise MyException(f"signature file size for {array} is uneven value of {sigsize}")
|
|
num_sigs = int(signatures_size / sigsize);
|
|
if num_sigs < 1:
|
|
raise MyException(f"invalid number of signatures {num_sigs}")
|
|
|
|
# First, add the nvfw_bin_hdr header
|
|
total_size = round_up_to_base(120 + signatures_size + firmware_size, 256)
|
|
firmware_offset = 120 + signatures_size
|
|
f.write(struct.pack("<6L", 0x10de, 1, total_size, 24, firmware_offset, firmware_size))
|
|
|
|
# Second, add the nvfw_hs_header_v2 header
|
|
patch_loc_offset = 60 + signatures_size
|
|
patch_sig_offset = patch_loc_offset + 4
|
|
meta_data_offset = patch_sig_offset + 4
|
|
num_sig_offset = meta_data_offset + 12
|
|
header_offset = num_sig_offset + 4
|
|
f.write(struct.pack("<9L", 60, signatures_size, patch_loc_offset,
|
|
patch_sig_offset, meta_data_offset, 12,
|
|
num_sig_offset, header_offset, 36))
|
|
|
|
# Third, the actual signatures
|
|
f.write(signatures)
|
|
|
|
# Extract the patch location
|
|
array = f"kgspBinArchiveBooter{LOAD}Ucode_{GPU}_patch_loc_data"
|
|
bytes = getbytes(filename, array)
|
|
patchloc = struct.unpack("<L", bytes)[0]
|
|
|
|
# Extract the patch meta variables
|
|
array = f"kgspBinArchiveBooter{LOAD}Ucode_{GPU}_patch_meta_data"
|
|
bytes = getbytes(filename, array)
|
|
fuse_ver, engine_id, ucode_id = struct.unpack("<LLL", bytes)
|
|
|
|
# Fourth, patch_loc[], patch_sig[], fuse_ver, engine_id, ucode_id, and num_sigs
|
|
f.write(struct.pack("<6L", patchloc, 0, fuse_ver, engine_id, ucode_id, num_sigs))
|
|
|
|
# Extract the descriptor (nvkm_gsp_booter_fw_hdr)
|
|
array = f"kgspBinArchiveBooter{LOAD}Ucode_{GPU}_header_prod_data"
|
|
descriptor = getbytes(filename, array)
|
|
|
|
# Fifth, the descriptor
|
|
f.write(descriptor)
|
|
|
|
# And finally, the actual scrubber image
|
|
f.write(firmware)
|
|
|
|
# GPU memory scrubber, needed for some GPUs and configurations
|
|
def scrubber(gpu, sigsize):
|
|
global outputpath
|
|
global version
|
|
|
|
# Unfortunately, RM breaks convention with the scrubber image and labels
|
|
# the files and arrays with AD10X instead of AD102.
|
|
GPUX = f"{gpu[:-1].upper()}X"
|
|
|
|
filename = f"src/nvidia/generated/g_bindata_ksec2GetBinArchiveSecurescrubUcode_{GPUX}.c"
|
|
|
|
print(f"Creating nvidia/{gpu}/gsp/scrubber-{version}.bin")
|
|
os.makedirs(f"{outputpath}/nvidia/{gpu}/gsp/", exist_ok = True)
|
|
|
|
with open(f"{outputpath}/nvidia/{gpu}/gsp/scrubber-{version}.bin", "wb") as f:
|
|
# Extract the actual scrubber firmware
|
|
array = f"ksec2BinArchiveSecurescrubUcode_{GPUX}_image_prod_data[]"
|
|
firmware = getbytes(filename, array)
|
|
firmware_size = len(firmware)
|
|
|
|
# Extract the signatures
|
|
array = f"ksec2BinArchiveSecurescrubUcode_{GPUX}_sig_prod_data"
|
|
signatures = getbytes(filename, array)
|
|
signatures_size = len(signatures)
|
|
if signatures_size % sigsize:
|
|
raise MyException(f"signature file size for {array} is uneven value of {sigsize}")
|
|
num_sigs = int(signatures_size / sigsize);
|
|
if num_sigs < 1:
|
|
raise MyException(f"invalid number of signatures {num_sigs}")
|
|
|
|
# First, add the nvfw_bin_hdr header
|
|
total_size = round_up_to_base(120 + signatures_size + firmware_size, 256)
|
|
firmware_offset = 120 + signatures_size
|
|
f.write(struct.pack("<6L", 0x10de, 1, total_size, 24, firmware_offset, firmware_size))
|
|
|
|
# Second, add the nvfw_hs_header_v2 header
|
|
patch_loc_offset = 60 + signatures_size
|
|
patch_sig_offset = patch_loc_offset + 4
|
|
meta_data_offset = patch_sig_offset + 4
|
|
num_sig_offset = meta_data_offset + 12
|
|
header_offset = num_sig_offset + 4
|
|
f.write(struct.pack("<9L", 60, signatures_size, patch_loc_offset,
|
|
patch_sig_offset, meta_data_offset, 12,
|
|
num_sig_offset, header_offset, 36))
|
|
|
|
# Third, the actual signatures
|
|
f.write(signatures)
|
|
|
|
# Extract the patch location
|
|
array = f"ksec2BinArchiveSecurescrubUcode_{GPUX}_patch_loc_data"
|
|
bytes = getbytes(filename, array)
|
|
patchloc = struct.unpack("<L", bytes)[0]
|
|
|
|
# Extract the patch meta variables
|
|
array = f"ksec2BinArchiveSecurescrubUcode_{GPUX}_patch_meta_data"
|
|
bytes = getbytes(filename, array)
|
|
fuse_ver, engine_id, ucode_id = struct.unpack("<LLL", bytes)
|
|
|
|
# Fourth, patch_loc[], patch_sig[], fuse_ver, engine_id, ucode_id, and num_sigs
|
|
f.write(struct.pack("<6L", patchloc, 0, fuse_ver, engine_id, ucode_id, num_sigs))
|
|
|
|
# Extract the descriptor (nvkm_gsp_booter_fw_hdr)
|
|
array = f"ksec2BinArchiveSecurescrubUcode_{GPUX}_header_prod_data"
|
|
descriptor = getbytes(filename, array)
|
|
|
|
# Fifth, the descriptor
|
|
f.write(descriptor)
|
|
|
|
# And finally, the actual scrubber image
|
|
f.write(firmware)
|
|
|
|
# Extract the GSP-RM firmware from the .run file and copy the binaries
|
|
# to the target directory.
|
|
def gsp_firmware(filename):
|
|
global outputpath
|
|
global version
|
|
|
|
import subprocess
|
|
import tempfile
|
|
import shutil
|
|
import time
|
|
|
|
with tempfile.TemporaryDirectory() as temp:
|
|
os.chdir(temp)
|
|
|
|
try:
|
|
print(f"Extracting {os.path.basename(filename)} to {temp}")
|
|
# The -x parameter tells the installer to only extract the
|
|
# contents and then exit.
|
|
subprocess.run(['/bin/sh', filename, '-x'], shell=False,
|
|
check=True, timeout=60,
|
|
stdout = subprocess.PIPE, stderr = subprocess.STDOUT)
|
|
except subprocess.CalledProcessError as error:
|
|
print(error.output.decode())
|
|
sys.exit(error.returncode)
|
|
except subprocess.TimeoutExpired as error:
|
|
print(error.output.decode())
|
|
sys.exit(error.returncode)
|
|
|
|
try:
|
|
# The .run file extracts its contents to a directory with the same
|
|
# name as the file itself, minus the .run. The GSP-RM firmware
|
|
# images are in the 'firmware' subdirectory.
|
|
directory = os.path.splitext(os.path.basename(filename))[0]
|
|
os.chdir(f"{directory}/firmware")
|
|
except:
|
|
print("Firmware failed to extract")
|
|
sys.exit(1)
|
|
|
|
if not os.path.exists('gsp_tu10x.bin') or not os.path.exists('gsp_ga10x.bin'):
|
|
print("Firmware files are missing")
|
|
sys.exit(1)
|
|
|
|
shutil.copyfile('gsp_tu10x.bin', f"{outputpath}/nvidia/tu102/gsp/gsp-{version}.bin")
|
|
print(f"Copied gsp_tu10x.bin to tu102/gsp/gsp-{version}.bin")
|
|
shutil.copyfile('gsp_ga10x.bin', f"{outputpath}/nvidia/ga102/gsp/gsp-{version}.bin")
|
|
print(f"Copied gsp_ga10x.bin to ga102/gsp/gsp-{version}.bin")
|
|
|
|
# Create a symlink, deleting the existing file/link if necessary
|
|
def symlink(src, dst, target_is_directory = False):
|
|
import errno
|
|
|
|
try:
|
|
os.symlink(src, dst, target_is_directory = target_is_directory)
|
|
except OSError as e:
|
|
if e.errno == errno.EEXIST:
|
|
os.remove(dst)
|
|
os.symlink(src, dst, target_is_directory = target_is_directory)
|
|
else:
|
|
raise e
|
|
|
|
# Create symlinks in the target directory for the other GPUs. This mirrors
|
|
# what the WHENCE file in linux-firmware does.
|
|
def symlinks():
|
|
global outputpath
|
|
global version
|
|
|
|
print(f"Creating symlinks in {outputpath}/nvidia")
|
|
os.chdir(f"{outputpath}/nvidia")
|
|
|
|
for d in ['tu104', 'tu106', 'tu117']:
|
|
os.makedirs(d, exist_ok = True)
|
|
symlink('../tu102/gsp', f"{d}/gsp", target_is_directory = True)
|
|
|
|
for d in ['ad103', 'ad104', 'ad106', 'ad107']:
|
|
os.makedirs(d, exist_ok = True)
|
|
symlink('../ad102/gsp', f"{d}/gsp", target_is_directory = True)
|
|
|
|
for d in ['ga103', 'ga104', 'ga106', 'ga107']:
|
|
os.makedirs(d, exist_ok = True)
|
|
symlink('../ga102/gsp', f"{d}/gsp", target_is_directory = True)
|
|
|
|
symlink(f"../../tu102/gsp/bootloader-{version}.bin", f"tu116/gsp/bootloader-{version}.bin")
|
|
|
|
symlink(f"../../tu102/gsp/gsp-{version}.bin", f"tu116/gsp/gsp-{version}.bin")
|
|
symlink(f"../../tu102/gsp/gsp-{version}.bin", f"ga100/gsp/gsp-{version}.bin")
|
|
symlink(f"../../ga102/gsp/gsp-{version}.bin", f"ad102/gsp/gsp-{version}.bin")
|
|
|
|
# Create a text file that can be inserted as-is to the WHENCE file of the
|
|
# linux-firmware git repository.
|
|
def whence():
|
|
global outputpath
|
|
global version
|
|
|
|
whence = f"""
|
|
File: nvidia/tu102/gsp/bootloader-{version}.bin
|
|
File: nvidia/tu102/gsp/booter_load-{version}.bin
|
|
File: nvidia/tu102/gsp/booter_unload-{version}.bin
|
|
Link: nvidia/tu104/gsp -> ../tu102/gsp
|
|
Link: nvidia/tu106/gsp -> ../tu102/gsp
|
|
|
|
File: nvidia/tu116/gsp/booter_load-{version}.bin
|
|
File: nvidia/tu116/gsp/booter_unload-{version}.bin
|
|
Link: nvidia/tu116/gsp/bootloader-{version}.bin -> ../../tu102/gsp/bootloader-{version}.bin
|
|
Link: nvidia/tu117/gsp -> ../tu116/gsp
|
|
|
|
File: nvidia/ga100/gsp/bootloader-{version}.bin
|
|
File: nvidia/ga100/gsp/booter_load-{version}.bin
|
|
File: nvidia/ga100/gsp/booter_unload-{version}.bin
|
|
|
|
File: nvidia/ad102/gsp/bootloader-{version}.bin
|
|
File: nvidia/ad102/gsp/booter_load-{version}.bin
|
|
File: nvidia/ad102/gsp/booter_unload-{version}.bin
|
|
Link: nvidia/ad103/gsp -> ../ad102/gsp
|
|
Link: nvidia/ad104/gsp -> ../ad102/gsp
|
|
Link: nvidia/ad106/gsp -> ../ad102/gsp
|
|
Link: nvidia/ad107/gsp -> ../ad102/gsp
|
|
|
|
File: nvidia/ga102/gsp/bootloader-{version}.bin
|
|
File: nvidia/ga102/gsp/booter_load-{version}.bin
|
|
File: nvidia/ga102/gsp/booter_unload-{version}.bin
|
|
Link: nvidia/ga103/gsp -> ../ga102/gsp
|
|
Link: nvidia/ga104/gsp -> ../ga102/gsp
|
|
Link: nvidia/ga106/gsp -> ../ga102/gsp
|
|
Link: nvidia/ga107/gsp -> ../ga102/gsp
|
|
|
|
File: nvidia/tu102/gsp/gsp-{version}.bin
|
|
Origin: gsp_tu10x.bin from NVIDIA-Linux-x86_64-{version}.run
|
|
Link: nvidia/tu116/gsp/gsp-{version}.bin -> ../../tu102/gsp/gsp-{version}.bin
|
|
Link: nvidia/ga100/gsp/gsp-{version}.bin -> ../../tu102/gsp/gsp-{version}.bin
|
|
|
|
File: nvidia/ga102/gsp/gsp-{version}.bin
|
|
Origin: gsp_ga10x.bin from NVIDIA-Linux-x86_64-{version}.run
|
|
Link: nvidia/ad102/gsp/gsp-{version}.bin -> ../../ga102/gsp/gsp-{version}.bin
|
|
"""
|
|
|
|
with open(f"{outputpath}/WHENCE.txt", 'w') as f:
|
|
f.writelines(whence)
|
|
|
|
print(f"Created {outputpath}/WHENCE.txt")
|
|
|
|
def main():
|
|
global outputpath
|
|
global version
|
|
|
|
parser = argparse.ArgumentParser(
|
|
description = 'Extract firmware binaries from the OpenRM git repository'
|
|
' in a format expected by the Nouveau device driver.',
|
|
epilog = 'Running as root and specifying -o /lib/firmware will install'
|
|
' the firmware files directly where Nouveau expects them.'
|
|
' The --revision option is useful for testing new firmware'
|
|
' versions without changing Nouveau source code.'
|
|
' The --driver option lets you specify the path to the .run file,'
|
|
' and this script also will extract and copy the GSP-RM firmware images.')
|
|
parser.add_argument('-i', '--input', default = os.getcwd(),
|
|
help = 'Path to source directory (where version.mk exists)')
|
|
parser.add_argument('-o', '--output', default = os.path.abspath(os.getcwd() + '/_out'),
|
|
help = 'Path to target directory (where files will be written)')
|
|
parser.add_argument('-r', '--revision',
|
|
help = 'Files will be named with this version number')
|
|
parser.add_argument('-d', '--driver',
|
|
help = 'Path to Nvidia driver .run package, for also extracting the GSP-RM firmware')
|
|
parser.add_argument('-s', '--symlink', action='store_true',
|
|
help = 'Also create symlinks for all supported GPUs')
|
|
parser.add_argument('-w', '--whence', action='store_true',
|
|
help = 'Also generate a WHENCE file')
|
|
args = parser.parse_args()
|
|
|
|
os.chdir(args.input)
|
|
|
|
if args.driver:
|
|
if not os.path.exists(args.driver):
|
|
print(f"File {args.driver} does not exist.")
|
|
sys.exit(1)
|
|
|
|
version = args.revision
|
|
if not version:
|
|
with open("version.mk") as f:
|
|
version = re.search(r'^NVIDIA_VERSION = ([^\s]+)', f.read(), re.MULTILINE).group(1)
|
|
|
|
print(f"Generating files for version {version}")
|
|
|
|
outputpath = args.output;
|
|
print(f"Writing files to {outputpath}")
|
|
|
|
os.makedirs(f"{outputpath}/nvidia", exist_ok = True)
|
|
|
|
booter("tu102", "load", 16)
|
|
booter("tu102", "unload", 16)
|
|
bootloader("tu102", "_")
|
|
|
|
booter("tu116", "load", 16)
|
|
booter("tu116", "unload", 16)
|
|
# TU11x uses the same bootloader as TU10x
|
|
|
|
booter("ga100", "load", 384)
|
|
booter("ga100", "unload", 384)
|
|
bootloader("ga100", "_")
|
|
|
|
booter("ga102", "load", 384)
|
|
booter("ga102", "unload", 384)
|
|
bootloader("ga102", "_prod_")
|
|
|
|
booter("ad102", "load", 384)
|
|
booter("ad102", "unload", 384)
|
|
bootloader("ad102", "_prod_")
|
|
# scrubber("ad102", 384) # Not currently used by Nouveau
|
|
|
|
if args.driver:
|
|
gsp_firmware(os.path.abspath(args.driver))
|
|
|
|
if args.symlink:
|
|
symlinks()
|
|
|
|
if args.whence:
|
|
whence()
|
|
|
|
if __name__ == "__main__":
|
|
main()
|
|
|