/* * VME Bridge Framework * * Author: Martyn Welch <martyn.welch@gefanuc.com> * Copyright 2008 GE Fanuc Intelligent Platforms Embedded Systems, Inc. * * Based on work by Tom Armistead and Ajit Prem * Copyright 2004 Motorola Inc. * * This program is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License as published by the * Free Software Foundation; either version 2 of the License, or (at your * option) any later version. */ #include <linux/module.h> #include <linux/moduleparam.h> #include <linux/mm.h> #include <linux/types.h> #include <linux/kernel.h> #include <linux/errno.h> #include <linux/pci.h> #include <linux/poll.h> #include <linux/highmem.h> #include <linux/interrupt.h> #include <linux/pagemap.h> #include <linux/device.h> #include <linux/dma-mapping.h> #include <linux/syscalls.h> #include <linux/mutex.h> #include <linux/spinlock.h> #include "vme.h" #include "vme_bridge.h" /* Bitmask and mutex to keep track of bridge numbers */ static unsigned int vme_bus_numbers; DEFINE_MUTEX(vme_bus_num_mtx); static void __exit vme_exit (void); static int __init vme_init (void); /* * Find the bridge resource associated with a specific device resource */ static struct vme_bridge *dev_to_bridge(struct device *dev) { return dev->platform_data; } /* * Find the bridge that the resource is associated with. */ static struct vme_bridge *find_bridge(struct vme_resource *resource) { /* Get list to search */ switch (resource->type) { case VME_MASTER: return list_entry(resource->entry, struct vme_master_resource, list)->parent; break; case VME_SLAVE: return list_entry(resource->entry, struct vme_slave_resource, list)->parent; break; case VME_DMA: return list_entry(resource->entry, struct vme_dma_resource, list)->parent; break; case VME_LM: return list_entry(resource->entry, struct vme_lm_resource, list)->parent; break; default: printk(KERN_ERR "Unknown resource type\n"); return NULL; break; } } /* * Allocate a contiguous block of memory for use by the driver. This is used to * create the buffers for the slave windows. * * XXX VME bridges could be available on buses other than PCI. At the momment * this framework only supports PCI devices. */ void * vme_alloc_consistent(struct vme_resource *resource, size_t size, dma_addr_t *dma) { struct vme_bridge *bridge; struct pci_dev *pdev; if(resource == NULL) { printk("No resource\n"); return NULL; } bridge = find_bridge(resource); if(bridge == NULL) { printk("Can't find bridge\n"); return NULL; } /* Find pci_dev container of dev */ if (bridge->parent == NULL) { printk("Dev entry NULL\n"); return NULL; } pdev = container_of(bridge->parent, struct pci_dev, dev); return pci_alloc_consistent(pdev, size, dma); } EXPORT_SYMBOL(vme_alloc_consistent); /* * Free previously allocated contiguous block of memory. * * XXX VME bridges could be available on buses other than PCI. At the momment * this framework only supports PCI devices. */ void vme_free_consistent(struct vme_resource *resource, size_t size, void *vaddr, dma_addr_t dma) { struct vme_bridge *bridge; struct pci_dev *pdev; if(resource == NULL) { printk("No resource\n"); return; } bridge = find_bridge(resource); if(bridge == NULL) { printk("Can't find bridge\n"); return; } /* Find pci_dev container of dev */ pdev = container_of(bridge->parent, struct pci_dev, dev); pci_free_consistent(pdev, size, vaddr, dma); } EXPORT_SYMBOL(vme_free_consistent); size_t vme_get_size(struct vme_resource *resource) { int enabled, retval; unsigned long long base, size; dma_addr_t buf_base; vme_address_t aspace; vme_cycle_t cycle; vme_width_t dwidth; switch (resource->type) { case VME_MASTER: retval = vme_master_get(resource, &enabled, &base, &size, &aspace, &cycle, &dwidth); return size; break; case VME_SLAVE: retval = vme_slave_get(resource, &enabled, &base, &size, &buf_base, &aspace, &cycle); return size; break; case VME_DMA: return 0; break; default: printk(KERN_ERR "Unknown resource type\n"); return 0; break; } } EXPORT_SYMBOL(vme_get_size); static int vme_check_window(vme_address_t aspace, unsigned long long vme_base, unsigned long long size) { int retval = 0; switch (aspace) { case VME_A16: if (((vme_base + size) > VME_A16_MAX) || (vme_base > VME_A16_MAX)) retval = -EFAULT; break; case VME_A24: if (((vme_base + size) > VME_A24_MAX) || (vme_base > VME_A24_MAX)) retval = -EFAULT; break; case VME_A32: if (((vme_base + size) > VME_A32_MAX) || (vme_base > VME_A32_MAX)) retval = -EFAULT; break; case VME_A64: /* * Any value held in an unsigned long long can be used as the * base */ break; case VME_CRCSR: if (((vme_base + size) > VME_CRCSR_MAX) || (vme_base > VME_CRCSR_MAX)) retval = -EFAULT; break; case VME_USER1: case VME_USER2: case VME_USER3: case VME_USER4: /* User Defined */ break; default: printk("Invalid address space\n"); retval = -EINVAL; break; } return retval; } /* * Request a slave image with specific attributes, return some unique * identifier. */ struct vme_resource * vme_slave_request(struct device *dev, vme_address_t address, vme_cycle_t cycle) { struct vme_bridge *bridge; struct list_head *slave_pos = NULL; struct vme_slave_resource *allocated_image = NULL; struct vme_slave_resource *slave_image = NULL; struct vme_resource *resource = NULL; bridge = dev_to_bridge(dev); if (bridge == NULL) { printk(KERN_ERR "Can't find VME bus\n"); goto err_bus; } /* Loop through slave resources */ list_for_each(slave_pos, &(bridge->slave_resources)) { slave_image = list_entry(slave_pos, struct vme_slave_resource, list); if (slave_image == NULL) { printk("Registered NULL Slave resource\n"); continue; } /* Find an unlocked and compatible image */ mutex_lock(&(slave_image->mtx)); if(((slave_image->address_attr & address) == address) && ((slave_image->cycle_attr & cycle) == cycle) && (slave_image->locked == 0)) { slave_image->locked = 1; mutex_unlock(&(slave_image->mtx)); allocated_image = slave_image; break; } mutex_unlock(&(slave_image->mtx)); } /* No free image */ if (allocated_image == NULL) goto err_image; resource = kmalloc(sizeof(struct vme_resource), GFP_KERNEL); if (resource == NULL) { printk(KERN_WARNING "Unable to allocate resource structure\n"); goto err_alloc; } resource->type = VME_SLAVE; resource->entry = &(allocated_image->list); return resource; err_alloc: /* Unlock image */ mutex_lock(&(slave_image->mtx)); slave_image->locked = 0; mutex_unlock(&(slave_image->mtx)); err_image: err_bus: return NULL; } EXPORT_SYMBOL(vme_slave_request); int vme_slave_set (struct vme_resource *resource, int enabled, unsigned long long vme_base, unsigned long long size, dma_addr_t buf_base, vme_address_t aspace, vme_cycle_t cycle) { struct vme_bridge *bridge = find_bridge(resource); struct vme_slave_resource *image; int retval; if (resource->type != VME_SLAVE) { printk("Not a slave resource\n"); return -EINVAL; } image = list_entry(resource->entry, struct vme_slave_resource, list); if (bridge->slave_set == NULL) { printk("Function not supported\n"); return -ENOSYS; } if(!(((image->address_attr & aspace) == aspace) && ((image->cycle_attr & cycle) == cycle))) { printk("Invalid attributes\n"); return -EINVAL; } retval = vme_check_window(aspace, vme_base, size); if(retval) return retval; return bridge->slave_set(image, enabled, vme_base, size, buf_base, aspace, cycle); } EXPORT_SYMBOL(vme_slave_set); int vme_slave_get (struct vme_resource *resource, int *enabled, unsigned long long *vme_base, unsigned long long *size, dma_addr_t *buf_base, vme_address_t *aspace, vme_cycle_t *cycle) { struct vme_bridge *bridge = find_bridge(resource); struct vme_slave_resource *image; if (resource->type != VME_SLAVE) { printk("Not a slave resource\n"); return -EINVAL; } image = list_entry(resource->entry, struct vme_slave_resource, list); if (bridge->slave_get == NULL) { printk("vme_slave_get not supported\n"); return -EINVAL; } return bridge->slave_get(image, enabled, vme_base, size, buf_base, aspace, cycle); } EXPORT_SYMBOL(vme_slave_get); void vme_slave_free(struct vme_resource *resource) { struct vme_slave_resource *slave_image; if (resource->type != VME_SLAVE) { printk("Not a slave resource\n"); return; } slave_image = list_entry(resource->entry, struct vme_slave_resource, list); if (slave_image == NULL) { printk("Can't find slave resource\n"); return; } /* Unlock image */ mutex_lock(&(slave_image->mtx)); if (slave_image->locked == 0) printk(KERN_ERR "Image is already free\n"); slave_image->locked = 0; mutex_unlock(&(slave_image->mtx)); /* Free up resource memory */ kfree(resource); } EXPORT_SYMBOL(vme_slave_free); /* * Request a master image with specific attributes, return some unique * identifier. */ struct vme_resource * vme_master_request(struct device *dev, vme_address_t address, vme_cycle_t cycle, vme_width_t dwidth) { struct vme_bridge *bridge; struct list_head *master_pos = NULL; struct vme_master_resource *allocated_image = NULL; struct vme_master_resource *master_image = NULL; struct vme_resource *resource = NULL; bridge = dev_to_bridge(dev); if (bridge == NULL) { printk(KERN_ERR "Can't find VME bus\n"); goto err_bus; } /* Loop through master resources */ list_for_each(master_pos, &(bridge->master_resources)) { master_image = list_entry(master_pos, struct vme_master_resource, list); if (master_image == NULL) { printk(KERN_WARNING "Registered NULL master resource\n"); continue; } /* Find an unlocked and compatible image */ spin_lock(&(master_image->lock)); if(((master_image->address_attr & address) == address) && ((master_image->cycle_attr & cycle) == cycle) && ((master_image->width_attr & dwidth) == dwidth) && (master_image->locked == 0)) { master_image->locked = 1; spin_unlock(&(master_image->lock)); allocated_image = master_image; break; } spin_unlock(&(master_image->lock)); } /* Check to see if we found a resource */ if (allocated_image == NULL) { printk(KERN_ERR "Can't find a suitable resource\n"); goto err_image; } resource = kmalloc(sizeof(struct vme_resource), GFP_KERNEL); if (resource == NULL) { printk(KERN_ERR "Unable to allocate resource structure\n"); goto err_alloc; } resource->type = VME_MASTER; resource->entry = &(allocated_image->list); return resource; kfree(resource); err_alloc: /* Unlock image */ spin_lock(&(master_image->lock)); master_image->locked = 0; spin_unlock(&(master_image->lock)); err_image: err_bus: return NULL; } EXPORT_SYMBOL(vme_master_request); int vme_master_set (struct vme_resource *resource, int enabled, unsigned long long vme_base, unsigned long long size, vme_address_t aspace, vme_cycle_t cycle, vme_width_t dwidth) { struct vme_bridge *bridge = find_bridge(resource); struct vme_master_resource *image; int retval; if (resource->type != VME_MASTER) { printk("Not a master resource\n"); return -EINVAL; } image = list_entry(resource->entry, struct vme_master_resource, list); if (bridge->master_set == NULL) { printk("vme_master_set not supported\n"); return -EINVAL; } if(!(((image->address_attr & aspace) == aspace) && ((image->cycle_attr & cycle) == cycle) && ((image->width_attr & dwidth) == dwidth))) { printk("Invalid attributes\n"); return -EINVAL; } retval = vme_check_window(aspace, vme_base, size); if(retval) return retval; return bridge->master_set(image, enabled, vme_base, size, aspace, cycle, dwidth); } EXPORT_SYMBOL(vme_master_set); int vme_master_get (struct vme_resource *resource, int *enabled, unsigned long long *vme_base, unsigned long long *size, vme_address_t *aspace, vme_cycle_t *cycle, vme_width_t *dwidth) { struct vme_bridge *bridge = find_bridge(resource); struct vme_master_resource *image; if (resource->type != VME_MASTER) { printk("Not a master resource\n"); return -EINVAL; } image = list_entry(resource->entry, struct vme_master_resource, list); if (bridge->master_get == NULL) { printk("vme_master_set not supported\n"); return -EINVAL; } return bridge->master_get(image, enabled, vme_base, size, aspace, cycle, dwidth); } EXPORT_SYMBOL(vme_master_get); /* * Read data out of VME space into a buffer. */ ssize_t vme_master_read (struct vme_resource *resource, void *buf, size_t count, loff_t offset) { struct vme_bridge *bridge = find_bridge(resource); struct vme_master_resource *image; size_t length; if (bridge->master_read == NULL) { printk("Reading from resource not supported\n"); return -EINVAL; } if (resource->type != VME_MASTER) { printk("Not a master resource\n"); return -EINVAL; } image = list_entry(resource->entry, struct vme_master_resource, list); length = vme_get_size(resource); if (offset > length) { printk("Invalid Offset\n"); return -EFAULT; } if ((offset + count) > length) count = length - offset; return bridge->master_read(image, buf, count, offset); } EXPORT_SYMBOL(vme_master_read); /* * Write data out to VME space from a buffer. */ ssize_t vme_master_write (struct vme_resource *resource, void *buf, size_t count, loff_t offset) { struct vme_bridge *bridge = find_bridge(resource); struct vme_master_resource *image; size_t length; if (bridge->master_write == NULL) { printk("Writing to resource not supported\n"); return -EINVAL; } if (resource->type != VME_MASTER) { printk("Not a master resource\n"); return -EINVAL; } image = list_entry(resource->entry, struct vme_master_resource, list); length = vme_get_size(resource); if (offset > length) { printk("Invalid Offset\n"); return -EFAULT; } if ((offset + count) > length) count = length - offset; return bridge->master_write(image, buf, count, offset); } EXPORT_SYMBOL(vme_master_write); /* * Perform RMW cycle to provided location. */ unsigned int vme_master_rmw (struct vme_resource *resource, unsigned int mask, unsigned int compare, unsigned int swap, loff_t offset) { struct vme_bridge *bridge = find_bridge(resource); struct vme_master_resource *image; if (bridge->master_rmw == NULL) { printk("Writing to resource not supported\n"); return -EINVAL; } if (resource->type != VME_MASTER) { printk("Not a master resource\n"); return -EINVAL; } image = list_entry(resource->entry, struct vme_master_resource, list); return bridge->master_rmw(image, mask, compare, swap, offset); } EXPORT_SYMBOL(vme_master_rmw); void vme_master_free(struct vme_resource *resource) { struct vme_master_resource *master_image; if (resource->type != VME_MASTER) { printk("Not a master resource\n"); return; } master_image = list_entry(resource->entry, struct vme_master_resource, list); if (master_image == NULL) { printk("Can't find master resource\n"); return; } /* Unlock image */ spin_lock(&(master_image->lock)); if (master_image->locked == 0) printk(KERN_ERR "Image is already free\n"); master_image->locked = 0; spin_unlock(&(master_image->lock)); /* Free up resource memory */ kfree(resource); } EXPORT_SYMBOL(vme_master_free); /* * Request a DMA controller with specific attributes, return some unique * identifier. */ struct vme_resource *vme_dma_request(struct device *dev) { struct vme_bridge *bridge; struct list_head *dma_pos = NULL; struct vme_dma_resource *allocated_ctrlr = NULL; struct vme_dma_resource *dma_ctrlr = NULL; struct vme_resource *resource = NULL; /* XXX Not checking resource attributes */ printk(KERN_ERR "No VME resource Attribute tests done\n"); bridge = dev_to_bridge(dev); if (bridge == NULL) { printk(KERN_ERR "Can't find VME bus\n"); goto err_bus; } /* Loop through DMA resources */ list_for_each(dma_pos, &(bridge->dma_resources)) { dma_ctrlr = list_entry(dma_pos, struct vme_dma_resource, list); if (dma_ctrlr == NULL) { printk("Registered NULL DMA resource\n"); continue; } /* Find an unlocked controller */ mutex_lock(&(dma_ctrlr->mtx)); if(dma_ctrlr->locked == 0) { dma_ctrlr->locked = 1; mutex_unlock(&(dma_ctrlr->mtx)); allocated_ctrlr = dma_ctrlr; break; } mutex_unlock(&(dma_ctrlr->mtx)); } /* Check to see if we found a resource */ if (allocated_ctrlr == NULL) goto err_ctrlr; resource = kmalloc(sizeof(struct vme_resource), GFP_KERNEL); if (resource == NULL) { printk(KERN_WARNING "Unable to allocate resource structure\n"); goto err_alloc; } resource->type = VME_DMA; resource->entry = &(allocated_ctrlr->list); return resource; err_alloc: /* Unlock image */ mutex_lock(&(dma_ctrlr->mtx)); dma_ctrlr->locked = 0; mutex_unlock(&(dma_ctrlr->mtx)); err_ctrlr: err_bus: return NULL; } EXPORT_SYMBOL(vme_dma_request); /* * Start new list */ struct vme_dma_list *vme_new_dma_list(struct vme_resource *resource) { struct vme_dma_resource *ctrlr; struct vme_dma_list *dma_list; if (resource->type != VME_DMA) { printk("Not a DMA resource\n"); return NULL; } ctrlr = list_entry(resource->entry, struct vme_dma_resource, list); dma_list = (struct vme_dma_list *)kmalloc( sizeof(struct vme_dma_list), GFP_KERNEL); if(dma_list == NULL) { printk("Unable to allocate memory for new dma list\n"); return NULL; } INIT_LIST_HEAD(&(dma_list->entries)); dma_list->parent = ctrlr; mutex_init(&(dma_list->mtx)); return dma_list; } EXPORT_SYMBOL(vme_new_dma_list); /* * Create "Pattern" type attributes */ struct vme_dma_attr *vme_dma_pattern_attribute(u32 pattern, vme_pattern_t type) { struct vme_dma_attr *attributes; struct vme_dma_pattern *pattern_attr; attributes = (struct vme_dma_attr *)kmalloc( sizeof(struct vme_dma_attr), GFP_KERNEL); if(attributes == NULL) { printk("Unable to allocate memory for attributes structure\n"); goto err_attr; } pattern_attr = (struct vme_dma_pattern *)kmalloc( sizeof(struct vme_dma_pattern), GFP_KERNEL); if(pattern_attr == NULL) { printk("Unable to allocate memory for pattern attributes\n"); goto err_pat; } attributes->type = VME_DMA_PATTERN; attributes->private = (void *)pattern_attr; pattern_attr->pattern = pattern; pattern_attr->type = type; return attributes; kfree(pattern_attr); err_pat: kfree(attributes); err_attr: return NULL; } EXPORT_SYMBOL(vme_dma_pattern_attribute); /* * Create "PCI" type attributes */ struct vme_dma_attr *vme_dma_pci_attribute(dma_addr_t address) { struct vme_dma_attr *attributes; struct vme_dma_pci *pci_attr; /* XXX Run some sanity checks here */ attributes = (struct vme_dma_attr *)kmalloc( sizeof(struct vme_dma_attr), GFP_KERNEL); if(attributes == NULL) { printk("Unable to allocate memory for attributes structure\n"); goto err_attr; } pci_attr = (struct vme_dma_pci *)kmalloc(sizeof(struct vme_dma_pci), GFP_KERNEL); if(pci_attr == NULL) { printk("Unable to allocate memory for pci attributes\n"); goto err_pci; } attributes->type = VME_DMA_PCI; attributes->private = (void *)pci_attr; pci_attr->address = address; return attributes; kfree(pci_attr); err_pci: kfree(attributes); err_attr: return NULL; } EXPORT_SYMBOL(vme_dma_pci_attribute); /* * Create "VME" type attributes */ struct vme_dma_attr *vme_dma_vme_attribute(unsigned long long address, vme_address_t aspace, vme_cycle_t cycle, vme_width_t dwidth) { struct vme_dma_attr *attributes; struct vme_dma_vme *vme_attr; /* XXX Run some sanity checks here */ attributes = (struct vme_dma_attr *)kmalloc( sizeof(struct vme_dma_attr), GFP_KERNEL); if(attributes == NULL) { printk("Unable to allocate memory for attributes structure\n"); goto err_attr; } vme_attr = (struct vme_dma_vme *)kmalloc(sizeof(struct vme_dma_vme), GFP_KERNEL); if(vme_attr == NULL) { printk("Unable to allocate memory for vme attributes\n"); goto err_vme; } attributes->type = VME_DMA_VME; attributes->private = (void *)vme_attr; vme_attr->address = address; vme_attr->aspace = aspace; vme_attr->cycle = cycle; vme_attr->dwidth = dwidth; return attributes; kfree(vme_attr); err_vme: kfree(attributes); err_attr: return NULL; } EXPORT_SYMBOL(vme_dma_vme_attribute); /* * Free attribute */ void vme_dma_free_attribute(struct vme_dma_attr *attributes) { kfree(attributes->private); kfree(attributes); } EXPORT_SYMBOL(vme_dma_free_attribute); int vme_dma_list_add(struct vme_dma_list *list, struct vme_dma_attr *src, struct vme_dma_attr *dest, size_t count) { struct vme_bridge *bridge = list->parent->parent; int retval; if (bridge->dma_list_add == NULL) { printk("Link List DMA generation not supported\n"); return -EINVAL; } if (!mutex_trylock(&(list->mtx))) { printk("Link List already submitted\n"); return -EINVAL; } retval = bridge->dma_list_add(list, src, dest, count); mutex_unlock(&(list->mtx)); return retval; } EXPORT_SYMBOL(vme_dma_list_add); int vme_dma_list_exec(struct vme_dma_list *list) { struct vme_bridge *bridge = list->parent->parent; int retval; if (bridge->dma_list_exec == NULL) { printk("Link List DMA execution not supported\n"); return -EINVAL; } mutex_lock(&(list->mtx)); retval = bridge->dma_list_exec(list); mutex_unlock(&(list->mtx)); return retval; } EXPORT_SYMBOL(vme_dma_list_exec); int vme_dma_list_free(struct vme_dma_list *list) { struct vme_bridge *bridge = list->parent->parent; int retval; if (bridge->dma_list_empty == NULL) { printk("Emptying of Link Lists not supported\n"); return -EINVAL; } if (!mutex_trylock(&(list->mtx))) { printk("Link List in use\n"); return -EINVAL; } /* * Empty out all of the entries from the dma list. We need to go to the * low level driver as dma entries are driver specific. */ retval = bridge->dma_list_empty(list); if (retval) { printk("Unable to empty link-list entries\n"); mutex_unlock(&(list->mtx)); return retval; } mutex_unlock(&(list->mtx)); kfree(list); return retval; } EXPORT_SYMBOL(vme_dma_list_free); int vme_dma_free(struct vme_resource *resource) { struct vme_dma_resource *ctrlr; if (resource->type != VME_DMA) { printk("Not a DMA resource\n"); return -EINVAL; } ctrlr = list_entry(resource->entry, struct vme_dma_resource, list); if (!mutex_trylock(&(ctrlr->mtx))) { printk("Resource busy, can't free\n"); return -EBUSY; } if (!(list_empty(&(ctrlr->pending)) && list_empty(&(ctrlr->running)))) { printk("Resource still processing transfers\n"); mutex_unlock(&(ctrlr->mtx)); return -EBUSY; } ctrlr->locked = 0; mutex_unlock(&(ctrlr->mtx)); return 0; } EXPORT_SYMBOL(vme_dma_free); void vme_irq_handler(struct vme_bridge *bridge, int level, int statid) { void (*call)(int, int, void *); void *priv_data; call = bridge->irq[level - 1].callback[statid].func; priv_data = bridge->irq[level - 1].callback[statid].priv_data; if (call != NULL) call(level, statid, priv_data); else printk(KERN_WARNING "Spurilous VME interrupt, level:%x, " "vector:%x\n", level, statid); } EXPORT_SYMBOL(vme_irq_handler); int vme_irq_request(struct device *dev, int level, int statid, void (*callback)(int level, int vector, void *priv_data), void *priv_data) { struct vme_bridge *bridge; bridge = dev_to_bridge(dev); if (bridge == NULL) { printk(KERN_ERR "Can't find VME bus\n"); return -EINVAL; } if((level < 1) || (level > 7)) { printk(KERN_ERR "Invalid interrupt level\n"); return -EINVAL; } if (bridge->irq_set == NULL) { printk(KERN_ERR "Configuring interrupts not supported\n"); return -EINVAL; } mutex_lock(&(bridge->irq_mtx)); if (bridge->irq[level - 1].callback[statid].func) { mutex_unlock(&(bridge->irq_mtx)); printk(KERN_WARNING "VME Interrupt already taken\n"); return -EBUSY; } bridge->irq[level - 1].count++; bridge->irq[level - 1].callback[statid].priv_data = priv_data; bridge->irq[level - 1].callback[statid].func = callback; /* Enable IRQ level */ bridge->irq_set(level, 1, 1); mutex_unlock(&(bridge->irq_mtx)); return 0; } EXPORT_SYMBOL(vme_irq_request); void vme_irq_free(struct device *dev, int level, int statid) { struct vme_bridge *bridge; bridge = dev_to_bridge(dev); if (bridge == NULL) { printk(KERN_ERR "Can't find VME bus\n"); return; } if((level < 1) || (level > 7)) { printk(KERN_ERR "Invalid interrupt level\n"); return; } if (bridge->irq_set == NULL) { printk(KERN_ERR "Configuring interrupts not supported\n"); return; } mutex_lock(&(bridge->irq_mtx)); bridge->irq[level - 1].count--; /* Disable IRQ level if no more interrupts attached at this level*/ if (bridge->irq[level - 1].count == 0) bridge->irq_set(level, 0, 1); bridge->irq[level - 1].callback[statid].func = NULL; bridge->irq[level - 1].callback[statid].priv_data = NULL; mutex_unlock(&(bridge->irq_mtx)); } EXPORT_SYMBOL(vme_irq_free); int vme_irq_generate(struct device *dev, int level, int statid) { struct vme_bridge *bridge; bridge = dev_to_bridge(dev); if (bridge == NULL) { printk(KERN_ERR "Can't find VME bus\n"); return -EINVAL; } if((level < 1) || (level > 7)) { printk(KERN_WARNING "Invalid interrupt level\n"); return -EINVAL; } if (bridge->irq_generate == NULL) { printk("Interrupt generation not supported\n"); return -EINVAL; } return bridge->irq_generate(level, statid); } EXPORT_SYMBOL(vme_irq_generate); /* * Request the location monitor, return resource or NULL */ struct vme_resource *vme_lm_request(struct device *dev) { struct vme_bridge *bridge; struct list_head *lm_pos = NULL; struct vme_lm_resource *allocated_lm = NULL; struct vme_lm_resource *lm = NULL; struct vme_resource *resource = NULL; bridge = dev_to_bridge(dev); if (bridge == NULL) { printk(KERN_ERR "Can't find VME bus\n"); goto err_bus; } /* Loop through DMA resources */ list_for_each(lm_pos, &(bridge->lm_resources)) { lm = list_entry(lm_pos, struct vme_lm_resource, list); if (lm == NULL) { printk(KERN_ERR "Registered NULL Location Monitor " "resource\n"); continue; } /* Find an unlocked controller */ mutex_lock(&(lm->mtx)); if (lm->locked == 0) { lm->locked = 1; mutex_unlock(&(lm->mtx)); allocated_lm = lm; break; } mutex_unlock(&(lm->mtx)); } /* Check to see if we found a resource */ if (allocated_lm == NULL) goto err_lm; resource = kmalloc(sizeof(struct vme_resource), GFP_KERNEL); if (resource == NULL) { printk(KERN_ERR "Unable to allocate resource structure\n"); goto err_alloc; } resource->type = VME_LM; resource->entry = &(allocated_lm->list); return resource; err_alloc: /* Unlock image */ mutex_lock(&(lm->mtx)); lm->locked = 0; mutex_unlock(&(lm->mtx)); err_lm: err_bus: return NULL; } EXPORT_SYMBOL(vme_lm_request); int vme_lm_count(struct vme_resource *resource) { struct vme_lm_resource *lm; if (resource->type != VME_LM) { printk(KERN_ERR "Not a Location Monitor resource\n"); return -EINVAL; } lm = list_entry(resource->entry, struct vme_lm_resource, list); return lm->monitors; } EXPORT_SYMBOL(vme_lm_count); int vme_lm_set(struct vme_resource *resource, unsigned long long lm_base, vme_address_t aspace, vme_cycle_t cycle) { struct vme_bridge *bridge = find_bridge(resource); struct vme_lm_resource *lm; if (resource->type != VME_LM) { printk(KERN_ERR "Not a Location Monitor resource\n"); return -EINVAL; } lm = list_entry(resource->entry, struct vme_lm_resource, list); if (bridge->lm_set == NULL) { printk(KERN_ERR "vme_lm_set not supported\n"); return -EINVAL; } /* XXX Check parameters */ return bridge->lm_set(lm, lm_base, aspace, cycle); } EXPORT_SYMBOL(vme_lm_set); int vme_lm_get(struct vme_resource *resource, unsigned long long *lm_base, vme_address_t *aspace, vme_cycle_t *cycle) { struct vme_bridge *bridge = find_bridge(resource); struct vme_lm_resource *lm; if (resource->type != VME_LM) { printk(KERN_ERR "Not a Location Monitor resource\n"); return -EINVAL; } lm = list_entry(resource->entry, struct vme_lm_resource, list); if (bridge->lm_get == NULL) { printk(KERN_ERR "vme_lm_get not supported\n"); return -EINVAL; } return bridge->lm_get(lm, lm_base, aspace, cycle); } EXPORT_SYMBOL(vme_lm_get); int vme_lm_attach(struct vme_resource *resource, int monitor, void (*callback)(int)) { struct vme_bridge *bridge = find_bridge(resource); struct vme_lm_resource *lm; if (resource->type != VME_LM) { printk(KERN_ERR "Not a Location Monitor resource\n"); return -EINVAL; } lm = list_entry(resource->entry, struct vme_lm_resource, list); if (bridge->lm_attach == NULL) { printk(KERN_ERR "vme_lm_attach not supported\n"); return -EINVAL; } return bridge->lm_attach(lm, monitor, callback); } EXPORT_SYMBOL(vme_lm_attach); int vme_lm_detach(struct vme_resource *resource, int monitor) { struct vme_bridge *bridge = find_bridge(resource); struct vme_lm_resource *lm; if (resource->type != VME_LM) { printk(KERN_ERR "Not a Location Monitor resource\n"); return -EINVAL; } lm = list_entry(resource->entry, struct vme_lm_resource, list); if (bridge->lm_detach == NULL) { printk(KERN_ERR "vme_lm_detach not supported\n"); return -EINVAL; } return bridge->lm_detach(lm, monitor); } EXPORT_SYMBOL(vme_lm_detach); void vme_lm_free(struct vme_resource *resource) { struct vme_lm_resource *lm; if (resource->type != VME_LM) { printk(KERN_ERR "Not a Location Monitor resource\n"); return; } lm = list_entry(resource->entry, struct vme_lm_resource, list); mutex_lock(&(lm->mtx)); /* XXX * Check to see that there aren't any callbacks still attached, if * there are we should probably be detaching them! */ lm->locked = 0; mutex_unlock(&(lm->mtx)); kfree(resource); } EXPORT_SYMBOL(vme_lm_free); int vme_slot_get(struct device *bus) { struct vme_bridge *bridge; bridge = dev_to_bridge(bus); if (bridge == NULL) { printk(KERN_ERR "Can't find VME bus\n"); return -EINVAL; } if (bridge->slot_get == NULL) { printk("vme_slot_get not supported\n"); return -EINVAL; } return bridge->slot_get(); } EXPORT_SYMBOL(vme_slot_get); /* - Bridge Registration --------------------------------------------------- */ static int vme_alloc_bus_num(void) { int i; mutex_lock(&vme_bus_num_mtx); for (i = 0; i < sizeof(vme_bus_numbers) * 8; i++) { if (((vme_bus_numbers >> i) & 0x1) == 0) { vme_bus_numbers |= (0x1 << i); break; } } mutex_unlock(&vme_bus_num_mtx); return i; } static void vme_free_bus_num(int bus) { mutex_lock(&vme_bus_num_mtx); vme_bus_numbers |= ~(0x1 << bus); mutex_unlock(&vme_bus_num_mtx); } int vme_register_bridge (struct vme_bridge *bridge) { struct device *dev; int retval; int i; bridge->num = vme_alloc_bus_num(); /* This creates 32 vme "slot" devices. This equates to a slot for each * ID available in a system conforming to the ANSI/VITA 1-1994 * specification. */ for (i = 0; i < VME_SLOTS_MAX; i++) { dev = &(bridge->dev[i]); memset(dev, 0, sizeof(struct device)); dev->parent = bridge->parent; dev->bus = &(vme_bus_type); /* * We save a pointer to the bridge in platform_data so that we * can get to it later. We keep driver_data for use by the * driver that binds against the slot */ dev->platform_data = bridge; dev_set_name(dev, "vme-%x.%x", bridge->num, i + 1); retval = device_register(dev); if(retval) goto err_reg; } return retval; i = VME_SLOTS_MAX; err_reg: while (i > -1) { dev = &(bridge->dev[i]); device_unregister(dev); } vme_free_bus_num(bridge->num); return retval; } EXPORT_SYMBOL(vme_register_bridge); void vme_unregister_bridge (struct vme_bridge *bridge) { int i; struct device *dev; for (i = 0; i < VME_SLOTS_MAX; i++) { dev = &(bridge->dev[i]); device_unregister(dev); } vme_free_bus_num(bridge->num); } EXPORT_SYMBOL(vme_unregister_bridge); /* - Driver Registration --------------------------------------------------- */ int vme_register_driver (struct vme_driver *drv) { drv->driver.name = drv->name; drv->driver.bus = &vme_bus_type; return driver_register(&drv->driver); } EXPORT_SYMBOL(vme_register_driver); void vme_unregister_driver (struct vme_driver *drv) { driver_unregister(&drv->driver); } EXPORT_SYMBOL(vme_unregister_driver); /* - Bus Registration ------------------------------------------------------ */ int vme_calc_slot(struct device *dev) { struct vme_bridge *bridge; int num; bridge = dev_to_bridge(dev); /* Determine slot number */ num = 0; while(num < VME_SLOTS_MAX) { if(&(bridge->dev[num]) == dev) { break; } num++; } if (num == VME_SLOTS_MAX) { dev_err(dev, "Failed to identify slot\n"); num = 0; goto err_dev; } num++; err_dev: return num; } static struct vme_driver *dev_to_vme_driver(struct device *dev) { if(dev->driver == NULL) printk("Bugger dev->driver is NULL\n"); return container_of(dev->driver, struct vme_driver, driver); } static int vme_bus_match(struct device *dev, struct device_driver *drv) { struct vme_bridge *bridge; struct vme_driver *driver; int i, num; bridge = dev_to_bridge(dev); driver = container_of(drv, struct vme_driver, driver); num = vme_calc_slot(dev); if (!num) goto err_dev; if (driver->bind_table == NULL) { dev_err(dev, "Bind table NULL\n"); goto err_table; } i = 0; while((driver->bind_table[i].bus != 0) || (driver->bind_table[i].slot != 0)) { if (bridge->num == driver->bind_table[i].bus) { if (num == driver->bind_table[i].slot) return 1; if (driver->bind_table[i].slot == VME_SLOT_ALL) return 1; if ((driver->bind_table[i].slot == VME_SLOT_CURRENT) && (num == vme_slot_get(dev))) return 1; } i++; } err_dev: err_table: return 0; } static int vme_bus_probe(struct device *dev) { struct vme_bridge *bridge; struct vme_driver *driver; int retval = -ENODEV; driver = dev_to_vme_driver(dev); bridge = dev_to_bridge(dev); if(driver->probe != NULL) { retval = driver->probe(dev, bridge->num, vme_calc_slot(dev)); } return retval; } static int vme_bus_remove(struct device *dev) { struct vme_bridge *bridge; struct vme_driver *driver; int retval = -ENODEV; driver = dev_to_vme_driver(dev); bridge = dev_to_bridge(dev); if(driver->remove != NULL) { retval = driver->remove(dev, bridge->num, vme_calc_slot(dev)); } return retval; } struct bus_type vme_bus_type = { .name = "vme", .match = vme_bus_match, .probe = vme_bus_probe, .remove = vme_bus_remove, }; EXPORT_SYMBOL(vme_bus_type); static int __init vme_init (void) { return bus_register(&vme_bus_type); } static void __exit vme_exit (void) { bus_unregister(&vme_bus_type); } MODULE_DESCRIPTION("VME bridge driver framework"); MODULE_AUTHOR("Martyn Welch <martyn.welch@gefanuc.com"); MODULE_LICENSE("GPL"); module_init(vme_init); module_exit(vme_exit);