1 /*****************************************************************************/
4 * comemlite.c -- PCI access code for embedded CO-MEM Lite PCI controller.
6 * (C) Copyright 1999-2003, Greg Ungerer (gerg@snapgear.com).
7 * (C) Copyright 2000, Lineo (www.lineo.com)
10 /*****************************************************************************/
12 #include <linux/kernel.h>
13 #include <linux/types.h>
14 #include <linux/pci.h>
15 #include <linux/ptrace.h>
16 #include <linux/spinlock.h>
17 #include <linux/interrupt.h>
18 #include <linux/sched.h>
19 #include <asm/coldfire.h>
20 #include <asm/mcfsim.h>
22 #include <asm/anchor.h>
28 /*****************************************************************************/
31 * Debug configuration defines. DEBUGRES sets debugging output for
32 * the resource allocation phase. DEBUGPCI traces on pcibios_ function
33 * calls, and DEBUGIO traces all accesses to devices on the PCI bus.
35 /*#define DEBUGRES 1*/
36 /*#define DEBUGPCI 1*/
39 /*****************************************************************************/
42 * PCI markers for bus present and active slots.
44 int pci_bus_is_present
= 0;
45 unsigned long pci_slotmask
= 0;
48 * We may or may not need to swap the bytes of PCI bus tranfers.
49 * The endianess is re-roder automatically by the CO-MEM, but it
50 * will get the wrong byte order for a pure data stream.
52 #define pci_byteswap 0
56 * Resource tracking. The CO-MEM part creates a virtual address
57 * space that all the PCI devices live in - it is not in any way
58 * directly mapped into the ColdFire address space. So we can
59 * really assign any resources we like to devices, as long as
60 * they do not clash with other PCI devices.
62 unsigned int pci_iobase
= PCIBIOS_MIN_IO
; /* Arbitrary start address */
63 unsigned int pci_membase
= PCIBIOS_MIN_MEM
; /* Arbitrary start address */
65 #define PCI_MINIO 0x100 /* 256 byte minimum I/O */
66 #define PCI_MINMEM 0x00010000 /* 64k minimum chunk */
69 * The CO-MEM's shared memory segment is visible inside the PCI
70 * memory address space. We need to keep track of the address that
71 * this is mapped at, to setup the bus masters pointers.
73 unsigned int pci_shmemaddr
;
75 /*****************************************************************************/
77 void pci_interrupt(int irq
, void *id
, struct pt_regs
*fp
);
79 /*****************************************************************************/
82 * Some platforms have custom ways of reseting the PCI bus.
85 void pci_resetbus(void)
91 printk(KERN_DEBUG
"pci_resetbus()\n");
94 *((volatile unsigned short *) (MCF_MBAR
+MCFSIM_PADDR
)) |= eLIA_PCIRESET
;
95 for (i
= 0; (i
< 1000); i
++) {
96 *((volatile unsigned short *) (MCF_MBAR
+ MCFSIM_PADAT
)) =
97 (ppdata
| eLIA_PCIRESET
);
101 *((volatile unsigned short *) (MCF_MBAR
+ MCFSIM_PADAT
)) = ppdata
;
105 /*****************************************************************************/
107 int pcibios_assign_resource_slot(int slot
)
109 volatile unsigned long *rp
;
110 volatile unsigned char *ip
;
111 unsigned int idsel
, addr
, val
, align
, i
;
115 printk(KERN_INFO
"pcibios_assign_resource_slot(slot=%x)\n", slot
);
118 rp
= (volatile unsigned long *) COMEM_BASE
;
119 idsel
= COMEM_DA_ADDR(0x1 << (slot
+ 16));
121 /* Try to assign resource to each BAR */
122 for (bar
= 0; (bar
< 6); bar
++) {
123 addr
= COMEM_PCIBUS
+ PCI_BASE_ADDRESS_0
+ (bar
* 4);
124 rp
[LREG(COMEM_DAHBASE
)] = COMEM_DA_CFGRD
| idsel
;
125 val
= rp
[LREG(addr
)];
127 printk(KERN_DEBUG
"-----------------------------------"
128 "-------------------------------------\n");
129 printk(KERN_DEBUG
"BAR[%d]: read=%08x ", bar
, val
);
132 rp
[LREG(COMEM_DAHBASE
)] = COMEM_DA_CFGWR
| idsel
;
133 rp
[LREG(addr
)] = 0xffffffff;
135 rp
[LREG(COMEM_DAHBASE
)] = COMEM_DA_CFGRD
| idsel
;
136 val
= rp
[LREG(addr
)];
138 printk(KERN_DEBUG
"write=%08x ", val
);
142 printk(KERN_DEBUG
"\n");
147 /* Determine space required by BAR */
148 /* FIXME: this should go backwords from 0x80000000... */
149 for (i
= 0; (i
< 32); i
++) {
150 if ((0x1 << i
) & (val
& 0xfffffffc))
155 printk(KERN_DEBUG
"size=%08x(%d)\n", (0x1 << i
), i
);
159 /* Assign a resource */
160 if (val
& PCI_BASE_ADDRESS_SPACE_IO
) {
164 printk(KERN_DEBUG
"BAR[%d]: IO size=%08x iobase=%08x\n",
169 val
= 0 | PCI_BASE_ADDRESS_SPACE_IO
;
171 printk(KERN_DEBUG
"BAR[%d]: too big for IO??\n", bar
);
174 /* Check for un-alignment */
175 if ((align
= pci_iobase
% i
))
176 pci_iobase
+= (i
- align
);
177 val
= pci_iobase
| PCI_BASE_ADDRESS_SPACE_IO
;
184 printk(KERN_DEBUG
"BAR[%d]: MEMORY size=%08x membase=%08x\n",
185 bar
, i
, pci_membase
);
187 /* Check for un-alignment */
188 if ((align
= pci_membase
% i
))
189 pci_membase
+= (i
- align
);
190 val
= pci_membase
| PCI_BASE_ADDRESS_SPACE_MEMORY
;
194 /* Write resource back into BAR register */
195 rp
[LREG(COMEM_DAHBASE
)] = COMEM_DA_CFGWR
| idsel
;
196 rp
[LREG(addr
)] = val
;
198 printk(KERN_DEBUG
"BAR[%d]: assigned bar=%08x\n", bar
, val
);
203 printk(KERN_DEBUG
"-----------------------------------"
204 "-------------------------------------\n");
207 /* Assign IRQ if one is wanted... */
208 ip
= (volatile unsigned char *) (COMEM_BASE
+ COMEM_PCIBUS
);
209 rp
[LREG(COMEM_DAHBASE
)] = COMEM_DA_CFGRD
| idsel
;
211 addr
= (PCI_INTERRUPT_PIN
& 0xfc) + (~PCI_INTERRUPT_PIN
& 0x03);
213 rp
[LREG(COMEM_DAHBASE
)] = COMEM_DA_CFGWR
| idsel
;
214 addr
= (PCI_INTERRUPT_LINE
& 0xfc)+(~PCI_INTERRUPT_LINE
& 0x03);
217 printk(KERN_DEBUG
"IRQ LINE=25\n");
224 /*****************************************************************************/
226 int pcibios_enable_slot(int slot
)
228 volatile unsigned long *rp
;
229 volatile unsigned short *wp
;
230 unsigned int idsel
, addr
;
234 printk(KERN_DEBUG
"pcibios_enbale_slot(slot=%x)\n", slot
);
237 rp
= (volatile unsigned long *) COMEM_BASE
;
238 wp
= (volatile unsigned short *) COMEM_BASE
;
239 idsel
= COMEM_DA_ADDR(0x1 << (slot
+ 16));
241 /* Get current command settings */
242 addr
= COMEM_PCIBUS
+ PCI_COMMAND
;
243 addr
= (addr
& ~0x3) + (~addr
& 0x02);
244 rp
[LREG(COMEM_DAHBASE
)] = COMEM_DA_CFGRD
| idsel
;
245 cmd
= wp
[WREG(addr
)];
246 /*val = ((val & 0xff) << 8) | ((val >> 8) & 0xff);*/
248 /* Enable I/O and memory accesses to this device */
249 rp
[LREG(COMEM_DAHBASE
)] = COMEM_DA_CFGWR
| idsel
;
250 cmd
|= PCI_COMMAND_IO
| PCI_COMMAND_MEMORY
| PCI_COMMAND_MASTER
;
251 wp
[WREG(addr
)] = cmd
;
256 /*****************************************************************************/
258 void pcibios_assign_resources(void)
260 volatile unsigned long *rp
;
261 unsigned long sel
, id
;
264 rp
= (volatile unsigned long *) COMEM_BASE
;
267 * Do a quick scan of the PCI bus and see what is here.
269 for (slot
= COMEM_MINDEV
; (slot
<= COMEM_MAXDEV
); slot
++) {
270 sel
= COMEM_DA_CFGRD
| COMEM_DA_ADDR(0x1 << (slot
+ 16));
271 rp
[LREG(COMEM_DAHBASE
)] = sel
;
272 rp
[LREG(COMEM_PCIBUS
)] = 0; /* Clear bus */
273 id
= rp
[LREG(COMEM_PCIBUS
)];
274 if ((id
!= 0) && ((id
& 0xffff0000) != (sel
& 0xffff0000))) {
275 printk(KERN_INFO
"PCI: slot=%d id=%08x\n", slot
, (int) id
);
276 pci_slotmask
|= 0x1 << slot
;
277 pcibios_assign_resource_slot(slot
);
278 pcibios_enable_slot(slot
);
283 /*****************************************************************************/
285 int pcibios_init(void)
287 volatile unsigned long *rp
;
288 unsigned long sel
, id
;
292 printk(KERN_DEBUG
"pcibios_init()\n");
298 * Do some sort of basic check to see if the CO-MEM part
299 * is present... This works ok, but I think we really need
300 * something better...
302 rp
= (volatile unsigned long *) COMEM_BASE
;
303 if ((rp
[LREG(COMEM_LBUSCFG
)] & 0xff) != 0x50) {
304 printk(KERN_INFO
"PCI: no PCI bus present\n");
308 #ifdef COMEM_BRIDGEDEV
310 * Setup the PCI bridge device first. It needs resources too,
311 * so that bus masters can get to its shared memory.
313 slot
= COMEM_BRIDGEDEV
;
314 sel
= COMEM_DA_CFGRD
| COMEM_DA_ADDR(0x1 << (slot
+ 16));
315 rp
[LREG(COMEM_DAHBASE
)] = sel
;
316 rp
[LREG(COMEM_PCIBUS
)] = 0; /* Clear bus */
317 id
= rp
[LREG(COMEM_PCIBUS
)];
318 if ((id
== 0) || ((id
& 0xffff0000) == (sel
& 0xffff0000))) {
319 printk(KERN_INFO
"PCI: no PCI bus bridge present\n");
323 printk(KERN_INFO
"PCI: bridge device at slot=%d id=%08x\n", slot
, (int) id
);
324 pci_slotmask
|= 0x1 << slot
;
325 pci_shmemaddr
= pci_membase
;
326 pcibios_assign_resource_slot(slot
);
327 pcibios_enable_slot(slot
);
330 pci_bus_is_present
= 1;
332 /* Get PCI irq for local vectoring */
333 if (request_irq(COMEM_IRQ
, pci_interrupt
, 0, "PCI bridge", NULL
)) {
334 printk(KERN_WARNING
"PCI: failed to acquire interrupt %d\n", COMEM_IRQ
);
336 mcf_autovector(COMEM_IRQ
);
339 pcibios_assign_resources();
344 /*****************************************************************************/
346 char *pcibios_setup(char *option
)
348 /* Nothing for us to handle. */
351 /*****************************************************************************/
353 void pcibios_fixup_bus(struct pci_bus
*b
)
357 /*****************************************************************************/
359 void pcibios_align_resource(void *data
, struct resource
*res
,
360 resource_size_t size
, resource_size_t align
)
364 /*****************************************************************************/
366 int pcibios_enable_device(struct pci_dev
*dev
, int mask
)
370 slot
= PCI_SLOT(dev
->devfn
);
371 if ((dev
->bus
== 0) && (pci_slotmask
& (1 << slot
)))
372 pcibios_enable_slot(slot
);
376 /*****************************************************************************/
378 void pcibios_update_resource(struct pci_dev
*dev
, struct resource
*root
, struct resource
*r
, int resource
)
380 printk(KERN_WARNING
"%s(%d): no support for changing PCI resources...\n",
385 /*****************************************************************************/
388 * Local routines to interrcept the standard I/O and vector handling
389 * code. Don't include this 'till now - initialization code above needs
390 * access to the real code too.
392 #include <asm/mcfpci.h>
394 /*****************************************************************************/
396 void pci_outb(unsigned char val
, unsigned int addr
)
398 volatile unsigned long *rp
;
399 volatile unsigned char *bp
;
402 printk(KERN_DEBUG
"pci_outb(val=%02x,addr=%x)\n", val
, addr
);
405 rp
= (volatile unsigned long *) COMEM_BASE
;
406 bp
= (volatile unsigned char *) COMEM_BASE
;
407 rp
[LREG(COMEM_DAHBASE
)] = COMEM_DA_IOWR
| COMEM_DA_ADDR(addr
);
408 addr
= (addr
& ~0x3) + (~addr
& 0x03);
409 bp
[(COMEM_PCIBUS
+ COMEM_DA_OFFSET(addr
))] = val
;
412 /*****************************************************************************/
414 void pci_outw(unsigned short val
, unsigned int addr
)
416 volatile unsigned long *rp
;
417 volatile unsigned short *sp
;
420 printk(KERN_DEBUG
"pci_outw(val=%04x,addr=%x)\n", val
, addr
);
423 rp
= (volatile unsigned long *) COMEM_BASE
;
424 sp
= (volatile unsigned short *) COMEM_BASE
;
425 rp
[LREG(COMEM_DAHBASE
)] = COMEM_DA_IOWR
| COMEM_DA_ADDR(addr
);
426 addr
= (addr
& ~0x3) + (~addr
& 0x02);
428 val
= ((val
& 0xff) << 8) | ((val
>> 8) & 0xff);
429 sp
[WREG(COMEM_PCIBUS
+ COMEM_DA_OFFSET(addr
))] = val
;
432 /*****************************************************************************/
434 void pci_outl(unsigned int val
, unsigned int addr
)
436 volatile unsigned long *rp
;
437 volatile unsigned int *lp
;
440 printk(KERN_DEBUG
"pci_outl(val=%08x,addr=%x)\n", val
, addr
);
443 rp
= (volatile unsigned long *) COMEM_BASE
;
444 lp
= (volatile unsigned int *) COMEM_BASE
;
445 rp
[LREG(COMEM_DAHBASE
)] = COMEM_DA_IOWR
| COMEM_DA_ADDR(addr
);
448 val
= (val
<< 24) | ((val
& 0x0000ff00) << 8) |
449 ((val
& 0x00ff0000) >> 8) | (val
>> 24);
451 lp
[LREG(COMEM_PCIBUS
+ COMEM_DA_OFFSET(addr
))] = val
;
454 /*****************************************************************************/
456 unsigned long pci_blmask
[] = {
463 unsigned char pci_inb(unsigned int addr
)
465 volatile unsigned long *rp
;
466 volatile unsigned char *bp
;
471 printk(KERN_DEBUG
"pci_inb(addr=%x)\n", addr
);
474 rp
= (volatile unsigned long *) COMEM_BASE
;
475 bp
= (volatile unsigned char *) COMEM_BASE
;
477 r
= COMEM_DA_IORD
| COMEM_DA_ADDR(addr
) | pci_blmask
[(addr
& 0x3)];
478 rp
[LREG(COMEM_DAHBASE
)] = r
;
480 addr
= (addr
& ~0x3) + (~addr
& 0x3);
481 val
= bp
[(COMEM_PCIBUS
+ COMEM_DA_OFFSET(addr
))];
485 /*****************************************************************************/
487 unsigned long pci_bwmask
[] = {
494 unsigned short pci_inw(unsigned int addr
)
496 volatile unsigned long *rp
;
497 volatile unsigned short *sp
;
502 printk(KERN_DEBUG
"pci_inw(addr=%x)", addr
);
505 rp
= (volatile unsigned long *) COMEM_BASE
;
506 r
= COMEM_DA_IORD
| COMEM_DA_ADDR(addr
) | pci_bwmask
[(addr
& 0x3)];
507 rp
[LREG(COMEM_DAHBASE
)] = r
;
509 sp
= (volatile unsigned short *) COMEM_BASE
;
510 addr
= (addr
& ~0x3) + (~addr
& 0x02);
511 val
= sp
[WREG(COMEM_PCIBUS
+ COMEM_DA_OFFSET(addr
))];
513 val
= ((val
& 0xff) << 8) | ((val
>> 8) & 0xff);
515 printk(KERN_DEBUG
"=%04x\n", val
);
520 /*****************************************************************************/
522 unsigned int pci_inl(unsigned int addr
)
524 volatile unsigned long *rp
;
525 volatile unsigned int *lp
;
529 printk(KERN_DEBUG
"pci_inl(addr=%x)", addr
);
532 rp
= (volatile unsigned long *) COMEM_BASE
;
533 lp
= (volatile unsigned int *) COMEM_BASE
;
534 rp
[LREG(COMEM_DAHBASE
)] = COMEM_DA_IORD
| COMEM_DA_ADDR(addr
);
535 val
= lp
[LREG(COMEM_PCIBUS
+ COMEM_DA_OFFSET(addr
))];
538 val
= (val
<< 24) | ((val
& 0x0000ff00) << 8) |
539 ((val
& 0x00ff0000) >> 8) | (val
>> 24);
542 printk(KERN_DEBUG
"=%08x\n", val
);
547 /*****************************************************************************/
549 void pci_outsb(void *addr
, void *buf
, int len
)
551 volatile unsigned long *rp
;
552 volatile unsigned char *bp
;
553 unsigned char *dp
= (unsigned char *) buf
;
554 unsigned int a
= (unsigned int) addr
;
557 printk(KERN_DEBUG
"pci_outsb(addr=%x,buf=%x,len=%d)\n", (int)addr
, (int)buf
, len
);
560 rp
= (volatile unsigned long *) COMEM_BASE
;
561 rp
[LREG(COMEM_DAHBASE
)] = COMEM_DA_IOWR
| COMEM_DA_ADDR(a
);
563 a
= (a
& ~0x3) + (~a
& 0x03);
564 bp
= (volatile unsigned char *)
565 (COMEM_BASE
+ COMEM_PCIBUS
+ COMEM_DA_OFFSET(a
));
571 /*****************************************************************************/
573 void pci_outsw(void *addr
, void *buf
, int len
)
575 volatile unsigned long *rp
;
576 volatile unsigned short *wp
;
577 unsigned short w
, *dp
= (unsigned short *) buf
;
578 unsigned int a
= (unsigned int) addr
;
581 printk(KERN_DEBUG
"pci_outsw(addr=%x,buf=%x,len=%d)\n", (int)addr
, (int)buf
, len
);
584 rp
= (volatile unsigned long *) COMEM_BASE
;
585 rp
[LREG(COMEM_DAHBASE
)] = COMEM_DA_IOWR
| COMEM_DA_ADDR(a
);
587 a
= (a
& ~0x3) + (~a
& 0x2);
588 wp
= (volatile unsigned short *)
589 (COMEM_BASE
+ COMEM_PCIBUS
+ COMEM_DA_OFFSET(a
));
594 w
= ((w
& 0xff) << 8) | ((w
>> 8) & 0xff);
599 /*****************************************************************************/
601 void pci_outsl(void *addr
, void *buf
, int len
)
603 volatile unsigned long *rp
;
604 volatile unsigned long *lp
;
605 unsigned long l
, *dp
= (unsigned long *) buf
;
606 unsigned int a
= (unsigned int) addr
;
609 printk(KERN_DEBUG
"pci_outsl(addr=%x,buf=%x,len=%d)\n", (int)addr
, (int)buf
, len
);
612 rp
= (volatile unsigned long *) COMEM_BASE
;
613 rp
[LREG(COMEM_DAHBASE
)] = COMEM_DA_IOWR
| COMEM_DA_ADDR(a
);
615 lp
= (volatile unsigned long *)
616 (COMEM_BASE
+ COMEM_PCIBUS
+ COMEM_DA_OFFSET(a
));
621 l
= (l
<< 24) | ((l
& 0x0000ff00) << 8) |
622 ((l
& 0x00ff0000) >> 8) | (l
>> 24);
627 /*****************************************************************************/
629 void pci_insb(void *addr
, void *buf
, int len
)
631 volatile unsigned long *rp
;
632 volatile unsigned char *bp
;
633 unsigned char *dp
= (unsigned char *) buf
;
634 unsigned int a
= (unsigned int) addr
;
637 printk(KERN_DEBUG
"pci_insb(addr=%x,buf=%x,len=%d)\n", (int)addr
, (int)buf
, len
);
640 rp
= (volatile unsigned long *) COMEM_BASE
;
641 rp
[LREG(COMEM_DAHBASE
)] = COMEM_DA_IORD
| COMEM_DA_ADDR(a
);
643 a
= (a
& ~0x3) + (~a
& 0x03);
644 bp
= (volatile unsigned char *)
645 (COMEM_BASE
+ COMEM_PCIBUS
+ COMEM_DA_OFFSET(a
));
651 /*****************************************************************************/
653 void pci_insw(void *addr
, void *buf
, int len
)
655 volatile unsigned long *rp
;
656 volatile unsigned short *wp
;
657 unsigned short w
, *dp
= (unsigned short *) buf
;
658 unsigned int a
= (unsigned int) addr
;
661 printk(KERN_DEBUG
"pci_insw(addr=%x,buf=%x,len=%d)\n", (int)addr
, (int)buf
, len
);
664 rp
= (volatile unsigned long *) COMEM_BASE
;
665 rp
[LREG(COMEM_DAHBASE
)] = COMEM_DA_IORD
| COMEM_DA_ADDR(a
);
667 a
= (a
& ~0x3) + (~a
& 0x2);
668 wp
= (volatile unsigned short *)
669 (COMEM_BASE
+ COMEM_PCIBUS
+ COMEM_DA_OFFSET(a
));
674 w
= ((w
& 0xff) << 8) | ((w
>> 8) & 0xff);
679 /*****************************************************************************/
681 void pci_insl(void *addr
, void *buf
, int len
)
683 volatile unsigned long *rp
;
684 volatile unsigned long *lp
;
685 unsigned long l
, *dp
= (unsigned long *) buf
;
686 unsigned int a
= (unsigned int) addr
;
689 printk(KERN_DEBUG
"pci_insl(addr=%x,buf=%x,len=%d)\n", (int)addr
, (int)buf
, len
);
692 rp
= (volatile unsigned long *) COMEM_BASE
;
693 rp
[LREG(COMEM_DAHBASE
)] = COMEM_DA_IORD
| COMEM_DA_ADDR(a
);
695 lp
= (volatile unsigned long *)
696 (COMEM_BASE
+ COMEM_PCIBUS
+ COMEM_DA_OFFSET(a
));
701 l
= (l
<< 24) | ((l
& 0x0000ff00) << 8) |
702 ((l
& 0x00ff0000) >> 8) | (l
>> 24);
707 /*****************************************************************************/
709 struct pci_localirqlist
{
710 void (*handler
)(int, void *, struct pt_regs
*);
715 struct pci_localirqlist pci_irqlist
[COMEM_MAXPCI
];
717 /*****************************************************************************/
719 int pci_request_irq(unsigned int irq
,
720 void (*handler
)(int, void *, struct pt_regs
*),
721 unsigned long flags
, const char *device
, void *dev_id
)
726 printk(KERN_DEBUG
"pci_request_irq(irq=%d,handler=%x,flags=%x,device=%s,"
727 "dev_id=%x)\n", irq
, (int) handler
, (int) flags
, device
,
731 /* Check if this interrupt handler is already lodged */
732 for (i
= 0; (i
< COMEM_MAXPCI
); i
++) {
733 if (pci_irqlist
[i
].handler
== handler
)
737 /* Find a free spot to put this handler */
738 for (i
= 0; (i
< COMEM_MAXPCI
); i
++) {
739 if (pci_irqlist
[i
].handler
== 0) {
740 pci_irqlist
[i
].handler
= handler
;
741 pci_irqlist
[i
].device
= device
;
742 pci_irqlist
[i
].dev_id
= dev_id
;
751 /*****************************************************************************/
753 void pci_free_irq(unsigned int irq
, void *dev_id
)
758 printk(KERN_DEBUG
"pci_free_irq(irq=%d,dev_id=%x)\n", irq
, (int) dev_id
);
761 if (dev_id
== (void *) NULL
)
764 /* Check if this interrupt handler is lodged */
765 for (i
= 0; (i
< COMEM_MAXPCI
); i
++) {
766 if (pci_irqlist
[i
].dev_id
== dev_id
) {
767 pci_irqlist
[i
].handler
= NULL
;
768 pci_irqlist
[i
].device
= NULL
;
769 pci_irqlist
[i
].dev_id
= NULL
;
775 /*****************************************************************************/
777 void pci_interrupt(int irq
, void *id
, struct pt_regs
*fp
)
782 printk(KERN_DEBUG
"pci_interrupt(irq=%d,id=%x,fp=%x)\n", irq
, (int) id
, (int) fp
);
785 for (i
= 0; (i
< COMEM_MAXPCI
); i
++) {
786 if (pci_irqlist
[i
].handler
)
787 (*pci_irqlist
[i
].handler
)(irq
,pci_irqlist
[i
].dev_id
,fp
);
791 /*****************************************************************************/
794 * The shared memory region is broken up into contiguous 512 byte
795 * regions for easy allocation... This is not an optimal solution
796 * but it makes allocation and freeing regions really easy.
799 #define PCI_MEMSLOTSIZE 512
800 #define PCI_MEMSLOTS (COMEM_SHMEMSIZE / PCI_MEMSLOTSIZE)
802 char pci_shmemmap
[PCI_MEMSLOTS
];
805 void *pci_bmalloc(int size
)
810 printk(KERN_DEBUG
"pci_bmalloc(size=%d)\n", size
);
814 return((void *) NULL
);
816 nrslots
= (size
- 1) / PCI_MEMSLOTSIZE
;
818 for (i
= 0; (i
< (PCI_MEMSLOTS
-nrslots
)); i
++) {
819 if (pci_shmemmap
[i
] == 0) {
820 for (j
= i
+1; (j
< (i
+nrslots
)); j
++) {
825 for (j
= i
; (j
<= i
+nrslots
); j
++)
832 return((void *) (COMEM_BASE
+ COMEM_SHMEM
+ (i
* PCI_MEMSLOTSIZE
)));
835 /*****************************************************************************/
837 void pci_bmfree(void *mp
, int size
)
842 printk(KERN_DEBUG
"pci_bmfree(mp=%x,size=%d)\n", (int) mp
, size
);
845 nrslots
= size
/ PCI_MEMSLOTSIZE
;
846 i
= (((unsigned long) mp
) - (COMEM_BASE
+ COMEM_SHMEM
)) /
849 for (j
= i
; (j
< (i
+nrslots
)); j
++)
853 /*****************************************************************************/
855 unsigned long pci_virt_to_bus(volatile void *address
)
860 printk(KERN_DEBUG
"pci_virt_to_bus(address=%x)", (int) address
);
863 l
= ((unsigned long) address
) - COMEM_BASE
;
865 printk(KERN_DEBUG
"=%x\n", (int) (l
+pci_shmemaddr
));
867 return(l
+ pci_shmemaddr
);
870 /*****************************************************************************/
872 void *pci_bus_to_virt(unsigned long address
)
877 printk(KERN_DEBUG
"pci_bus_to_virt(address=%x)", (int) address
);
880 l
= address
- pci_shmemaddr
;
882 printk(KERN_DEBUG
"=%x\n", (int) (address
+ COMEM_BASE
));
884 return((void *) (address
+ COMEM_BASE
));
887 /*****************************************************************************/
889 void pci_bmcpyto(void *dst
, void *src
, int len
)
891 unsigned long *dp
, *sp
, val
;
892 unsigned char *dcp
, *scp
;
896 printk(KERN_DEBUG
"pci_bmcpyto(dst=%x,src=%x,len=%d)\n", (int)dst
, (int)src
, len
);
899 dp
= (unsigned long *) dst
;
900 sp
= (unsigned long *) src
;
904 printk(KERN_INFO
"DATA:");
905 scp
= (unsigned char *) sp
;
906 for (i
= 0; (i
< len
); i
++) {
907 if ((i
% 16) == 0) printk(KERN_INFO
"\n%04x: ", i
);
908 printk(KERN_INFO
"%02x ", *scp
++);
910 printk(KERN_INFO
"\n");
913 for (j
= 0; (i
>= 0); i
--, j
++) {
915 val
= (val
<< 24) | ((val
& 0x0000ff00) << 8) |
916 ((val
& 0x00ff0000) >> 8) | (val
>> 24);
921 dcp
= (unsigned char *) dp
;
922 scp
= ((unsigned char *) sp
) + 3;
923 for (i
= 0; (i
< (len
& 0x3)); i
++)
928 /*****************************************************************************/
930 void pci_bmcpyfrom(void *dst
, void *src
, int len
)
932 unsigned long *dp
, *sp
, val
;
933 unsigned char *dcp
, *scp
;
937 printk(KERN_DEBUG
"pci_bmcpyfrom(dst=%x,src=%x,len=%d)\n",(int)dst
,(int)src
,len
);
940 dp
= (unsigned long *) dst
;
941 sp
= (unsigned long *) src
;
944 for (; (i
>= 0); i
--) {
946 val
= (val
<< 24) | ((val
& 0x0000ff00) << 8) |
947 ((val
& 0x00ff0000) >> 8) | (val
>> 24);
952 dcp
= ((unsigned char *) dp
) + 3;
953 scp
= (unsigned char *) sp
;
954 for (i
= 0; (i
< (len
& 0x3)); i
++)
959 printk(KERN_INFO
"DATA:");
960 dcp
= (unsigned char *) dst
;
961 for (i
= 0; (i
< len
); i
++) {
962 if ((i
% 16) == 0) printk(KERN_INFO
"\n%04x: ", i
);
963 printk(KERN_INFO
"%02x ", *dcp
++);
965 printk(KERN_INFO
"\n");
969 /*****************************************************************************/
971 void *pci_alloc_consistent(struct pci_dev
*dev
, size_t size
, dma_addr_t
*dma_addr
)
974 if ((mp
= pci_bmalloc(size
)) != NULL
) {
975 dma_addr
= mp
- (COMEM_BASE
+ COMEM_SHMEM
);
978 *dma_addr
= (dma_addr_t
) NULL
;
982 /*****************************************************************************/
984 void pci_free_consistent(struct pci_dev
*dev
, size_t size
, void *cpu_addr
, dma_addr_t dma_addr
)
986 pci_bmfree(cpu_addr
, size
);
989 /*****************************************************************************/