xref: /linux/arch/sh/drivers/pci/common.c (revision a1944676767e855869b6af8e1c7e185372feaf31)
1 // SPDX-License-Identifier: GPL-2.0
2 #include <linux/pci.h>
3 #include <linux/interrupt.h>
4 #include <linux/timer.h>
5 #include <linux/kernel.h>
6 
7 /*
8  * These functions are used early on before PCI scanning is done
9  * and all of the pci_dev and pci_bus structures have been created.
10  */
11 static struct pci_dev *fake_pci_dev(struct pci_channel *hose,
12 	int top_bus, int busnr, int devfn)
13 {
14 	static struct pci_dev dev;
15 	static struct pci_bus bus;
16 
17 	dev.bus = &bus;
18 	dev.sysdata = hose;
19 	dev.devfn = devfn;
20 	bus.number = busnr;
21 	bus.sysdata = hose;
22 	bus.ops = hose->pci_ops;
23 
24 	if(busnr != top_bus)
25 		/* Fake a parent bus structure. */
26 		bus.parent = &bus;
27 	else
28 		bus.parent = NULL;
29 
30 	return &dev;
31 }
32 
33 #define EARLY_PCI_OP(rw, size, type)					\
34 int __init early_##rw##_config_##size(struct pci_channel *hose,		\
35 	int top_bus, int bus, int devfn, int offset, type value)	\
36 {									\
37 	return pci_##rw##_config_##size(				\
38 		fake_pci_dev(hose, top_bus, bus, devfn),		\
39 		offset, value);						\
40 }
41 
42 EARLY_PCI_OP(read, byte, u8 *)
43 EARLY_PCI_OP(read, word, u16 *)
44 EARLY_PCI_OP(read, dword, u32 *)
45 EARLY_PCI_OP(write, byte, u8)
46 EARLY_PCI_OP(write, word, u16)
47 EARLY_PCI_OP(write, dword, u32)
48 
49 int __init pci_is_66mhz_capable(struct pci_channel *hose,
50 				int top_bus, int current_bus)
51 {
52 	u32 pci_devfn;
53 	u16 vid;
54 	int cap66 = -1;
55 	u16 stat;
56 	int ret;
57 
58 	pr_info("PCI: Checking 66MHz capabilities...\n");
59 
60 	for (pci_devfn = 0; pci_devfn < 0xff; pci_devfn++) {
61 		if (PCI_FUNC(pci_devfn))
62 			continue;
63 		ret = early_read_config_word(hose, top_bus, current_bus,
64 					     pci_devfn, PCI_VENDOR_ID, &vid);
65 		if (ret != PCIBIOS_SUCCESSFUL)
66 			continue;
67 		if (PCI_POSSIBLE_ERROR(vid))
68 			continue;
69 
70 		/* check 66MHz capability */
71 		if (cap66 < 0)
72 			cap66 = 1;
73 		if (cap66) {
74 			early_read_config_word(hose, top_bus, current_bus,
75 					       pci_devfn, PCI_STATUS, &stat);
76 			if (!(stat & PCI_STATUS_66MHZ)) {
77 				printk(KERN_DEBUG
78 				       "PCI: %02x:%02x not 66MHz capable.\n",
79 				       current_bus, pci_devfn);
80 				cap66 = 0;
81 				break;
82 			}
83 		}
84 	}
85 
86 	return cap66 > 0;
87 }
88 
89 static void pcibios_enable_err(struct timer_list *t)
90 {
91 	struct pci_channel *hose = from_timer(hose, t, err_timer);
92 
93 	del_timer(&hose->err_timer);
94 	printk(KERN_DEBUG "PCI: re-enabling error IRQ.\n");
95 	enable_irq(hose->err_irq);
96 }
97 
98 static void pcibios_enable_serr(struct timer_list *t)
99 {
100 	struct pci_channel *hose = from_timer(hose, t, serr_timer);
101 
102 	del_timer(&hose->serr_timer);
103 	printk(KERN_DEBUG "PCI: re-enabling system error IRQ.\n");
104 	enable_irq(hose->serr_irq);
105 }
106 
107 void pcibios_enable_timers(struct pci_channel *hose)
108 {
109 	if (hose->err_irq) {
110 		timer_setup(&hose->err_timer, pcibios_enable_err, 0);
111 	}
112 
113 	if (hose->serr_irq) {
114 		timer_setup(&hose->serr_timer, pcibios_enable_serr, 0);
115 	}
116 }
117 
118 /*
119  * A simple handler for the regular PCI status errors, called from IRQ
120  * context.
121  */
122 unsigned int pcibios_handle_status_errors(unsigned long addr,
123 					  unsigned int status,
124 					  struct pci_channel *hose)
125 {
126 	unsigned int cmd = 0;
127 
128 	if (status & PCI_STATUS_REC_MASTER_ABORT) {
129 		printk(KERN_DEBUG "PCI: master abort, pc=0x%08lx\n", addr);
130 		cmd |= PCI_STATUS_REC_MASTER_ABORT;
131 	}
132 
133 	if (status & PCI_STATUS_REC_TARGET_ABORT) {
134 		printk(KERN_DEBUG "PCI: target abort: ");
135 		pcibios_report_status(PCI_STATUS_REC_TARGET_ABORT |
136 				      PCI_STATUS_SIG_TARGET_ABORT |
137 				      PCI_STATUS_REC_MASTER_ABORT, 1);
138 		pr_cont("\n");
139 
140 		cmd |= PCI_STATUS_REC_TARGET_ABORT;
141 	}
142 
143 	if (status & (PCI_STATUS_PARITY | PCI_STATUS_DETECTED_PARITY)) {
144 		printk(KERN_DEBUG "PCI: parity error detected: ");
145 		pcibios_report_status(PCI_STATUS_PARITY |
146 				      PCI_STATUS_DETECTED_PARITY, 1);
147 		pr_cont("\n");
148 
149 		cmd |= PCI_STATUS_PARITY | PCI_STATUS_DETECTED_PARITY;
150 
151 		/* Now back off of the IRQ for awhile */
152 		if (hose->err_irq) {
153 			disable_irq_nosync(hose->err_irq);
154 			hose->err_timer.expires = jiffies + HZ;
155 			add_timer(&hose->err_timer);
156 		}
157 	}
158 
159 	return cmd;
160 }
161