nvidia gfx: fixed SMAP faults, added 2 G98 IDs to kerneldriver.
Change-Id: I3592bec014a94873eb044c35e4e041f64fbb6b3d
Reviewed-on: https://review.haiku-os.org/c/haiku/+/5819
Reviewed-by: waddlesplash <waddlesplash@gmail.com>
Diff
headers/private/graphics/nvidia/DriverInterface.h | 7 +++++--
headers/private/graphics/nvidia/nv_macros.h | 12 ++++++++++++
src/add-ons/kernel/drivers/graphics/nvidia/driver.c | 211 +++++++++++++++++++++++++++++++++++++++++++++++++++++---------------------------
3 files changed, 150 insertions(+), 80 deletions(-)
@@ -123,7 +123,9 @@
G70,
G71,
G72,
G73
G73,
G86,
G92
};
@@ -132,7 +134,8 @@
NV10A,
NV20A,
NV30A,
NV40A
NV40A,
NV50A
};
@@ -615,6 +615,13 @@
#define NV32_NV4E_I2CBUS_1 0x00600874
#define NV32_NV4E_I2CBUS_2 0x00600850
#define NV32_NV50_I2CBUS_0 0x0000e138 //confirmed
#define NV32_NV50_I2CBUS_8 0x0000e150 //?
#define NV32_NV50_I2CBUS_1 0x0000e168 //confirmed HDMI
#define NV32_NV50_I2CBUS_2 0x0000e180 //confirmed laptop panel
#define NV32_NV50_I2CBUS_6 0x0000e254 //?
#define NV32_NV50_I2CBUS_7 0x0000e274 //?
#define NV8_ATTR2INDW 0x006033c0
#define NV8_ATTR2DATW 0x006033c0
@@ -692,6 +699,7 @@
#define NVDAC_FP_DEBUG3 0x0068088c
#define NVDAC_FP_TMDS_CTRL 0x006808b0
#define NVDAC_FP_TMDS_DATA 0x006808b4
#define NVDAC2_CURPOS 0x00682300
#define NVDAC2_NV10_CURSYNC 0x00682404
@@ -725,6 +733,10 @@
#define NVDAC2_FP_DEBUG3 0x0068288c
#define NVDAC2_FP_TMDS_CTRL 0x006828b0//verify!!!
#define NVDAC2_FP_TMDS_DATA 0x006828b4//verify!!!
#define NVDAC_CRTC_FB_PITCH 0x00610b20 // | linear = 0x00100000
#define NVDAC_CRTC_FB_POS 0x00610b28
@@ -1,10 +1,10 @@
/*
Copyright 1999, Be Incorporated. All Rights Reserved.
This file may be used under the terms of the Be Sample Code License.
Other authors:
Mark Watson;
Rudolf Cornelissen 3/2002-1/2016.
Rudolf Cornelissen 3/2002-11/2022.
*/
@@ -14,6 +14,7 @@
#include <graphic_driver.h>
#include <KernelExport.h>
#include <SupportDefs.h>
#include <ISA.h>
#include <PCI.h>
#include <OS.h>
@@ -24,6 +25,9 @@
#include <stdio.h>
#include <string.h>
#define TRACE(x...) dprintf("nvidia: " x)
#define CALLED(x...) TRACE("CALLED %s\n", __PRETTY_FUNCTION__)
#define get_pci(o, s) (*pci_bus->read_pci_config)(pcii->bus, pcii->device, pcii->function, (o), (s))
#define set_pci(o, s, v) (*pci_bus->write_pci_config)(pcii->bus, pcii->device, pcii->function, (o), (s), (v))
@@ -324,6 +328,8 @@
0x03d2,
0x03d5,
0x03d6,
0x06e4,
0x06e8,
0x07e1,
0
};
@@ -397,6 +403,7 @@
uint32 cnt;
char fname[64];
CALLED();
sprintf (fname, kUserDirectory "
pcii.vendor_id, pcii.device_id, pcii.bus, pcii.device, pcii.function);
@@ -522,6 +529,7 @@
phys_addr_t physicalAddress;
system_info sysinfo;
CALLED();
uint8* rom_temp;
area_id rom_area = -1;
@@ -555,6 +563,13 @@
sprintf(buffer, DEVICE_FORMAT " regs",
di->pcii.vendor_id, di->pcii.device_id,
di->pcii.bus, di->pcii.device, di->pcii.function);
if ((di->pcii.u.h0.base_register_flags[registers] & PCI_address_type)
== PCI_address_type_64) {
TRACE("registers is 64 bit\n");
} else {
TRACE("registers is 32 bit\n");
}
si->regs_area = map_physical_memory(
@@ -601,7 +616,7 @@
di->pcii.u.h0.rom_base_pci,
di->pcii.u.h0.rom_size,
B_ANY_KERNEL_ADDRESS,
B_READ_AREA,
B_KERNEL_READ_AREA,
(void **)&(rom_temp)
);
@@ -623,7 +638,7 @@
if (!tmpUlong) {
rom_area = map_physical_memory(buffer, 0x000c0000,
65536, B_ANY_KERNEL_ADDRESS, B_READ_AREA, (void **)&(rom_temp));
65536, B_ANY_KERNEL_ADDRESS, B_KERNEL_READ_AREA, (void **)&(rom_temp));
}
@@ -658,8 +673,11 @@
physicalAddress = di->pcii.u.h0.base_registers_pci[frame_buffer];
if ((di->pcii.u.h0.base_register_flags[frame_buffer] & PCI_address_type)
== PCI_address_type_64) {
TRACE("framebuffer is 64 bit\n");
physicalAddress
|= (uint64)di->pcii.u.h0.base_registers_pci[frame_buffer + 1] << 32;
} else {
TRACE("framebuffer is 32 bit\n");
}
@@ -699,6 +717,12 @@
si->settings = sSettings;
if (si->fb_area >= 0) {
TRACE("framebuffer mapped OK\n");
} else {
TRACE("framebuffer mapping failed!\n");
}
return si->fb_area;
@@ -712,6 +736,7 @@
uint32 tmpUlong;
pci_info *pcii = &(di->pcii);
CALLED();
tmpUlong = get_pci(PCI_command, 4);
tmpUlong &= 0xfffffffc;
@@ -735,6 +760,7 @@
device_info *di = pd->di;
char tmp_name[B_OS_NAME_LENGTH];
CALLED();
while (count < MAX_DEVICES
&& (*pci_bus->get_nth_pci_info)(pci_index, &(di->pcii)) == B_OK) {
@@ -869,6 +895,7 @@
void *unaligned_dma_buffer;
uint32 mem_size;
CALLED();
while (pd->device_names[index]
@@ -890,7 +917,6 @@
sprintf(shared_name, DEVICE_FORMAT " shared",
di->pcii.vendor_id, di->pcii.device_id,
di->pcii.bus, di->pcii.device, di->pcii.function);
di->shared_area = create_area(shared_name, (void **)&(di->si), B_ANY_KERNEL_ADDRESS,
((sizeof(shared_info) + (B_PAGE_SIZE - 1)) & ~(B_PAGE_SIZE - 1)), B_FULL_LOCK,
B_KERNEL_READ_AREA | B_KERNEL_WRITE_AREA | B_CLONEABLE_AREA);
@@ -1052,6 +1078,7 @@
*cookie = di;
TRACE("open_hook: device is open\n");
goto done;
@@ -1072,6 +1099,7 @@
delete_area(di->shared_area);
di->shared_area = -1;
di->si = NULL;
TRACE("open_hook: device is freed\n");
done:
@@ -1101,6 +1129,7 @@
static status_t
close_hook(void* dev)
{
CALLED();
return B_NO_ERROR;
}
@@ -1113,6 +1142,7 @@
shared_info *si = di->si;
vuint32 *regs = di->regs;
CALLED();
AQUIRE_BEN(pd->kernel);
@@ -1170,71 +1200,77 @@
status_t result = B_DEV_INVALID_IOCTL;
uint32 tmpUlong;
CALLED();
switch (msg) {
case B_GET_ACCELERANT_SIGNATURE:
{
strcpy((char* )buf, sSettings.accelerant);
case B_GET_ACCELERANT_SIGNATURE: {
TRACE("return signature\n");
if (user_strlcpy((char* )buf, sSettings.accelerant, len) < B_OK)
return B_BAD_ADDRESS;
result = B_OK;
break;
}
case NV_GET_PRIVATE_DATA:
{
nv_get_private_data *gpd = (nv_get_private_data *)buf;
if (gpd->magic == NV_PRIVATE_DATA_MAGIC) {
gpd->shared_info_area = di->shared_area;
result = B_OK;
case NV_GET_PRIVATE_DATA: {
TRACE("return private data\n");
nv_get_private_data gpd;
if (user_memcpy(&gpd, buf, sizeof(nv_get_private_data)) < B_OK)
return B_BAD_ADDRESS;
if (gpd.magic == NV_PRIVATE_DATA_MAGIC) {
gpd.shared_info_area = di->shared_area;
result = user_memcpy(buf, &gpd, sizeof(nv_get_private_data));
}
break;
}
case NV_GET_PCI:
{
nv_get_set_pci *gsp = (nv_get_set_pci *)buf;
if (gsp->magic == NV_PRIVATE_DATA_MAGIC) {
case NV_GET_PCI: {
nv_get_set_pci gsp;
if (user_memcpy(&gsp, buf, sizeof(nv_get_set_pci)) < B_OK)
return B_BAD_ADDRESS;
if (gsp.magic == NV_PRIVATE_DATA_MAGIC) {
pci_info *pcii = &(di->pcii);
gsp->value = get_pci(gsp->offset, gsp->size);
result = B_OK;
gsp.value = get_pci(gsp.offset, gsp.size);
result = user_memcpy(buf, &gsp, sizeof(nv_get_set_pci));
}
break;
}
case NV_SET_PCI:
{
nv_get_set_pci *gsp = (nv_get_set_pci *)buf;
if (gsp->magic == NV_PRIVATE_DATA_MAGIC) {
case NV_SET_PCI: {
nv_get_set_pci gsp;
if (user_memcpy(&gsp, buf, sizeof(nv_get_set_pci)) < B_OK)
return B_BAD_ADDRESS;
if (gsp.magic == NV_PRIVATE_DATA_MAGIC) {
pci_info *pcii = &(di->pcii);
set_pci(gsp->offset, gsp->size, gsp->value);
set_pci(gsp.offset, gsp.size, gsp.value);
result = B_OK;
}
break;
}
case NV_DEVICE_NAME:
{
nv_device_name *dn = (nv_device_name *)buf;
if (dn->magic == NV_PRIVATE_DATA_MAGIC) {
strcpy(dn->name, di->name);
case NV_DEVICE_NAME: {
TRACE("return device name\n");
nv_device_name dn;
if (user_memcpy(&dn, buf, sizeof(nv_device_name)) < B_OK)
return B_BAD_ADDRESS;
if (dn.magic == NV_PRIVATE_DATA_MAGIC) {
if (user_strlcpy(dn.name, di->name, B_OS_NAME_LENGTH) < B_OK)
return B_BAD_ADDRESS;
result = B_OK;
}
break;
}
case NV_RUN_INTERRUPTS:
{
nv_set_vblank_int *vi = (nv_set_vblank_int *)buf;
if (vi->magic == NV_PRIVATE_DATA_MAGIC) {
case NV_RUN_INTERRUPTS: {
nv_set_vblank_int vi;
if (user_memcpy(&vi, buf, sizeof(nv_set_vblank_int)) < B_OK)
return B_BAD_ADDRESS;
if (vi.magic == NV_PRIVATE_DATA_MAGIC) {
vuint32 *regs = di->regs;
if (!(vi->crtc)) {
if (vi->do_it) {
if (!(vi.crtc)) {
if (vi.do_it) {
enable_vbi_crtc1(regs);
} else {
disable_vbi_crtc1(regs);
}
} else {
if (vi->do_it) {
if (vi.do_it) {
enable_vbi_crtc2(regs);
} else {
disable_vbi_crtc2(regs);
@@ -1244,44 +1280,44 @@
}
break;
}
case NV_GET_NTH_AGP_INFO:
{
nv_nth_agp_info *nai = (nv_nth_agp_info *)buf;
if (nai->magic == NV_PRIVATE_DATA_MAGIC) {
nai->exist = false;
nai->agp_bus = false;
case NV_GET_NTH_AGP_INFO: {
nv_nth_agp_info nai;
if (user_memcpy(&nai, buf, sizeof(nv_nth_agp_info)) < B_OK)
return B_BAD_ADDRESS;
if (nai.magic == NV_PRIVATE_DATA_MAGIC) {
nai.exist = false;
nai.agp_bus = false;
if (agp_bus) {
nai->agp_bus = true;
if ((*agp_bus->get_nth_agp_info)(nai->index, &(nai->agpi)) == B_NO_ERROR) {
nai->exist = true;
nai.agp_bus = true;
if ((*agp_bus->get_nth_agp_info)(nai.index, &(nai.agpi)) == B_NO_ERROR) {
nai.exist = true;
}
}
result = B_OK;
result = user_memcpy(buf, &nai, sizeof(nv_nth_agp_info));
}
break;
}
case NV_ENABLE_AGP:
{
nv_cmd_agp *nca = (nv_cmd_agp *)buf;
if (nca->magic == NV_PRIVATE_DATA_MAGIC) {
case NV_ENABLE_AGP: {
nv_cmd_agp nca;
if (user_memcpy(&nca, buf, sizeof(nv_cmd_agp)) < B_OK)
return B_BAD_ADDRESS;
if (nca.magic == NV_PRIVATE_DATA_MAGIC) {
if (agp_bus) {
nca->agp_bus = true;
nca->cmd = agp_bus->set_agp_mode(nca->cmd);
nca.agp_bus = true;
nca.cmd = agp_bus->set_agp_mode(nca.cmd);
} else {
nca->agp_bus = false;
nca->cmd = 0;
nca.agp_bus = false;
nca.cmd = 0;
}
result = B_OK;
result = user_memcpy(buf, &nca, sizeof(nv_cmd_agp));
}
break;
}
case NV_ISA_OUT:
{
nv_in_out_isa *io_isa = (nv_in_out_isa *)buf;
if (io_isa->magic == NV_PRIVATE_DATA_MAGIC) {
case NV_ISA_OUT: {
nv_in_out_isa io_isa;
if (user_memcpy(&io_isa, buf, sizeof(nv_in_out_isa)) < B_OK)
return B_BAD_ADDRESS;
if (io_isa.magic == NV_PRIVATE_DATA_MAGIC) {
pci_info *pcii = &(di->pcii);
/* lock the driver:
@@ -1293,10 +1329,10 @@
tmpUlong |= PCI_command_io;
set_pci(PCI_command, 2, tmpUlong);
if (io_isa->size == 1)
isa_bus->write_io_8(io_isa->adress, (uint8)io_isa->data);
if (io_isa.size == 1)
isa_bus->write_io_8(io_isa.adress, (uint8)io_isa.data);
else
isa_bus->write_io_16(io_isa->adress, io_isa->data);
isa_bus->write_io_16(io_isa.adress, io_isa.data);
result = B_OK;
@@ -1309,11 +1345,11 @@
}
break;
}
case NV_ISA_IN:
{
nv_in_out_isa *io_isa = (nv_in_out_isa *)buf;
if (io_isa->magic == NV_PRIVATE_DATA_MAGIC) {
case NV_ISA_IN: {
nv_in_out_isa io_isa;
if (user_memcpy(&io_isa, buf, sizeof(nv_in_out_isa)) < B_OK)
return B_BAD_ADDRESS;
if (io_isa.magic == NV_PRIVATE_DATA_MAGIC) {
pci_info *pcii = &(di->pcii);
/* lock the driver:
@@ -1325,11 +1361,11 @@
tmpUlong |= PCI_command_io;
set_pci(PCI_command, 2, tmpUlong);
if (io_isa->size == 1)
io_isa->data = isa_bus->read_io_8(io_isa->adress);
if (io_isa.size == 1)
io_isa.data = isa_bus->read_io_8(io_isa.adress);
else
io_isa->data = isa_bus->read_io_16(io_isa->adress);
result = B_OK;
io_isa.data = isa_bus->read_io_16(io_isa.adress);
result = user_memcpy(buf, &io_isa, sizeof(nv_in_out_isa));
tmpUlong = get_pci(PCI_command, 2);
@@ -1357,6 +1393,7 @@
pci_info pcii;
bool found = false;
CALLED();
if (get_module(B_PCI_MODULE_NAME, (module_info **)&pci_bus) != B_OK)
return B_ERROR;
@@ -1395,6 +1432,12 @@
}
done:
if (found) {
TRACE ("init_hardware: found device\n");
} else {
TRACE ("init_hardware: no supported device found\n");
}
put_module(B_PCI_MODULE_NAME);
return found ? B_OK : B_ERROR;
@@ -1406,6 +1449,7 @@
{
void *settings;
CALLED();
settings = load_driver_settings(DRIVER_PREFIX ".settings");
if (settings != NULL) {
@@ -1413,6 +1457,7 @@
char *end;
uint32 value;
TRACE("init_driver: nvidia.settings loaded\n");
item = get_driver_parameter(settings, "accelerant", "", "");
if (item[0] && strlen(item) < sizeof(sSettings.accelerant) - 1)
@@ -1424,6 +1469,12 @@
sSettings.dumprom = get_driver_boolean_parameter(settings,
"dumprom", false, false);
if (sSettings.dumprom) {
TRACE("dumprom requested\n");
} else {
TRACE("no dumprom requested\n");
}
item = get_driver_parameter(settings, "logmask",
@@ -1503,6 +1554,8 @@
INIT_BEN(pd->kernel);
probe_devices();
TRACE("init_driver: completed OK\n");
return B_OK;
}
@@ -1510,6 +1563,7 @@
const char **
publish_devices(void)
{
CALLED();
return (const char **)pd->device_names;
}
@@ -1532,6 +1586,7 @@
void
uninit_driver(void)
{
CALLED();
DELETE_BEN(pd->kernel);
free(pd);