1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16#include <linux/version.h>
17#include <linux/module.h>
18#include <linux/moduleparam.h>
19#include <linux/mm.h>
20#include <linux/types.h>
21#include <linux/kernel.h>
22#include <linux/errno.h>
23#include <linux/pci.h>
24#include <linux/poll.h>
25#include <linux/highmem.h>
26#include <linux/interrupt.h>
27#include <linux/pagemap.h>
28#include <linux/device.h>
29#include <linux/dma-mapping.h>
30#include <linux/syscalls.h>
31#include <linux/mutex.h>
32#include <linux/spinlock.h>
33
34#include "vme.h"
35#include "vme_bridge.h"
36
37
38static unsigned int vme_bus_numbers;
39DEFINE_MUTEX(vme_bus_num_mtx);
40
41static void __exit vme_exit (void);
42static int __init vme_init (void);
43
44
45
46
47
48static struct vme_bridge *dev_to_bridge(struct device *dev)
49{
50 return dev->platform_data;
51}
52
53
54
55
56static struct vme_bridge *find_bridge(struct vme_resource *resource)
57{
58
59 switch (resource->type) {
60 case VME_MASTER:
61 return list_entry(resource->entry, struct vme_master_resource,
62 list)->parent;
63 break;
64 case VME_SLAVE:
65 return list_entry(resource->entry, struct vme_slave_resource,
66 list)->parent;
67 break;
68 case VME_DMA:
69 return list_entry(resource->entry, struct vme_dma_resource,
70 list)->parent;
71 break;
72 case VME_LM:
73 return list_entry(resource->entry, struct vme_lm_resource,
74 list)->parent;
75 break;
76 default:
77 printk(KERN_ERR "Unknown resource type\n");
78 return NULL;
79 break;
80 }
81}
82
83
84
85
86
87
88
89
90void * vme_alloc_consistent(struct vme_resource *resource, size_t size,
91 dma_addr_t *dma)
92{
93 struct vme_bridge *bridge;
94 struct pci_dev *pdev;
95
96 if(resource == NULL) {
97 printk("No resource\n");
98 return NULL;
99 }
100
101 bridge = find_bridge(resource);
102 if(bridge == NULL) {
103 printk("Can't find bridge\n");
104 return NULL;
105 }
106
107
108 if (bridge->parent == NULL) {
109 printk("Dev entry NULL\n");
110 return NULL;
111 }
112 pdev = container_of(bridge->parent, struct pci_dev, dev);
113
114 return pci_alloc_consistent(pdev, size, dma);
115}
116EXPORT_SYMBOL(vme_alloc_consistent);
117
118
119
120
121
122
123
124void vme_free_consistent(struct vme_resource *resource, size_t size,
125 void *vaddr, dma_addr_t dma)
126{
127 struct vme_bridge *bridge;
128 struct pci_dev *pdev;
129
130 if(resource == NULL) {
131 printk("No resource\n");
132 return;
133 }
134
135 bridge = find_bridge(resource);
136 if(bridge == NULL) {
137 printk("Can't find bridge\n");
138 return;
139 }
140
141
142 pdev = container_of(bridge->parent, struct pci_dev, dev);
143
144 pci_free_consistent(pdev, size, vaddr, dma);
145}
146EXPORT_SYMBOL(vme_free_consistent);
147
148size_t vme_get_size(struct vme_resource *resource)
149{
150 int enabled, retval;
151 unsigned long long base, size;
152 dma_addr_t buf_base;
153 vme_address_t aspace;
154 vme_cycle_t cycle;
155 vme_width_t dwidth;
156
157 switch (resource->type) {
158 case VME_MASTER:
159 retval = vme_master_get(resource, &enabled, &base, &size,
160 &aspace, &cycle, &dwidth);
161
162 return size;
163 break;
164 case VME_SLAVE:
165 retval = vme_slave_get(resource, &enabled, &base, &size,
166 &buf_base, &aspace, &cycle);
167
168 return size;
169 break;
170 case VME_DMA:
171 return 0;
172 break;
173 default:
174 printk(KERN_ERR "Unknown resource type\n");
175 return 0;
176 break;
177 }
178}
179EXPORT_SYMBOL(vme_get_size);
180
181static int vme_check_window(vme_address_t aspace, unsigned long long vme_base,
182 unsigned long long size)
183{
184 int retval = 0;
185
186 switch (aspace) {
187 case VME_A16:
188 if (((vme_base + size) > VME_A16_MAX) ||
189 (vme_base > VME_A16_MAX))
190 retval = -EFAULT;
191 break;
192 case VME_A24:
193 if (((vme_base + size) > VME_A24_MAX) ||
194 (vme_base > VME_A24_MAX))
195 retval = -EFAULT;
196 break;
197 case VME_A32:
198 if (((vme_base + size) > VME_A32_MAX) ||
199 (vme_base > VME_A32_MAX))
200 retval = -EFAULT;
201 break;
202 case VME_A64:
203
204
205
206
207 break;
208 case VME_CRCSR:
209 if (((vme_base + size) > VME_CRCSR_MAX) ||
210 (vme_base > VME_CRCSR_MAX))
211 retval = -EFAULT;
212 break;
213 case VME_USER1:
214 case VME_USER2:
215 case VME_USER3:
216 case VME_USER4:
217
218 break;
219 default:
220 printk("Invalid address space\n");
221 retval = -EINVAL;
222 break;
223 }
224
225 return retval;
226}
227
228
229
230
231
232struct vme_resource * vme_slave_request(struct device *dev,
233 vme_address_t address, vme_cycle_t cycle)
234{
235 struct vme_bridge *bridge;
236 struct list_head *slave_pos = NULL;
237 struct vme_slave_resource *allocated_image = NULL;
238 struct vme_slave_resource *slave_image = NULL;
239 struct vme_resource *resource = NULL;
240
241 bridge = dev_to_bridge(dev);
242 if (bridge == NULL) {
243 printk(KERN_ERR "Can't find VME bus\n");
244 goto err_bus;
245 }
246
247
248 list_for_each(slave_pos, &(bridge->slave_resources)) {
249 slave_image = list_entry(slave_pos,
250 struct vme_slave_resource, list);
251
252 if (slave_image == NULL) {
253 printk("Registered NULL Slave resource\n");
254 continue;
255 }
256
257
258 mutex_lock(&(slave_image->mtx));
259 if(((slave_image->address_attr & address) == address) &&
260 ((slave_image->cycle_attr & cycle) == cycle) &&
261 (slave_image->locked == 0)) {
262
263 slave_image->locked = 1;
264 mutex_unlock(&(slave_image->mtx));
265 allocated_image = slave_image;
266 break;
267 }
268 mutex_unlock(&(slave_image->mtx));
269 }
270
271
272 if (allocated_image == NULL)
273 goto err_image;
274
275 resource = kmalloc(sizeof(struct vme_resource), GFP_KERNEL);
276 if (resource == NULL) {
277 printk(KERN_WARNING "Unable to allocate resource structure\n");
278 goto err_alloc;
279 }
280 resource->type = VME_SLAVE;
281 resource->entry = &(allocated_image->list);
282
283 return resource;
284
285err_alloc:
286
287 mutex_lock(&(slave_image->mtx));
288 slave_image->locked = 0;
289 mutex_unlock(&(slave_image->mtx));
290err_image:
291err_bus:
292 return NULL;
293}
294EXPORT_SYMBOL(vme_slave_request);
295
296int vme_slave_set (struct vme_resource *resource, int enabled,
297 unsigned long long vme_base, unsigned long long size,
298 dma_addr_t buf_base, vme_address_t aspace, vme_cycle_t cycle)
299{
300 struct vme_bridge *bridge = find_bridge(resource);
301 struct vme_slave_resource *image;
302 int retval;
303
304 if (resource->type != VME_SLAVE) {
305 printk("Not a slave resource\n");
306 return -EINVAL;
307 }
308
309 image = list_entry(resource->entry, struct vme_slave_resource, list);
310
311 if (bridge->slave_set == NULL) {
312 printk("Function not supported\n");
313 return -ENOSYS;
314 }
315
316 if(!(((image->address_attr & aspace) == aspace) &&
317 ((image->cycle_attr & cycle) == cycle))) {
318 printk("Invalid attributes\n");
319 return -EINVAL;
320 }
321
322 retval = vme_check_window(aspace, vme_base, size);
323 if(retval)
324 return retval;
325
326 return bridge->slave_set(image, enabled, vme_base, size, buf_base,
327 aspace, cycle);
328}
329EXPORT_SYMBOL(vme_slave_set);
330
331int vme_slave_get (struct vme_resource *resource, int *enabled,
332 unsigned long long *vme_base, unsigned long long *size,
333 dma_addr_t *buf_base, vme_address_t *aspace, vme_cycle_t *cycle)
334{
335 struct vme_bridge *bridge = find_bridge(resource);
336 struct vme_slave_resource *image;
337
338 if (resource->type != VME_SLAVE) {
339 printk("Not a slave resource\n");
340 return -EINVAL;
341 }
342
343 image = list_entry(resource->entry, struct vme_slave_resource, list);
344
345 if (bridge->slave_get == NULL) {
346 printk("vme_slave_get not supported\n");
347 return -EINVAL;
348 }
349
350 return bridge->slave_get(image, enabled, vme_base, size, buf_base,
351 aspace, cycle);
352}
353EXPORT_SYMBOL(vme_slave_get);
354
355void vme_slave_free(struct vme_resource *resource)
356{
357 struct vme_slave_resource *slave_image;
358
359 if (resource->type != VME_SLAVE) {
360 printk("Not a slave resource\n");
361 return;
362 }
363
364 slave_image = list_entry(resource->entry, struct vme_slave_resource,
365 list);
366 if (slave_image == NULL) {
367 printk("Can't find slave resource\n");
368 return;
369 }
370
371
372 mutex_lock(&(slave_image->mtx));
373 if (slave_image->locked == 0)
374 printk(KERN_ERR "Image is already free\n");
375
376 slave_image->locked = 0;
377 mutex_unlock(&(slave_image->mtx));
378
379
380 kfree(resource);
381}
382EXPORT_SYMBOL(vme_slave_free);
383
384
385
386
387
388struct vme_resource * vme_master_request(struct device *dev,
389 vme_address_t address, vme_cycle_t cycle, vme_width_t dwidth)
390{
391 struct vme_bridge *bridge;
392 struct list_head *master_pos = NULL;
393 struct vme_master_resource *allocated_image = NULL;
394 struct vme_master_resource *master_image = NULL;
395 struct vme_resource *resource = NULL;
396
397 bridge = dev_to_bridge(dev);
398 if (bridge == NULL) {
399 printk(KERN_ERR "Can't find VME bus\n");
400 goto err_bus;
401 }
402
403
404 list_for_each(master_pos, &(bridge->master_resources)) {
405 master_image = list_entry(master_pos,
406 struct vme_master_resource, list);
407
408 if (master_image == NULL) {
409 printk(KERN_WARNING "Registered NULL master resource\n");
410 continue;
411 }
412
413
414 spin_lock(&(master_image->lock));
415 if(((master_image->address_attr & address) == address) &&
416 ((master_image->cycle_attr & cycle) == cycle) &&
417 ((master_image->width_attr & dwidth) == dwidth) &&
418 (master_image->locked == 0)) {
419
420 master_image->locked = 1;
421 spin_unlock(&(master_image->lock));
422 allocated_image = master_image;
423 break;
424 }
425 spin_unlock(&(master_image->lock));
426 }
427
428
429 if (allocated_image == NULL) {
430 printk(KERN_ERR "Can't find a suitable resource\n");
431 goto err_image;
432 }
433
434 resource = kmalloc(sizeof(struct vme_resource), GFP_KERNEL);
435 if (resource == NULL) {
436 printk(KERN_ERR "Unable to allocate resource structure\n");
437 goto err_alloc;
438 }
439 resource->type = VME_MASTER;
440 resource->entry = &(allocated_image->list);
441
442 return resource;
443
444 kfree(resource);
445err_alloc:
446
447 spin_lock(&(master_image->lock));
448 master_image->locked = 0;
449 spin_unlock(&(master_image->lock));
450err_image:
451err_bus:
452 return NULL;
453}
454EXPORT_SYMBOL(vme_master_request);
455
456int vme_master_set (struct vme_resource *resource, int enabled,
457 unsigned long long vme_base, unsigned long long size,
458 vme_address_t aspace, vme_cycle_t cycle, vme_width_t dwidth)
459{
460 struct vme_bridge *bridge = find_bridge(resource);
461 struct vme_master_resource *image;
462 int retval;
463
464 if (resource->type != VME_MASTER) {
465 printk("Not a master resource\n");
466 return -EINVAL;
467 }
468
469 image = list_entry(resource->entry, struct vme_master_resource, list);
470
471 if (bridge->master_set == NULL) {
472 printk("vme_master_set not supported\n");
473 return -EINVAL;
474 }
475
476 if(!(((image->address_attr & aspace) == aspace) &&
477 ((image->cycle_attr & cycle) == cycle) &&
478 ((image->width_attr & dwidth) == dwidth))) {
479 printk("Invalid attributes\n");
480 return -EINVAL;
481 }
482
483 retval = vme_check_window(aspace, vme_base, size);
484 if(retval)
485 return retval;
486
487 return bridge->master_set(image, enabled, vme_base, size, aspace,
488 cycle, dwidth);
489}
490EXPORT_SYMBOL(vme_master_set);
491
492int vme_master_get (struct vme_resource *resource, int *enabled,
493 unsigned long long *vme_base, unsigned long long *size,
494 vme_address_t *aspace, vme_cycle_t *cycle, vme_width_t *dwidth)
495{
496 struct vme_bridge *bridge = find_bridge(resource);
497 struct vme_master_resource *image;
498
499 if (resource->type != VME_MASTER) {
500 printk("Not a master resource\n");
501 return -EINVAL;
502 }
503
504 image = list_entry(resource->entry, struct vme_master_resource, list);
505
506 if (bridge->master_get == NULL) {
507 printk("vme_master_set not supported\n");
508 return -EINVAL;
509 }
510
511 return bridge->master_get(image, enabled, vme_base, size, aspace,
512 cycle, dwidth);
513}
514EXPORT_SYMBOL(vme_master_get);
515
516
517
518
519ssize_t vme_master_read (struct vme_resource *resource, void *buf, size_t count,
520 loff_t offset)
521{
522 struct vme_bridge *bridge = find_bridge(resource);
523 struct vme_master_resource *image;
524 size_t length;
525
526 if (bridge->master_read == NULL) {
527 printk("Reading from resource not supported\n");
528 return -EINVAL;
529 }
530
531 if (resource->type != VME_MASTER) {
532 printk("Not a master resource\n");
533 return -EINVAL;
534 }
535
536 image = list_entry(resource->entry, struct vme_master_resource, list);
537
538 length = vme_get_size(resource);
539
540 if (offset > length) {
541 printk("Invalid Offset\n");
542 return -EFAULT;
543 }
544
545 if ((offset + count) > length)
546 count = length - offset;
547
548 return bridge->master_read(image, buf, count, offset);
549
550}
551EXPORT_SYMBOL(vme_master_read);
552
553
554
555
556ssize_t vme_master_write (struct vme_resource *resource, void *buf,
557 size_t count, loff_t offset)
558{
559 struct vme_bridge *bridge = find_bridge(resource);
560 struct vme_master_resource *image;
561 size_t length;
562
563 if (bridge->master_write == NULL) {
564 printk("Writing to resource not supported\n");
565 return -EINVAL;
566 }
567
568 if (resource->type != VME_MASTER) {
569 printk("Not a master resource\n");
570 return -EINVAL;
571 }
572
573 image = list_entry(resource->entry, struct vme_master_resource, list);
574
575 length = vme_get_size(resource);
576
577 if (offset > length) {
578 printk("Invalid Offset\n");
579 return -EFAULT;
580 }
581
582 if ((offset + count) > length)
583 count = length - offset;
584
585 return bridge->master_write(image, buf, count, offset);
586}
587EXPORT_SYMBOL(vme_master_write);
588
589
590
591
592unsigned int vme_master_rmw (struct vme_resource *resource, unsigned int mask,
593 unsigned int compare, unsigned int swap, loff_t offset)
594{
595 struct vme_bridge *bridge = find_bridge(resource);
596 struct vme_master_resource *image;
597
598 if (bridge->master_rmw == NULL) {
599 printk("Writing to resource not supported\n");
600 return -EINVAL;
601 }
602
603 if (resource->type != VME_MASTER) {
604 printk("Not a master resource\n");
605 return -EINVAL;
606 }
607
608 image = list_entry(resource->entry, struct vme_master_resource, list);
609
610 return bridge->master_rmw(image, mask, compare, swap, offset);
611}
612EXPORT_SYMBOL(vme_master_rmw);
613
614void vme_master_free(struct vme_resource *resource)
615{
616 struct vme_master_resource *master_image;
617
618 if (resource->type != VME_MASTER) {
619 printk("Not a master resource\n");
620 return;
621 }
622
623 master_image = list_entry(resource->entry, struct vme_master_resource,
624 list);
625 if (master_image == NULL) {
626 printk("Can't find master resource\n");
627 return;
628 }
629
630
631 spin_lock(&(master_image->lock));
632 if (master_image->locked == 0)
633 printk(KERN_ERR "Image is already free\n");
634
635 master_image->locked = 0;
636 spin_unlock(&(master_image->lock));
637
638
639 kfree(resource);
640}
641EXPORT_SYMBOL(vme_master_free);
642
643
644
645
646
647struct vme_resource *vme_request_dma(struct device *dev)
648{
649 struct vme_bridge *bridge;
650 struct list_head *dma_pos = NULL;
651 struct vme_dma_resource *allocated_ctrlr = NULL;
652 struct vme_dma_resource *dma_ctrlr = NULL;
653 struct vme_resource *resource = NULL;
654
655
656 printk(KERN_ERR "No VME resource Attribute tests done\n");
657
658 bridge = dev_to_bridge(dev);
659 if (bridge == NULL) {
660 printk(KERN_ERR "Can't find VME bus\n");
661 goto err_bus;
662 }
663
664
665 list_for_each(dma_pos, &(bridge->dma_resources)) {
666 dma_ctrlr = list_entry(dma_pos,
667 struct vme_dma_resource, list);
668
669 if (dma_ctrlr == NULL) {
670 printk("Registered NULL DMA resource\n");
671 continue;
672 }
673
674
675 mutex_lock(&(dma_ctrlr->mtx));
676 if(dma_ctrlr->locked == 0) {
677 dma_ctrlr->locked = 1;
678 mutex_unlock(&(dma_ctrlr->mtx));
679 allocated_ctrlr = dma_ctrlr;
680 break;
681 }
682 mutex_unlock(&(dma_ctrlr->mtx));
683 }
684
685
686 if (allocated_ctrlr == NULL)
687 goto err_ctrlr;
688
689 resource = kmalloc(sizeof(struct vme_resource), GFP_KERNEL);
690 if (resource == NULL) {
691 printk(KERN_WARNING "Unable to allocate resource structure\n");
692 goto err_alloc;
693 }
694 resource->type = VME_DMA;
695 resource->entry = &(allocated_ctrlr->list);
696
697 return resource;
698
699err_alloc:
700
701 mutex_lock(&(dma_ctrlr->mtx));
702 dma_ctrlr->locked = 0;
703 mutex_unlock(&(dma_ctrlr->mtx));
704err_ctrlr:
705err_bus:
706 return NULL;
707}
708EXPORT_SYMBOL(vme_request_dma);
709
710
711
712
713struct vme_dma_list *vme_new_dma_list(struct vme_resource *resource)
714{
715 struct vme_dma_resource *ctrlr;
716 struct vme_dma_list *dma_list;
717
718 if (resource->type != VME_DMA) {
719 printk("Not a DMA resource\n");
720 return NULL;
721 }
722
723 ctrlr = list_entry(resource->entry, struct vme_dma_resource, list);
724
725 dma_list = (struct vme_dma_list *)kmalloc(
726 sizeof(struct vme_dma_list), GFP_KERNEL);
727 if(dma_list == NULL) {
728 printk("Unable to allocate memory for new dma list\n");
729 return NULL;
730 }
731 INIT_LIST_HEAD(&(dma_list->entries));
732 dma_list->parent = ctrlr;
733 mutex_init(&(dma_list->mtx));
734
735 return dma_list;
736}
737EXPORT_SYMBOL(vme_new_dma_list);
738
739
740
741
742struct vme_dma_attr *vme_dma_pattern_attribute(u32 pattern,
743 vme_pattern_t type)
744{
745 struct vme_dma_attr *attributes;
746 struct vme_dma_pattern *pattern_attr;
747
748 attributes = (struct vme_dma_attr *)kmalloc(
749 sizeof(struct vme_dma_attr), GFP_KERNEL);
750 if(attributes == NULL) {
751 printk("Unable to allocate memory for attributes structure\n");
752 goto err_attr;
753 }
754
755 pattern_attr = (struct vme_dma_pattern *)kmalloc(
756 sizeof(struct vme_dma_pattern), GFP_KERNEL);
757 if(pattern_attr == NULL) {
758 printk("Unable to allocate memory for pattern attributes\n");
759 goto err_pat;
760 }
761
762 attributes->type = VME_DMA_PATTERN;
763 attributes->private = (void *)pattern_attr;
764
765 pattern_attr->pattern = pattern;
766 pattern_attr->type = type;
767
768 return attributes;
769
770 kfree(pattern_attr);
771err_pat:
772 kfree(attributes);
773err_attr:
774 return NULL;
775}
776EXPORT_SYMBOL(vme_dma_pattern_attribute);
777
778
779
780
781struct vme_dma_attr *vme_dma_pci_attribute(dma_addr_t address)
782{
783 struct vme_dma_attr *attributes;
784 struct vme_dma_pci *pci_attr;
785
786
787
788 attributes = (struct vme_dma_attr *)kmalloc(
789 sizeof(struct vme_dma_attr), GFP_KERNEL);
790 if(attributes == NULL) {
791 printk("Unable to allocate memory for attributes structure\n");
792 goto err_attr;
793 }
794
795 pci_attr = (struct vme_dma_pci *)kmalloc(sizeof(struct vme_dma_pci),
796 GFP_KERNEL);
797 if(pci_attr == NULL) {
798 printk("Unable to allocate memory for pci attributes\n");
799 goto err_pci;
800 }
801
802
803
804 attributes->type = VME_DMA_PCI;
805 attributes->private = (void *)pci_attr;
806
807 pci_attr->address = address;
808
809 return attributes;
810
811 kfree(pci_attr);
812err_pci:
813 kfree(attributes);
814err_attr:
815 return NULL;
816}
817EXPORT_SYMBOL(vme_dma_pci_attribute);
818
819
820
821
822struct vme_dma_attr *vme_dma_vme_attribute(unsigned long long address,
823 vme_address_t aspace, vme_cycle_t cycle, vme_width_t dwidth)
824{
825 struct vme_dma_attr *attributes;
826 struct vme_dma_vme *vme_attr;
827
828
829
830 attributes = (struct vme_dma_attr *)kmalloc(
831 sizeof(struct vme_dma_attr), GFP_KERNEL);
832 if(attributes == NULL) {
833 printk("Unable to allocate memory for attributes structure\n");
834 goto err_attr;
835 }
836
837 vme_attr = (struct vme_dma_vme *)kmalloc(sizeof(struct vme_dma_vme),
838 GFP_KERNEL);
839 if(vme_attr == NULL) {
840 printk("Unable to allocate memory for vme attributes\n");
841 goto err_vme;
842 }
843
844 attributes->type = VME_DMA_VME;
845 attributes->private = (void *)vme_attr;
846
847 vme_attr->address = address;
848 vme_attr->aspace = aspace;
849 vme_attr->cycle = cycle;
850 vme_attr->dwidth = dwidth;
851
852 return attributes;
853
854 kfree(vme_attr);
855err_vme:
856 kfree(attributes);
857err_attr:
858 return NULL;
859}
860EXPORT_SYMBOL(vme_dma_vme_attribute);
861
862
863
864
865void vme_dma_free_attribute(struct vme_dma_attr *attributes)
866{
867 kfree(attributes->private);
868 kfree(attributes);
869}
870EXPORT_SYMBOL(vme_dma_free_attribute);
871
872int vme_dma_list_add(struct vme_dma_list *list, struct vme_dma_attr *src,
873 struct vme_dma_attr *dest, size_t count)
874{
875 struct vme_bridge *bridge = list->parent->parent;
876 int retval;
877
878 if (bridge->dma_list_add == NULL) {
879 printk("Link List DMA generation not supported\n");
880 return -EINVAL;
881 }
882
883 if (mutex_trylock(&(list->mtx))) {
884 printk("Link List already submitted\n");
885 return -EINVAL;
886 }
887
888 retval = bridge->dma_list_add(list, src, dest, count);
889
890 mutex_unlock(&(list->mtx));
891
892 return retval;
893}
894EXPORT_SYMBOL(vme_dma_list_add);
895
896int vme_dma_list_exec(struct vme_dma_list *list)
897{
898 struct vme_bridge *bridge = list->parent->parent;
899 int retval;
900
901 if (bridge->dma_list_exec == NULL) {
902 printk("Link List DMA execution not supported\n");
903 return -EINVAL;
904 }
905
906 mutex_lock(&(list->mtx));
907
908 retval = bridge->dma_list_exec(list);
909
910 mutex_unlock(&(list->mtx));
911
912 return retval;
913}
914EXPORT_SYMBOL(vme_dma_list_exec);
915
916int vme_dma_list_free(struct vme_dma_list *list)
917{
918 struct vme_bridge *bridge = list->parent->parent;
919 int retval;
920
921 if (bridge->dma_list_empty == NULL) {
922 printk("Emptying of Link Lists not supported\n");
923 return -EINVAL;
924 }
925
926 if (mutex_trylock(&(list->mtx))) {
927 printk("Link List in use\n");
928 return -EINVAL;
929 }
930
931
932
933
934
935 retval = bridge->dma_list_empty(list);
936 if (retval) {
937 printk("Unable to empty link-list entries\n");
938 mutex_unlock(&(list->mtx));
939 return retval;
940 }
941 mutex_unlock(&(list->mtx));
942 kfree(list);
943
944 return retval;
945}
946EXPORT_SYMBOL(vme_dma_list_free);
947
948int vme_dma_free(struct vme_resource *resource)
949{
950 struct vme_dma_resource *ctrlr;
951
952 if (resource->type != VME_DMA) {
953 printk("Not a DMA resource\n");
954 return -EINVAL;
955 }
956
957 ctrlr = list_entry(resource->entry, struct vme_dma_resource, list);
958
959 if (mutex_trylock(&(ctrlr->mtx))) {
960 printk("Resource busy, can't free\n");
961 return -EBUSY;
962 }
963
964 if (!(list_empty(&(ctrlr->pending)) && list_empty(&(ctrlr->running)))) {
965 printk("Resource still processing transfers\n");
966 mutex_unlock(&(ctrlr->mtx));
967 return -EBUSY;
968 }
969
970 ctrlr->locked = 0;
971
972 mutex_unlock(&(ctrlr->mtx));
973
974 return 0;
975}
976EXPORT_SYMBOL(vme_dma_free);
977
978int vme_request_irq(struct device *dev, int level, int statid,
979 void (*callback)(int level, int vector, void *priv_data),
980 void *priv_data)
981{
982 struct vme_bridge *bridge;
983
984 bridge = dev_to_bridge(dev);
985 if (bridge == NULL) {
986 printk(KERN_ERR "Can't find VME bus\n");
987 return -EINVAL;
988 }
989
990 if((level < 1) || (level > 7)) {
991 printk(KERN_WARNING "Invalid interrupt level\n");
992 return -EINVAL;
993 }
994
995 if (bridge->request_irq == NULL) {
996 printk("Registering interrupts not supported\n");
997 return -EINVAL;
998 }
999
1000 return bridge->request_irq(level, statid, callback, priv_data);
1001}
1002EXPORT_SYMBOL(vme_request_irq);
1003
1004void vme_free_irq(struct device *dev, int level, int statid)
1005{
1006 struct vme_bridge *bridge;
1007
1008 bridge = dev_to_bridge(dev);
1009 if (bridge == NULL) {
1010 printk(KERN_ERR "Can't find VME bus\n");
1011 return;
1012 }
1013
1014 if((level < 1) || (level > 7)) {
1015 printk(KERN_WARNING "Invalid interrupt level\n");
1016 return;
1017 }
1018
1019 if (bridge->free_irq == NULL) {
1020 printk("Freeing interrupts not supported\n");
1021 return;
1022 }
1023
1024 bridge->free_irq(level, statid);
1025}
1026EXPORT_SYMBOL(vme_free_irq);
1027
1028int vme_generate_irq(struct device *dev, int level, int statid)
1029{
1030 struct vme_bridge *bridge;
1031
1032 bridge = dev_to_bridge(dev);
1033 if (bridge == NULL) {
1034 printk(KERN_ERR "Can't find VME bus\n");
1035 return -EINVAL;
1036 }
1037
1038 if((level < 1) || (level > 7)) {
1039 printk(KERN_WARNING "Invalid interrupt level\n");
1040 return -EINVAL;
1041 }
1042
1043 if (bridge->generate_irq == NULL) {
1044 printk("Interrupt generation not supported\n");
1045 return -EINVAL;
1046 }
1047
1048 return bridge->generate_irq(level, statid);
1049}
1050EXPORT_SYMBOL(vme_generate_irq);
1051
1052
1053
1054
1055struct vme_resource *vme_lm_request(struct device *dev)
1056{
1057 struct vme_bridge *bridge;
1058 struct list_head *lm_pos = NULL;
1059 struct vme_lm_resource *allocated_lm = NULL;
1060 struct vme_lm_resource *lm = NULL;
1061 struct vme_resource *resource = NULL;
1062
1063 bridge = dev_to_bridge(dev);
1064 if (bridge == NULL) {
1065 printk(KERN_ERR "Can't find VME bus\n");
1066 goto err_bus;
1067 }
1068
1069
1070 list_for_each(lm_pos, &(bridge->lm_resources)) {
1071 lm = list_entry(lm_pos,
1072 struct vme_lm_resource, list);
1073
1074 if (lm == NULL) {
1075 printk(KERN_ERR "Registered NULL Location Monitor "
1076 "resource\n");
1077 continue;
1078 }
1079
1080
1081 mutex_lock(&(lm->mtx));
1082 if (lm->locked == 0) {
1083 lm->locked = 1;
1084 mutex_unlock(&(lm->mtx));
1085 allocated_lm = lm;
1086 break;
1087 }
1088 mutex_unlock(&(lm->mtx));
1089 }
1090
1091
1092 if (allocated_lm == NULL)
1093 goto err_lm;
1094
1095 resource = kmalloc(sizeof(struct vme_resource), GFP_KERNEL);
1096 if (resource == NULL) {
1097 printk(KERN_ERR "Unable to allocate resource structure\n");
1098 goto err_alloc;
1099 }
1100 resource->type = VME_LM;
1101 resource->entry = &(allocated_lm->list);
1102
1103 return resource;
1104
1105err_alloc:
1106
1107 mutex_lock(&(lm->mtx));
1108 lm->locked = 0;
1109 mutex_unlock(&(lm->mtx));
1110err_lm:
1111err_bus:
1112 return NULL;
1113}
1114EXPORT_SYMBOL(vme_lm_request);
1115
1116int vme_lm_count(struct vme_resource *resource)
1117{
1118 struct vme_lm_resource *lm;
1119
1120 if (resource->type != VME_LM) {
1121 printk(KERN_ERR "Not a Location Monitor resource\n");
1122 return -EINVAL;
1123 }
1124
1125 lm = list_entry(resource->entry, struct vme_lm_resource, list);
1126
1127 return lm->monitors;
1128}
1129EXPORT_SYMBOL(vme_lm_count);
1130
1131int vme_lm_set(struct vme_resource *resource, unsigned long long lm_base,
1132 vme_address_t aspace, vme_cycle_t cycle)
1133{
1134 struct vme_bridge *bridge = find_bridge(resource);
1135 struct vme_lm_resource *lm;
1136
1137 if (resource->type != VME_LM) {
1138 printk(KERN_ERR "Not a Location Monitor resource\n");
1139 return -EINVAL;
1140 }
1141
1142 lm = list_entry(resource->entry, struct vme_lm_resource, list);
1143
1144 if (bridge->lm_set == NULL) {
1145 printk(KERN_ERR "vme_lm_set not supported\n");
1146 return -EINVAL;
1147 }
1148
1149
1150
1151 return lm->parent->lm_set(lm, lm_base, aspace, cycle);
1152}
1153EXPORT_SYMBOL(vme_lm_set);
1154
1155int vme_lm_get(struct vme_resource *resource, unsigned long long *lm_base,
1156 vme_address_t *aspace, vme_cycle_t *cycle)
1157{
1158 struct vme_bridge *bridge = find_bridge(resource);
1159 struct vme_lm_resource *lm;
1160
1161 if (resource->type != VME_LM) {
1162 printk(KERN_ERR "Not a Location Monitor resource\n");
1163 return -EINVAL;
1164 }
1165
1166 lm = list_entry(resource->entry, struct vme_lm_resource, list);
1167
1168 if (bridge->lm_get == NULL) {
1169 printk(KERN_ERR "vme_lm_get not supported\n");
1170 return -EINVAL;
1171 }
1172
1173 return bridge->lm_get(lm, lm_base, aspace, cycle);
1174}
1175EXPORT_SYMBOL(vme_lm_get);
1176
1177int vme_lm_attach(struct vme_resource *resource, int monitor,
1178 void (*callback)(int))
1179{
1180 struct vme_bridge *bridge = find_bridge(resource);
1181 struct vme_lm_resource *lm;
1182
1183 if (resource->type != VME_LM) {
1184 printk(KERN_ERR "Not a Location Monitor resource\n");
1185 return -EINVAL;
1186 }
1187
1188 lm = list_entry(resource->entry, struct vme_lm_resource, list);
1189
1190 if (bridge->lm_attach == NULL) {
1191 printk(KERN_ERR "vme_lm_attach not supported\n");
1192 return -EINVAL;
1193 }
1194
1195 return bridge->lm_attach(lm, monitor, callback);
1196}
1197EXPORT_SYMBOL(vme_lm_attach);
1198
1199int vme_lm_detach(struct vme_resource *resource, int monitor)
1200{
1201 struct vme_bridge *bridge = find_bridge(resource);
1202 struct vme_lm_resource *lm;
1203
1204 if (resource->type != VME_LM) {
1205 printk(KERN_ERR "Not a Location Monitor resource\n");
1206 return -EINVAL;
1207 }
1208
1209 lm = list_entry(resource->entry, struct vme_lm_resource, list);
1210
1211 if (bridge->lm_detach == NULL) {
1212 printk(KERN_ERR "vme_lm_detach not supported\n");
1213 return -EINVAL;
1214 }
1215
1216 return bridge->lm_detach(lm, monitor);
1217}
1218EXPORT_SYMBOL(vme_lm_detach);
1219
1220void vme_lm_free(struct vme_resource *resource)
1221{
1222 struct vme_lm_resource *lm;
1223
1224 if (resource->type != VME_LM) {
1225 printk(KERN_ERR "Not a Location Monitor resource\n");
1226 return;
1227 }
1228
1229 lm = list_entry(resource->entry, struct vme_lm_resource, list);
1230
1231 if (mutex_trylock(&(lm->mtx))) {
1232 printk(KERN_ERR "Resource busy, can't free\n");
1233 return;
1234 }
1235
1236
1237
1238 lm->locked = 0;
1239
1240 mutex_unlock(&(lm->mtx));
1241}
1242EXPORT_SYMBOL(vme_lm_free);
1243
1244int vme_slot_get(struct device *bus)
1245{
1246 struct vme_bridge *bridge;
1247
1248 bridge = dev_to_bridge(bus);
1249 if (bridge == NULL) {
1250 printk(KERN_ERR "Can't find VME bus\n");
1251 return -EINVAL;
1252 }
1253
1254 if (bridge->slot_get == NULL) {
1255 printk("vme_slot_get not supported\n");
1256 return -EINVAL;
1257 }
1258
1259 return bridge->slot_get();
1260}
1261EXPORT_SYMBOL(vme_slot_get);
1262
1263
1264
1265
1266static int vme_alloc_bus_num(void)
1267{
1268 int i;
1269
1270 mutex_lock(&vme_bus_num_mtx);
1271 for (i = 0; i < sizeof(vme_bus_numbers) * 8; i++) {
1272 if (((vme_bus_numbers >> i) & 0x1) == 0) {
1273 vme_bus_numbers |= (0x1 << i);
1274 break;
1275 }
1276 }
1277 mutex_unlock(&vme_bus_num_mtx);
1278
1279 return i;
1280}
1281
1282static void vme_free_bus_num(int bus)
1283{
1284 mutex_lock(&vme_bus_num_mtx);
1285 vme_bus_numbers |= ~(0x1 << bus);
1286 mutex_unlock(&vme_bus_num_mtx);
1287}
1288
1289int vme_register_bridge (struct vme_bridge *bridge)
1290{
1291 struct device *dev;
1292 int retval;
1293 int i;
1294
1295 bridge->num = vme_alloc_bus_num();
1296
1297
1298
1299
1300
1301 for (i = 0; i < VME_SLOTS_MAX; i++) {
1302 dev = &(bridge->dev[i]);
1303 memset(dev, 0, sizeof(struct device));
1304
1305 dev->parent = bridge->parent;
1306 dev->bus = &(vme_bus_type);
1307
1308
1309
1310
1311
1312 dev->platform_data = bridge;
1313 dev_set_name(dev, "vme-%x.%x", bridge->num, i + 1);
1314
1315 retval = device_register(dev);
1316 if(retval)
1317 goto err_reg;
1318 }
1319
1320 return retval;
1321
1322 i = VME_SLOTS_MAX;
1323err_reg:
1324 while (i > -1) {
1325 dev = &(bridge->dev[i]);
1326 device_unregister(dev);
1327 }
1328 vme_free_bus_num(bridge->num);
1329 return retval;
1330}
1331EXPORT_SYMBOL(vme_register_bridge);
1332
1333void vme_unregister_bridge (struct vme_bridge *bridge)
1334{
1335 int i;
1336 struct device *dev;
1337
1338
1339 for (i = 0; i < VME_SLOTS_MAX; i++) {
1340 dev = &(bridge->dev[i]);
1341 device_unregister(dev);
1342 }
1343 vme_free_bus_num(bridge->num);
1344}
1345EXPORT_SYMBOL(vme_unregister_bridge);
1346
1347
1348
1349
1350int vme_register_driver (struct vme_driver *drv)
1351{
1352 drv->driver.name = drv->name;
1353 drv->driver.bus = &vme_bus_type;
1354
1355 return driver_register(&drv->driver);
1356}
1357EXPORT_SYMBOL(vme_register_driver);
1358
1359void vme_unregister_driver (struct vme_driver *drv)
1360{
1361 driver_unregister(&drv->driver);
1362}
1363EXPORT_SYMBOL(vme_unregister_driver);
1364
1365
1366
1367int vme_calc_slot(struct device *dev)
1368{
1369 struct vme_bridge *bridge;
1370 int num;
1371
1372 bridge = dev_to_bridge(dev);
1373
1374
1375 num = 0;
1376 while(num < VME_SLOTS_MAX) {
1377 if(&(bridge->dev[num]) == dev) {
1378 break;
1379 }
1380 num++;
1381 }
1382 if (num == VME_SLOTS_MAX) {
1383 dev_err(dev, "Failed to identify slot\n");
1384 num = 0;
1385 goto err_dev;
1386 }
1387 num++;
1388
1389err_dev:
1390 return num;
1391}
1392
1393static struct vme_driver *dev_to_vme_driver(struct device *dev)
1394{
1395 if(dev->driver == NULL)
1396 printk("Bugger dev->driver is NULL\n");
1397
1398 return container_of(dev->driver, struct vme_driver, driver);
1399}
1400
1401static int vme_bus_match(struct device *dev, struct device_driver *drv)
1402{
1403 struct vme_bridge *bridge;
1404 struct vme_driver *driver;
1405 int i, num;
1406
1407 bridge = dev_to_bridge(dev);
1408 driver = container_of(drv, struct vme_driver, driver);
1409
1410 num = vme_calc_slot(dev);
1411 if (!num)
1412 goto err_dev;
1413
1414 if (driver->bind_table == NULL) {
1415 dev_err(dev, "Bind table NULL\n");
1416 goto err_table;
1417 }
1418
1419 i = 0;
1420 while((driver->bind_table[i].bus != 0) ||
1421 (driver->bind_table[i].slot != 0)) {
1422
1423 if (bridge->num == driver->bind_table[i].bus) {
1424 if (num == driver->bind_table[i].slot)
1425 return 1;
1426
1427 if (driver->bind_table[i].slot == VME_SLOT_ALL)
1428 return 1;
1429
1430 if ((driver->bind_table[i].slot == VME_SLOT_CURRENT) &&
1431 (num == vme_slot_get(dev)))
1432 return 1;
1433 }
1434 i++;
1435 }
1436
1437err_dev:
1438err_table:
1439 return 0;
1440}
1441
1442static int vme_bus_probe(struct device *dev)
1443{
1444 struct vme_bridge *bridge;
1445 struct vme_driver *driver;
1446 int retval = -ENODEV;
1447
1448 driver = dev_to_vme_driver(dev);
1449 bridge = dev_to_bridge(dev);
1450
1451 if(driver->probe != NULL) {
1452 retval = driver->probe(dev, bridge->num, vme_calc_slot(dev));
1453 }
1454
1455 return retval;
1456}
1457
1458static int vme_bus_remove(struct device *dev)
1459{
1460 struct vme_bridge *bridge;
1461 struct vme_driver *driver;
1462 int retval = -ENODEV;
1463
1464 driver = dev_to_vme_driver(dev);
1465 bridge = dev_to_bridge(dev);
1466
1467 if(driver->remove != NULL) {
1468 retval = driver->remove(dev, bridge->num, vme_calc_slot(dev));
1469 }
1470
1471 return retval;
1472}
1473
1474struct bus_type vme_bus_type = {
1475 .name = "vme",
1476 .match = vme_bus_match,
1477 .probe = vme_bus_probe,
1478 .remove = vme_bus_remove,
1479};
1480EXPORT_SYMBOL(vme_bus_type);
1481
1482static int __init vme_init (void)
1483{
1484 return bus_register(&vme_bus_type);
1485}
1486
1487static void __exit vme_exit (void)
1488{
1489 bus_unregister(&vme_bus_type);
1490}
1491
1492MODULE_DESCRIPTION("VME bridge driver framework");
1493MODULE_AUTHOR("Martyn Welch <martyn.welch@gefanuc.com");
1494MODULE_LICENSE("GPL");
1495
1496module_init(vme_init);
1497module_exit(vme_exit);
1498