| #include <linux/module.h> | 
| #include <linux/pci.h> | 
| #include <linux/init.h> | 
| #include <linux/agp_backend.h> | 
| #include <linux/mm.h> | 
| #include <linux/slab.h> | 
|   | 
| #include <asm/machvec.h> | 
| #include <asm/agp_backend.h> | 
| #include "../../../arch/alpha/kernel/pci_impl.h" | 
|   | 
| #include "agp.h" | 
|   | 
| static vm_fault_t alpha_core_agp_vm_fault(struct vm_fault *vmf) | 
| { | 
|     alpha_agp_info *agp = agp_bridge->dev_private_data; | 
|     dma_addr_t dma_addr; | 
|     unsigned long pa; | 
|     struct page *page; | 
|   | 
|     dma_addr = vmf->address - vmf->vma->vm_start + agp->aperture.bus_base; | 
|     pa = agp->ops->translate(agp, dma_addr); | 
|   | 
|     if (pa == (unsigned long)-EINVAL) | 
|         return VM_FAULT_SIGBUS;    /* no translation */ | 
|   | 
|     /* | 
|      * Get the page, inc the use count, and return it | 
|      */ | 
|     page = virt_to_page(__va(pa)); | 
|     get_page(page); | 
|     vmf->page = page; | 
|     return 0; | 
| } | 
|   | 
| static struct aper_size_info_fixed alpha_core_agp_sizes[] = | 
| { | 
|     { 0, 0, 0 }, /* filled in by alpha_core_agp_setup */ | 
| }; | 
|   | 
| static const struct vm_operations_struct alpha_core_agp_vm_ops = { | 
|     .fault = alpha_core_agp_vm_fault, | 
| }; | 
|   | 
|   | 
| static int alpha_core_agp_fetch_size(void) | 
| { | 
|     return alpha_core_agp_sizes[0].size; | 
| } | 
|   | 
| static int alpha_core_agp_configure(void) | 
| { | 
|     alpha_agp_info *agp = agp_bridge->dev_private_data; | 
|     agp_bridge->gart_bus_addr = agp->aperture.bus_base; | 
|     return 0; | 
| } | 
|   | 
| static void alpha_core_agp_cleanup(void) | 
| { | 
|     alpha_agp_info *agp = agp_bridge->dev_private_data; | 
|   | 
|     agp->ops->cleanup(agp); | 
| } | 
|   | 
| static void alpha_core_agp_tlbflush(struct agp_memory *mem) | 
| { | 
|     alpha_agp_info *agp = agp_bridge->dev_private_data; | 
|     alpha_mv.mv_pci_tbi(agp->hose, 0, -1); | 
| } | 
|   | 
| static void alpha_core_agp_enable(struct agp_bridge_data *bridge, u32 mode) | 
| { | 
|     alpha_agp_info *agp = bridge->dev_private_data; | 
|   | 
|     agp->mode.lw = agp_collect_device_status(bridge, mode, | 
|                     agp->capability.lw); | 
|   | 
|     agp->mode.bits.enable = 1; | 
|     agp->ops->configure(agp); | 
|   | 
|     agp_device_command(agp->mode.lw, false); | 
| } | 
|   | 
| static int alpha_core_agp_insert_memory(struct agp_memory *mem, off_t pg_start, | 
|                     int type) | 
| { | 
|     alpha_agp_info *agp = agp_bridge->dev_private_data; | 
|     int num_entries, status; | 
|     void *temp; | 
|   | 
|     if (type >= AGP_USER_TYPES || mem->type >= AGP_USER_TYPES) | 
|         return -EINVAL; | 
|   | 
|     temp = agp_bridge->current_size; | 
|     num_entries = A_SIZE_FIX(temp)->num_entries; | 
|     if ((pg_start + mem->page_count) > num_entries) | 
|         return -EINVAL; | 
|   | 
|     status = agp->ops->bind(agp, pg_start, mem); | 
|     mb(); | 
|     alpha_core_agp_tlbflush(mem); | 
|   | 
|     return status; | 
| } | 
|   | 
| static int alpha_core_agp_remove_memory(struct agp_memory *mem, off_t pg_start, | 
|                     int type) | 
| { | 
|     alpha_agp_info *agp = agp_bridge->dev_private_data; | 
|     int status; | 
|   | 
|     status = agp->ops->unbind(agp, pg_start, mem); | 
|     alpha_core_agp_tlbflush(mem); | 
|     return status; | 
| } | 
|   | 
| static int alpha_core_agp_create_free_gatt_table(struct agp_bridge_data *a) | 
| { | 
|     return 0; | 
| } | 
|   | 
| struct agp_bridge_driver alpha_core_agp_driver = { | 
|     .owner            = THIS_MODULE, | 
|     .aperture_sizes        = alpha_core_agp_sizes, | 
|     .num_aperture_sizes    = 1, | 
|     .size_type        = FIXED_APER_SIZE, | 
|     .cant_use_aperture    = true, | 
|     .masks            = NULL, | 
|   | 
|     .fetch_size        = alpha_core_agp_fetch_size, | 
|     .configure        = alpha_core_agp_configure, | 
|     .agp_enable        = alpha_core_agp_enable, | 
|     .cleanup        = alpha_core_agp_cleanup, | 
|     .tlb_flush        = alpha_core_agp_tlbflush, | 
|     .mask_memory        = agp_generic_mask_memory, | 
|     .cache_flush        = global_cache_flush, | 
|     .create_gatt_table    = alpha_core_agp_create_free_gatt_table, | 
|     .free_gatt_table    = alpha_core_agp_create_free_gatt_table, | 
|     .insert_memory        = alpha_core_agp_insert_memory, | 
|     .remove_memory        = alpha_core_agp_remove_memory, | 
|     .alloc_by_type        = agp_generic_alloc_by_type, | 
|     .free_by_type        = agp_generic_free_by_type, | 
|     .agp_alloc_page        = agp_generic_alloc_page, | 
|     .agp_alloc_pages    = agp_generic_alloc_pages, | 
|     .agp_destroy_page    = agp_generic_destroy_page, | 
|     .agp_destroy_pages    = agp_generic_destroy_pages, | 
|     .agp_type_to_mask_type  = agp_generic_type_to_mask_type, | 
| }; | 
|   | 
| struct agp_bridge_data *alpha_bridge; | 
|   | 
| int __init | 
| alpha_core_agp_setup(void) | 
| { | 
|     alpha_agp_info *agp = alpha_mv.agp_info(); | 
|     struct pci_dev *pdev;    /* faked */ | 
|     struct aper_size_info_fixed *aper_size; | 
|   | 
|     if (!agp) | 
|         return -ENODEV; | 
|     if (agp->ops->setup(agp)) | 
|         return -ENODEV; | 
|   | 
|     /* | 
|      * Build the aperture size descriptor | 
|      */ | 
|     aper_size = alpha_core_agp_sizes; | 
|     aper_size->size = agp->aperture.size / (1024 * 1024); | 
|     aper_size->num_entries = agp->aperture.size / PAGE_SIZE; | 
|     aper_size->page_order = __ffs(aper_size->num_entries / 1024); | 
|   | 
|     /* | 
|      * Build a fake pci_dev struct | 
|      */ | 
|     pdev = pci_alloc_dev(NULL); | 
|     if (!pdev) | 
|         return -ENOMEM; | 
|     pdev->vendor = 0xffff; | 
|     pdev->device = 0xffff; | 
|     pdev->sysdata = agp->hose; | 
|   | 
|     alpha_bridge = agp_alloc_bridge(); | 
|     if (!alpha_bridge) | 
|         goto fail; | 
|   | 
|     alpha_bridge->driver = &alpha_core_agp_driver; | 
|     alpha_bridge->vm_ops = &alpha_core_agp_vm_ops; | 
|     alpha_bridge->current_size = aper_size; /* only 1 size */ | 
|     alpha_bridge->dev_private_data = agp; | 
|     alpha_bridge->dev = pdev; | 
|     alpha_bridge->mode = agp->capability.lw; | 
|   | 
|     printk(KERN_INFO PFX "Detected AGP on hose %d\n", agp->hose->index); | 
|     return agp_add_bridge(alpha_bridge); | 
|   | 
|  fail: | 
|     kfree(pdev); | 
|     return -ENOMEM; | 
| } | 
|   | 
| static int __init agp_alpha_core_init(void) | 
| { | 
|     if (agp_off) | 
|         return -EINVAL; | 
|     if (alpha_mv.agp_info) | 
|         return alpha_core_agp_setup(); | 
|     return -ENODEV; | 
| } | 
|   | 
| static void __exit agp_alpha_core_cleanup(void) | 
| { | 
|     agp_remove_bridge(alpha_bridge); | 
|     agp_put_bridge(alpha_bridge); | 
| } | 
|   | 
| module_init(agp_alpha_core_init); | 
| module_exit(agp_alpha_core_cleanup); | 
|   | 
| MODULE_AUTHOR("Jeff Wiedemeier <Jeff.Wiedemeier@hp.com>"); | 
| MODULE_LICENSE("GPL and additional rights"); |