0001
0002
0003
0004
0005
0006
0007
0008
0009
0010
0011
0012
0013
0014
0015
0016
0017
0018
0019
0020
0021
0022
0023
0024 #include "mem.h"
0025 #include "vmm.h"
0026
0027 #include <core/option.h>
0028
0029 #include <nvif/class.h>
0030
0031 static void
0032 nv44_mmu_init(struct nvkm_mmu *mmu)
0033 {
0034 struct nvkm_device *device = mmu->subdev.device;
0035 struct nvkm_memory *pt = mmu->vmm->pd->pt[0]->memory;
0036 u32 addr;
0037
0038
0039
0040
0041
0042 addr = nvkm_rd32(device, 0x10020c);
0043 addr -= ((nvkm_memory_addr(pt) >> 19) + 1) << 19;
0044
0045 nvkm_wr32(device, 0x100850, 0x80000000);
0046 nvkm_wr32(device, 0x100818, mmu->vmm->null);
0047 nvkm_wr32(device, 0x100804, (nvkm_memory_size(pt) / 4) * 4096);
0048 nvkm_wr32(device, 0x100850, 0x00008000);
0049 nvkm_mask(device, 0x10008c, 0x00000200, 0x00000200);
0050 nvkm_wr32(device, 0x100820, 0x00000000);
0051 nvkm_wr32(device, 0x10082c, 0x00000001);
0052 nvkm_wr32(device, 0x100800, addr | 0x00000010);
0053 }
0054
0055 static const struct nvkm_mmu_func
0056 nv44_mmu = {
0057 .init = nv44_mmu_init,
0058 .dma_bits = 39,
0059 .mmu = {{ -1, -1, NVIF_CLASS_MMU_NV04}},
0060 .mem = {{ -1, -1, NVIF_CLASS_MEM_NV04}, nv04_mem_new, nv04_mem_map },
0061 .vmm = {{ -1, -1, NVIF_CLASS_VMM_NV04}, nv44_vmm_new, true },
0062 };
0063
0064 int
0065 nv44_mmu_new(struct nvkm_device *device, enum nvkm_subdev_type type, int inst,
0066 struct nvkm_mmu **pmmu)
0067 {
0068 if (device->type == NVKM_DEVICE_AGP ||
0069 !nvkm_boolopt(device->cfgopt, "NvPCIE", true))
0070 return nv04_mmu_new(device, type, inst, pmmu);
0071
0072 return nvkm_mmu_new_(&nv44_mmu, device, type, inst, pmmu);
0073 }