/drivers/char/agp/intel-gtt.c
C | 1624 lines | 1274 code | 233 blank | 117 comment | 196 complexity | ecc6fdd1f1979313ae302f8f52093123 MD5 | raw file
Possible License(s): LGPL-2.0, AGPL-1.0, GPL-2.0
- /*
- * Intel GTT (Graphics Translation Table) routines
- *
- * Caveat: This driver implements the linux agp interface, but this is far from
- * a agp driver! GTT support ended up here for purely historical reasons: The
- * old userspace intel graphics drivers needed an interface to map memory into
- * the GTT. And the drm provides a default interface for graphic devices sitting
- * on an agp port. So it made sense to fake the GTT support as an agp port to
- * avoid having to create a new api.
- *
- * With gem this does not make much sense anymore, just needlessly complicates
- * the code. But as long as the old graphics stack is still support, it's stuck
- * here.
- *
- * /fairy-tale-mode off
- */
- /*
- * If we have Intel graphics, we're not going to have anything other than
- * an Intel IOMMU. So make the correct use of the PCI DMA API contingent
- * on the Intel IOMMU support (CONFIG_DMAR).
- * Only newer chipsets need to bother with this, of course.
- */
- #ifdef CONFIG_DMAR
- #define USE_PCI_DMA_API 1
- #endif
- /* Max amount of stolen space, anything above will be returned to Linux */
- int intel_max_stolen = 32 * 1024 * 1024;
- EXPORT_SYMBOL(intel_max_stolen);
- static const struct aper_size_info_fixed intel_i810_sizes[] =
- {
- {64, 16384, 4},
- /* The 32M mode still requires a 64k gatt */
- {32, 8192, 4}
- };
- #define AGP_DCACHE_MEMORY 1
- #define AGP_PHYS_MEMORY 2
- #define INTEL_AGP_CACHED_MEMORY 3
- static struct gatt_mask intel_i810_masks[] =
- {
- {.mask = I810_PTE_VALID, .type = 0},
- {.mask = (I810_PTE_VALID | I810_PTE_LOCAL), .type = AGP_DCACHE_MEMORY},
- {.mask = I810_PTE_VALID, .type = 0},
- {.mask = I810_PTE_VALID | I830_PTE_SYSTEM_CACHED,
- .type = INTEL_AGP_CACHED_MEMORY}
- };
- #define INTEL_AGP_UNCACHED_MEMORY 0
- #define INTEL_AGP_CACHED_MEMORY_LLC 1
- #define INTEL_AGP_CACHED_MEMORY_LLC_GFDT 2
- #define INTEL_AGP_CACHED_MEMORY_LLC_MLC 3
- #define INTEL_AGP_CACHED_MEMORY_LLC_MLC_GFDT 4
- static struct gatt_mask intel_gen6_masks[] =
- {
- {.mask = I810_PTE_VALID | GEN6_PTE_UNCACHED,
- .type = INTEL_AGP_UNCACHED_MEMORY },
- {.mask = I810_PTE_VALID | GEN6_PTE_LLC,
- .type = INTEL_AGP_CACHED_MEMORY_LLC },
- {.mask = I810_PTE_VALID | GEN6_PTE_LLC | GEN6_PTE_GFDT,
- .type = INTEL_AGP_CACHED_MEMORY_LLC_GFDT },
- {.mask = I810_PTE_VALID | GEN6_PTE_LLC_MLC,
- .type = INTEL_AGP_CACHED_MEMORY_LLC_MLC },
- {.mask = I810_PTE_VALID | GEN6_PTE_LLC_MLC | GEN6_PTE_GFDT,
- .type = INTEL_AGP_CACHED_MEMORY_LLC_MLC_GFDT },
- };
- static struct _intel_private {
- struct pci_dev *pcidev; /* device one */
- u8 __iomem *registers;
- u32 __iomem *gtt; /* I915G */
- int num_dcache_entries;
- /* gtt_entries is the number of gtt entries that are already mapped
- * to stolen memory. Stolen memory is larger than the memory mapped
- * through gtt_entries, as it includes some reserved space for the BIOS
- * popup and for the GTT.
- */
- int gtt_entries; /* i830+ */
- int gtt_total_size;
- union {
- void __iomem *i9xx_flush_page;
- void *i8xx_flush_page;
- };
- struct page *i8xx_page;
- struct resource ifp_resource;
- int resource_valid;
- } intel_private;
- #ifdef USE_PCI_DMA_API
- static int intel_agp_map_page(struct page *page, dma_addr_t *ret)
- {
- *ret = pci_map_page(intel_private.pcidev, page, 0,
- PAGE_SIZE, PCI_DMA_BIDIRECTIONAL);
- if (pci_dma_mapping_error(intel_private.pcidev, *ret))
- return -EINVAL;
- return 0;
- }
- static void intel_agp_unmap_page(struct page *page, dma_addr_t dma)
- {
- pci_unmap_page(intel_private.pcidev, dma,
- PAGE_SIZE, PCI_DMA_BIDIRECTIONAL);
- }
- static void intel_agp_free_sglist(struct agp_memory *mem)
- {
- struct sg_table st;
- st.sgl = mem->sg_list;
- st.orig_nents = st.nents = mem->page_count;
- sg_free_table(&st);
- mem->sg_list = NULL;
- mem->num_sg = 0;
- }
- static int intel_agp_map_memory(struct agp_memory *mem)
- {
- struct sg_table st;
- struct scatterlist *sg;
- int i;
- DBG("try mapping %lu pages\n", (unsigned long)mem->page_count);
- if (sg_alloc_table(&st, mem->page_count, GFP_KERNEL))
- return -ENOMEM;
- mem->sg_list = sg = st.sgl;
- for (i = 0 ; i < mem->page_count; i++, sg = sg_next(sg))
- sg_set_page(sg, mem->pages[i], PAGE_SIZE, 0);
- mem->num_sg = pci_map_sg(intel_private.pcidev, mem->sg_list,
- mem->page_count, PCI_DMA_BIDIRECTIONAL);
- if (unlikely(!mem->num_sg)) {
- intel_agp_free_sglist(mem);
- return -ENOMEM;
- }
- return 0;
- }
- static void intel_agp_unmap_memory(struct agp_memory *mem)
- {
- DBG("try unmapping %lu pages\n", (unsigned long)mem->page_count);
- pci_unmap_sg(intel_private.pcidev, mem->sg_list,
- mem->page_count, PCI_DMA_BIDIRECTIONAL);
- intel_agp_free_sglist(mem);
- }
- static void intel_agp_insert_sg_entries(struct agp_memory *mem,
- off_t pg_start, int mask_type)
- {
- struct scatterlist *sg;
- int i, j;
- j = pg_start;
- WARN_ON(!mem->num_sg);
- if (mem->num_sg == mem->page_count) {
- for_each_sg(mem->sg_list, sg, mem->page_count, i) {
- writel(agp_bridge->driver->mask_memory(agp_bridge,
- sg_dma_address(sg), mask_type),
- intel_private.gtt+j);
- j++;
- }
- } else {
- /* sg may merge pages, but we have to separate
- * per-page addr for GTT */
- unsigned int len, m;
- for_each_sg(mem->sg_list, sg, mem->num_sg, i) {
- len = sg_dma_len(sg) / PAGE_SIZE;
- for (m = 0; m < len; m++) {
- writel(agp_bridge->driver->mask_memory(agp_bridge,
- sg_dma_address(sg) + m * PAGE_SIZE,
- mask_type),
- intel_private.gtt+j);
- j++;
- }
- }
- }
- readl(intel_private.gtt+j-1);
- }
- #else
- static void intel_agp_insert_sg_entries(struct agp_memory *mem,
- off_t pg_start, int mask_type)
- {
- int i, j;
- for (i = 0, j = pg_start; i < mem->page_count; i++, j++) {
- writel(agp_bridge->driver->mask_memory(agp_bridge,
- page_to_phys(mem->pages[i]), mask_type),
- intel_private.gtt+j);
- }
- readl(intel_private.gtt+j-1);
- }
- #endif
- static int intel_i810_fetch_size(void)
- {
- u32 smram_miscc;
- struct aper_size_info_fixed *values;
- pci_read_config_dword(agp_bridge->dev, I810_SMRAM_MISCC, &smram_miscc);
- values = A_SIZE_FIX(agp_bridge->driver->aperture_sizes);
- if ((smram_miscc & I810_GMS) == I810_GMS_DISABLE) {
- dev_warn(&agp_bridge->dev->dev, "i810 is disabled\n");
- return 0;
- }
- if ((smram_miscc & I810_GFX_MEM_WIN_SIZE) == I810_GFX_MEM_WIN_32M) {
- agp_bridge->current_size = (void *) (values + 1);
- agp_bridge->aperture_size_idx = 1;
- return values[1].size;
- } else {
- agp_bridge->current_size = (void *) (values);
- agp_bridge->aperture_size_idx = 0;
- return values[0].size;
- }
- return 0;
- }
- static int intel_i810_configure(void)
- {
- struct aper_size_info_fixed *current_size;
- u32 temp;
- int i;
- current_size = A_SIZE_FIX(agp_bridge->current_size);
- if (!intel_private.registers) {
- pci_read_config_dword(intel_private.pcidev, I810_MMADDR, &temp);
- temp &= 0xfff80000;
- intel_private.registers = ioremap(temp, 128 * 4096);
-