diff options
author | Ishizaki Kou <kou.ishizaki@toshiba.co.jp> | 2008-04-24 05:26:28 -0400 |
---|---|---|
committer | Paul Mackerras <paulus@samba.org> | 2008-04-24 07:08:13 -0400 |
commit | 11eef455c2834e192c6ffe9f3ffd09af70fafe81 (patch) | |
tree | 199940f8ca3f83c58f2cec2ffdef3ad9c8829921 /arch/powerpc/platforms/cell | |
parent | 116bdc425c7e01e97cff2f3e6d0134511e8f13e3 (diff) |
[POWERPC] celleb: Move the SCC related code for celleb
This moves the SCC (Super Companion Chip) related code for celleb
into platforms/cell/.
All files in this patch are used by celleb-beat and celleb-native
commonly.
Signed-off-by: Kou Ishizaki <kou.ishizaki@toshiba.co.jp>
Acked-by: Arnd Bergmann <arnd@arndb.de>
Acked-by: Benjamin Herrenschmidt <benh@kernel.crashing.org>
Signed-off-by: Paul Mackerras <paulus@samba.org>
Diffstat (limited to 'arch/powerpc/platforms/cell')
-rw-r--r-- | arch/powerpc/platforms/cell/Makefile | 5 | ||||
-rw-r--r-- | arch/powerpc/platforms/cell/celleb_scc.h | 145 | ||||
-rw-r--r-- | arch/powerpc/platforms/cell/celleb_scc_epci.c | 438 | ||||
-rw-r--r-- | arch/powerpc/platforms/cell/celleb_scc_sio.c | 101 | ||||
-rw-r--r-- | arch/powerpc/platforms/cell/celleb_scc_uhc.c | 95 |
5 files changed, 783 insertions, 1 deletions
diff --git a/arch/powerpc/platforms/cell/Makefile b/arch/powerpc/platforms/cell/Makefile index b644f5717fc..3b6ee08701f 100644 --- a/arch/powerpc/platforms/cell/Makefile +++ b/arch/powerpc/platforms/cell/Makefile | |||
@@ -32,6 +32,9 @@ obj-$(CONFIG_PCI_MSI) += axon_msi.o | |||
32 | # celleb stuff | 32 | # celleb stuff |
33 | ifeq ($(CONFIG_PPC_CELLEB),y) | 33 | ifeq ($(CONFIG_PPC_CELLEB),y) |
34 | obj-y += celleb_setup.o \ | 34 | obj-y += celleb_setup.o \ |
35 | celleb_pci.o \ | 35 | celleb_pci.o celleb_scc_epci.o \ |
36 | celleb_scc_uhc.o \ | ||
36 | io-workarounds.o spider-pci.o | 37 | io-workarounds.o spider-pci.o |
38 | |||
39 | obj-$(CONFIG_SERIAL_TXX9) += celleb_scc_sio.o | ||
37 | endif | 40 | endif |
diff --git a/arch/powerpc/platforms/cell/celleb_scc.h b/arch/powerpc/platforms/cell/celleb_scc.h new file mode 100644 index 00000000000..6be1542a6e6 --- /dev/null +++ b/arch/powerpc/platforms/cell/celleb_scc.h | |||
@@ -0,0 +1,145 @@ | |||
1 | /* | ||
2 | * SCC (Super Companion Chip) definitions | ||
3 | * | ||
4 | * (C) Copyright 2004-2006 TOSHIBA CORPORATION | ||
5 | * | ||
6 | * This program is free software; you can redistribute it and/or modify | ||
7 | * it under the terms of the GNU General Public License as published by | ||
8 | * the Free Software Foundation; either version 2 of the License, or | ||
9 | * (at your option) any later version. | ||
10 | * | ||
11 | * This program is distributed in the hope that it will be useful, | ||
12 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
13 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
14 | * GNU General Public License for more details. | ||
15 | * | ||
16 | * You should have received a copy of the GNU General Public License along | ||
17 | * with this program; if not, write to the Free Software Foundation, Inc., | ||
18 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. | ||
19 | */ | ||
20 | |||
21 | #ifndef _CELLEB_SCC_H | ||
22 | #define _CELLEB_SCC_H | ||
23 | |||
24 | #define PCI_VENDOR_ID_TOSHIBA_2 0x102f | ||
25 | #define PCI_DEVICE_ID_TOSHIBA_SCC_PCIEXC_BRIDGE 0x01b0 | ||
26 | #define PCI_DEVICE_ID_TOSHIBA_SCC_EPCI_BRIDGE 0x01b1 | ||
27 | #define PCI_DEVICE_ID_TOSHIBA_SCC_BRIDGE 0x01b2 | ||
28 | #define PCI_DEVICE_ID_TOSHIBA_SCC_GBE 0x01b3 | ||
29 | #define PCI_DEVICE_ID_TOSHIBA_SCC_ATA 0x01b4 | ||
30 | #define PCI_DEVICE_ID_TOSHIBA_SCC_USB2 0x01b5 | ||
31 | #define PCI_DEVICE_ID_TOSHIBA_SCC_USB 0x01b6 | ||
32 | #define PCI_DEVICE_ID_TOSHIBA_SCC_ENCDEC 0x01b7 | ||
33 | |||
34 | #define SCC_EPCI_REG 0x0000d000 | ||
35 | |||
36 | /* EPCI registers */ | ||
37 | #define SCC_EPCI_CNF10_REG 0x010 | ||
38 | #define SCC_EPCI_CNF14_REG 0x014 | ||
39 | #define SCC_EPCI_CNF18_REG 0x018 | ||
40 | #define SCC_EPCI_PVBAT 0x100 | ||
41 | #define SCC_EPCI_VPMBAT 0x104 | ||
42 | #define SCC_EPCI_VPIBAT 0x108 | ||
43 | #define SCC_EPCI_VCSR 0x110 | ||
44 | #define SCC_EPCI_VIENAB 0x114 | ||
45 | #define SCC_EPCI_VISTAT 0x118 | ||
46 | #define SCC_EPCI_VRDCOUNT 0x124 | ||
47 | #define SCC_EPCI_BAM0 0x12c | ||
48 | #define SCC_EPCI_BAM1 0x134 | ||
49 | #define SCC_EPCI_BAM2 0x13c | ||
50 | #define SCC_EPCI_IADR 0x164 | ||
51 | #define SCC_EPCI_CLKRST 0x800 | ||
52 | #define SCC_EPCI_INTSET 0x804 | ||
53 | #define SCC_EPCI_STATUS 0x808 | ||
54 | #define SCC_EPCI_ABTSET 0x80c | ||
55 | #define SCC_EPCI_WATRP 0x810 | ||
56 | #define SCC_EPCI_DUMYRADR 0x814 | ||
57 | #define SCC_EPCI_SWRESP 0x818 | ||
58 | #define SCC_EPCI_CNTOPT 0x81c | ||
59 | #define SCC_EPCI_ECMODE 0xf00 | ||
60 | #define SCC_EPCI_IOM_AC_NUM 5 | ||
61 | #define SCC_EPCI_IOM_ACTE(n) (0xf10 + (n) * 4) | ||
62 | #define SCC_EPCI_IOT_AC_NUM 4 | ||
63 | #define SCC_EPCI_IOT_ACTE(n) (0xf30 + (n) * 4) | ||
64 | #define SCC_EPCI_MAEA 0xf50 | ||
65 | #define SCC_EPCI_MAEC 0xf54 | ||
66 | #define SCC_EPCI_CKCTRL 0xff0 | ||
67 | |||
68 | /* bits for SCC_EPCI_VCSR */ | ||
69 | #define SCC_EPCI_VCSR_FRE 0x00020000 | ||
70 | #define SCC_EPCI_VCSR_FWE 0x00010000 | ||
71 | #define SCC_EPCI_VCSR_DR 0x00000400 | ||
72 | #define SCC_EPCI_VCSR_SR 0x00000008 | ||
73 | #define SCC_EPCI_VCSR_AT 0x00000004 | ||
74 | |||
75 | /* bits for SCC_EPCI_VIENAB/SCC_EPCI_VISTAT */ | ||
76 | #define SCC_EPCI_VISTAT_PMPE 0x00000008 | ||
77 | #define SCC_EPCI_VISTAT_PMFE 0x00000004 | ||
78 | #define SCC_EPCI_VISTAT_PRA 0x00000002 | ||
79 | #define SCC_EPCI_VISTAT_PRD 0x00000001 | ||
80 | #define SCC_EPCI_VISTAT_ALL 0x0000000f | ||
81 | |||
82 | #define SCC_EPCI_VIENAB_PMPEE 0x00000008 | ||
83 | #define SCC_EPCI_VIENAB_PMFEE 0x00000004 | ||
84 | #define SCC_EPCI_VIENAB_PRA 0x00000002 | ||
85 | #define SCC_EPCI_VIENAB_PRD 0x00000001 | ||
86 | #define SCC_EPCI_VIENAB_ALL 0x0000000f | ||
87 | |||
88 | /* bits for SCC_EPCI_CLKRST */ | ||
89 | #define SCC_EPCI_CLKRST_CKS_MASK 0x00030000 | ||
90 | #define SCC_EPCI_CLKRST_CKS_2 0x00000000 | ||
91 | #define SCC_EPCI_CLKRST_CKS_4 0x00010000 | ||
92 | #define SCC_EPCI_CLKRST_CKS_8 0x00020000 | ||
93 | #define SCC_EPCI_CLKRST_PCICRST 0x00000400 | ||
94 | #define SCC_EPCI_CLKRST_BC 0x00000200 | ||
95 | #define SCC_EPCI_CLKRST_PCIRST 0x00000100 | ||
96 | #define SCC_EPCI_CLKRST_PCKEN 0x00000001 | ||
97 | |||
98 | /* bits for SCC_EPCI_INTSET/SCC_EPCI_STATUS */ | ||
99 | #define SCC_EPCI_INT_2M 0x01000000 | ||
100 | #define SCC_EPCI_INT_RERR 0x00200000 | ||
101 | #define SCC_EPCI_INT_SERR 0x00100000 | ||
102 | #define SCC_EPCI_INT_PRTER 0x00080000 | ||
103 | #define SCC_EPCI_INT_SER 0x00040000 | ||
104 | #define SCC_EPCI_INT_PER 0x00020000 | ||
105 | #define SCC_EPCI_INT_PAI 0x00010000 | ||
106 | #define SCC_EPCI_INT_1M 0x00000100 | ||
107 | #define SCC_EPCI_INT_PME 0x00000010 | ||
108 | #define SCC_EPCI_INT_INTD 0x00000008 | ||
109 | #define SCC_EPCI_INT_INTC 0x00000004 | ||
110 | #define SCC_EPCI_INT_INTB 0x00000002 | ||
111 | #define SCC_EPCI_INT_INTA 0x00000001 | ||
112 | #define SCC_EPCI_INT_DEVINT 0x0000000f | ||
113 | #define SCC_EPCI_INT_ALL 0x003f001f | ||
114 | #define SCC_EPCI_INT_ALLERR 0x003f0000 | ||
115 | |||
116 | /* bits for SCC_EPCI_CKCTRL */ | ||
117 | #define SCC_EPCI_CKCTRL_CRST0 0x00010000 | ||
118 | #define SCC_EPCI_CKCTRL_CRST1 0x00020000 | ||
119 | #define SCC_EPCI_CKCTRL_OCLKEN 0x00000100 | ||
120 | #define SCC_EPCI_CKCTRL_LCLKEN 0x00000001 | ||
121 | |||
122 | #define SCC_EPCI_IDSEL_AD_TO_SLOT(ad) ((ad) - 10) | ||
123 | #define SCC_EPCI_MAX_DEVNU SCC_EPCI_IDSEL_AD_TO_SLOT(32) | ||
124 | |||
125 | /* bits for SCC_EPCI_CNTOPT */ | ||
126 | #define SCC_EPCI_CNTOPT_O2PMB 0x00000002 | ||
127 | |||
128 | /* UHC registers */ | ||
129 | #define SCC_UHC_CKRCTRL 0xff0 | ||
130 | #define SCC_UHC_ECMODE 0xf00 | ||
131 | |||
132 | /* bits for SCC_UHC_CKRCTRL */ | ||
133 | #define SCC_UHC_F48MCKLEN 0x00000001 | ||
134 | #define SCC_UHC_P_SUSPEND 0x00000002 | ||
135 | #define SCC_UHC_PHY_SUSPEND_SEL 0x00000004 | ||
136 | #define SCC_UHC_HCLKEN 0x00000100 | ||
137 | #define SCC_UHC_USBEN 0x00010000 | ||
138 | #define SCC_UHC_USBCEN 0x00020000 | ||
139 | #define SCC_UHC_PHYEN 0x00040000 | ||
140 | |||
141 | /* bits for SCC_UHC_ECMODE */ | ||
142 | #define SCC_UHC_ECMODE_BY_BYTE 0x00000555 | ||
143 | #define SCC_UHC_ECMODE_BY_WORD 0x00000aaa | ||
144 | |||
145 | #endif /* _CELLEB_SCC_H */ | ||
diff --git a/arch/powerpc/platforms/cell/celleb_scc_epci.c b/arch/powerpc/platforms/cell/celleb_scc_epci.c new file mode 100644 index 00000000000..08c285b10e3 --- /dev/null +++ b/arch/powerpc/platforms/cell/celleb_scc_epci.c | |||
@@ -0,0 +1,438 @@ | |||
1 | /* | ||
2 | * Support for SCC external PCI | ||
3 | * | ||
4 | * (C) Copyright 2004-2007 TOSHIBA CORPORATION | ||
5 | * | ||
6 | * This program is free software; you can redistribute it and/or modify | ||
7 | * it under the terms of the GNU General Public License as published by | ||
8 | * the Free Software Foundation; either version 2 of the License, or | ||
9 | * (at your option) any later version. | ||
10 | * | ||
11 | * This program is distributed in the hope that it will be useful, | ||
12 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
13 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
14 | * GNU General Public License for more details. | ||
15 | * | ||
16 | * You should have received a copy of the GNU General Public License along | ||
17 | * with this program; if not, write to the Free Software Foundation, Inc., | ||
18 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. | ||
19 | */ | ||
20 | |||
21 | #undef DEBUG | ||
22 | |||
23 | #include <linux/kernel.h> | ||
24 | #include <linux/threads.h> | ||
25 | #include <linux/pci.h> | ||
26 | #include <linux/init.h> | ||
27 | #include <linux/pci_regs.h> | ||
28 | #include <linux/bootmem.h> | ||
29 | |||
30 | #include <asm/io.h> | ||
31 | #include <asm/irq.h> | ||
32 | #include <asm/prom.h> | ||
33 | #include <asm/pci-bridge.h> | ||
34 | #include <asm/ppc-pci.h> | ||
35 | |||
36 | #include "celleb_scc.h" | ||
37 | #include "celleb_pci.h" | ||
38 | |||
39 | #define MAX_PCI_DEVICES 32 | ||
40 | #define MAX_PCI_FUNCTIONS 8 | ||
41 | |||
42 | #define iob() __asm__ __volatile__("eieio; sync":::"memory") | ||
43 | |||
44 | static inline PCI_IO_ADDR celleb_epci_get_epci_base( | ||
45 | struct pci_controller *hose) | ||
46 | { | ||
47 | /* | ||
48 | * Note: | ||
49 | * Celleb epci uses cfg_addr as a base address for | ||
50 | * epci control registers. | ||
51 | */ | ||
52 | |||
53 | return hose->cfg_addr; | ||
54 | } | ||
55 | |||
56 | static inline PCI_IO_ADDR celleb_epci_get_epci_cfg( | ||
57 | struct pci_controller *hose) | ||
58 | { | ||
59 | /* | ||
60 | * Note: | ||
61 | * Celleb epci uses cfg_data as a base address for | ||
62 | * configuration area for epci devices. | ||
63 | */ | ||
64 | |||
65 | return hose->cfg_data; | ||
66 | } | ||
67 | |||
68 | static inline void clear_and_disable_master_abort_interrupt( | ||
69 | struct pci_controller *hose) | ||
70 | { | ||
71 | PCI_IO_ADDR epci_base; | ||
72 | PCI_IO_ADDR reg; | ||
73 | epci_base = celleb_epci_get_epci_base(hose); | ||
74 | reg = epci_base + PCI_COMMAND; | ||
75 | out_be32(reg, in_be32(reg) | (PCI_STATUS_REC_MASTER_ABORT << 16)); | ||
76 | } | ||
77 | |||
78 | static int celleb_epci_check_abort(struct pci_controller *hose, | ||
79 | PCI_IO_ADDR addr) | ||
80 | { | ||
81 | PCI_IO_ADDR reg; | ||
82 | PCI_IO_ADDR epci_base; | ||
83 | u32 val; | ||
84 | |||
85 | iob(); | ||
86 | epci_base = celleb_epci_get_epci_base(hose); | ||
87 | |||
88 | reg = epci_base + PCI_COMMAND; | ||
89 | val = in_be32(reg); | ||
90 | |||
91 | if (val & (PCI_STATUS_REC_MASTER_ABORT << 16)) { | ||
92 | out_be32(reg, | ||
93 | (val & 0xffff) | (PCI_STATUS_REC_MASTER_ABORT << 16)); | ||
94 | |||
95 | /* clear PCI Controller error, FRE, PMFE */ | ||
96 | reg = epci_base + SCC_EPCI_STATUS; | ||
97 | out_be32(reg, SCC_EPCI_INT_PAI); | ||
98 | |||
99 | reg = epci_base + SCC_EPCI_VCSR; | ||
100 | val = in_be32(reg) & 0xffff; | ||
101 | val |= SCC_EPCI_VCSR_FRE; | ||
102 | out_be32(reg, val); | ||
103 | |||
104 | reg = epci_base + SCC_EPCI_VISTAT; | ||
105 | out_be32(reg, SCC_EPCI_VISTAT_PMFE); | ||
106 | return PCIBIOS_DEVICE_NOT_FOUND; | ||
107 | } | ||
108 | |||
109 | return PCIBIOS_SUCCESSFUL; | ||
110 | } | ||
111 | |||
112 | static PCI_IO_ADDR celleb_epci_make_config_addr(struct pci_bus *bus, | ||
113 | struct pci_controller *hose, unsigned int devfn, int where) | ||
114 | { | ||
115 | PCI_IO_ADDR addr; | ||
116 | |||
117 | if (bus != hose->bus) | ||
118 | addr = celleb_epci_get_epci_cfg(hose) + | ||
119 | (((bus->number & 0xff) << 16) | ||
120 | | ((devfn & 0xff) << 8) | ||
121 | | (where & 0xff) | ||
122 | | 0x01000000); | ||
123 | else | ||
124 | addr = celleb_epci_get_epci_cfg(hose) + | ||
125 | (((devfn & 0xff) << 8) | (where & 0xff)); | ||
126 | |||
127 | pr_debug("EPCI: config_addr = 0x%p\n", addr); | ||
128 | |||
129 | return addr; | ||
130 | } | ||
131 | |||
132 | static int celleb_epci_read_config(struct pci_bus *bus, | ||
133 | unsigned int devfn, int where, int size, u32 *val) | ||
134 | { | ||
135 | PCI_IO_ADDR epci_base; | ||
136 | PCI_IO_ADDR addr; | ||
137 | struct device_node *node; | ||
138 | struct pci_controller *hose; | ||
139 | |||
140 | /* allignment check */ | ||
141 | BUG_ON(where % size); | ||
142 | |||
143 | node = (struct device_node *)bus->sysdata; | ||
144 | hose = pci_find_hose_for_OF_device(node); | ||
145 | |||
146 | if (!celleb_epci_get_epci_cfg(hose)) | ||
147 | return PCIBIOS_DEVICE_NOT_FOUND; | ||
148 | |||
149 | if (bus->number == hose->first_busno && devfn == 0) { | ||
150 | /* EPCI controller self */ | ||
151 | |||
152 | epci_base = celleb_epci_get_epci_base(hose); | ||
153 | addr = epci_base + where; | ||
154 | |||
155 | switch (size) { | ||
156 | case 1: | ||
157 | *val = in_8(addr); | ||
158 | break; | ||
159 | case 2: | ||
160 | *val = in_be16(addr); | ||
161 | break; | ||
162 | case 4: | ||
163 | *val = in_be32(addr); | ||
164 | break; | ||
165 | default: | ||
166 | return PCIBIOS_DEVICE_NOT_FOUND; | ||
167 | } | ||
168 | |||
169 | } else { | ||
170 | |||
171 | clear_and_disable_master_abort_interrupt(hose); | ||
172 | addr = celleb_epci_make_config_addr(bus, hose, devfn, where); | ||
173 | |||
174 | switch (size) { | ||
175 | case 1: | ||
176 | *val = in_8(addr); | ||
177 | break; | ||
178 | case 2: | ||
179 | *val = in_le16(addr); | ||
180 | break; | ||
181 | case 4: | ||
182 | *val = in_le32(addr); | ||
183 | break; | ||
184 | default: | ||
185 | return PCIBIOS_DEVICE_NOT_FOUND; | ||
186 | } | ||
187 | } | ||
188 | |||
189 | pr_debug("EPCI: " | ||
190 | "addr=0x%p, devfn=0x%x, where=0x%x, size=0x%x, val=0x%x\n", | ||
191 | addr, devfn, where, size, *val); | ||
192 | |||
193 | return celleb_epci_check_abort(hose, NULL); | ||
194 | } | ||
195 | |||
196 | static int celleb_epci_write_config(struct pci_bus *bus, | ||
197 | unsigned int devfn, int where, int size, u32 val) | ||
198 | { | ||
199 | PCI_IO_ADDR epci_base; | ||
200 | PCI_IO_ADDR addr; | ||
201 | struct device_node *node; | ||
202 | struct pci_controller *hose; | ||
203 | |||
204 | /* allignment check */ | ||
205 | BUG_ON(where % size); | ||
206 | |||
207 | node = (struct device_node *)bus->sysdata; | ||
208 | hose = pci_find_hose_for_OF_device(node); | ||
209 | |||
210 | |||
211 | if (!celleb_epci_get_epci_cfg(hose)) | ||
212 | return PCIBIOS_DEVICE_NOT_FOUND; | ||
213 | |||
214 | if (bus->number == hose->first_busno && devfn == 0) { | ||
215 | /* EPCI controller self */ | ||
216 | |||
217 | epci_base = celleb_epci_get_epci_base(hose); | ||
218 | addr = epci_base + where; | ||
219 | |||
220 | switch (size) { | ||
221 | case 1: | ||
222 | out_8(addr, val); | ||
223 | break; | ||
224 | case 2: | ||
225 | out_be16(addr, val); | ||
226 | break; | ||
227 | case 4: | ||
228 | out_be32(addr, val); | ||
229 | break; | ||
230 | default: | ||
231 | return PCIBIOS_DEVICE_NOT_FOUND; | ||
232 | } | ||
233 | |||
234 | } else { | ||
235 | |||
236 | clear_and_disable_master_abort_interrupt(hose); | ||
237 | addr = celleb_epci_make_config_addr(bus, hose, devfn, where); | ||
238 | |||
239 | switch (size) { | ||
240 | case 1: | ||
241 | out_8(addr, val); | ||
242 | break; | ||
243 | case 2: | ||
244 | out_le16(addr, val); | ||
245 | break; | ||
246 | case 4: | ||
247 | out_le32(addr, val); | ||
248 | break; | ||
249 | default: | ||
250 | return PCIBIOS_DEVICE_NOT_FOUND; | ||
251 | } | ||
252 | } | ||
253 | |||
254 | return celleb_epci_check_abort(hose, addr); | ||
255 | } | ||
256 | |||
257 | struct pci_ops celleb_epci_ops = { | ||
258 | .read = celleb_epci_read_config, | ||
259 | .write = celleb_epci_write_config, | ||
260 | }; | ||
261 | |||
262 | /* to be moved in FW */ | ||
263 | static int __init celleb_epci_init(struct pci_controller *hose) | ||
264 | { | ||
265 | u32 val; | ||
266 | PCI_IO_ADDR reg; | ||
267 | PCI_IO_ADDR epci_base; | ||
268 | int hwres = 0; | ||
269 | |||
270 | epci_base = celleb_epci_get_epci_base(hose); | ||
271 | |||
272 | /* PCI core reset(Internal bus and PCI clock) */ | ||
273 | reg = epci_base + SCC_EPCI_CKCTRL; | ||
274 | val = in_be32(reg); | ||
275 | if (val == 0x00030101) | ||
276 | hwres = 1; | ||
277 | else { | ||
278 | val &= ~(SCC_EPCI_CKCTRL_CRST0 | SCC_EPCI_CKCTRL_CRST1); | ||
279 | out_be32(reg, val); | ||
280 | |||
281 | /* set PCI core clock */ | ||
282 | val = in_be32(reg); | ||
283 | val |= (SCC_EPCI_CKCTRL_OCLKEN | SCC_EPCI_CKCTRL_LCLKEN); | ||
284 | out_be32(reg, val); | ||
285 | |||
286 | /* release PCI core reset (internal bus) */ | ||
287 | val = in_be32(reg); | ||
288 | val |= SCC_EPCI_CKCTRL_CRST0; | ||
289 | out_be32(reg, val); | ||
290 | |||
291 | /* set PCI clock select */ | ||
292 | reg = epci_base + SCC_EPCI_CLKRST; | ||
293 | val = in_be32(reg); | ||
294 | val &= ~SCC_EPCI_CLKRST_CKS_MASK; | ||
295 | val |= SCC_EPCI_CLKRST_CKS_2; | ||
296 | out_be32(reg, val); | ||
297 | |||
298 | /* set arbiter */ | ||
299 | reg = epci_base + SCC_EPCI_ABTSET; | ||
300 | out_be32(reg, 0x0f1f001f); /* temporary value */ | ||
301 | |||
302 | /* buffer on */ | ||
303 | reg = epci_base + SCC_EPCI_CLKRST; | ||
304 | val = in_be32(reg); | ||
305 | val |= SCC_EPCI_CLKRST_BC; | ||
306 | out_be32(reg, val); | ||
307 | |||
308 | /* PCI clock enable */ | ||
309 | val = in_be32(reg); | ||
310 | val |= SCC_EPCI_CLKRST_PCKEN; | ||
311 | out_be32(reg, val); | ||
312 | |||
313 | /* release PCI core reset (all) */ | ||
314 | reg = epci_base + SCC_EPCI_CKCTRL; | ||
315 | val = in_be32(reg); | ||
316 | val |= (SCC_EPCI_CKCTRL_CRST0 | SCC_EPCI_CKCTRL_CRST1); | ||
317 | out_be32(reg, val); | ||
318 | |||
319 | /* set base translation registers. (already set by Beat) */ | ||
320 | |||
321 | /* set base address masks. (already set by Beat) */ | ||
322 | } | ||
323 | |||
324 | /* release interrupt masks and clear all interrupts */ | ||
325 | reg = epci_base + SCC_EPCI_INTSET; | ||
326 | out_be32(reg, 0x013f011f); /* all interrupts enable */ | ||
327 | reg = epci_base + SCC_EPCI_VIENAB; | ||
328 | val = SCC_EPCI_VIENAB_PMPEE | SCC_EPCI_VIENAB_PMFEE; | ||
329 | out_be32(reg, val); | ||
330 | reg = epci_base + SCC_EPCI_STATUS; | ||
331 | out_be32(reg, 0xffffffff); | ||
332 | reg = epci_base + SCC_EPCI_VISTAT; | ||
333 | out_be32(reg, 0xffffffff); | ||
334 | |||
335 | /* disable PCI->IB address translation */ | ||
336 | reg = epci_base + SCC_EPCI_VCSR; | ||
337 | val = in_be32(reg); | ||
338 | val &= ~(SCC_EPCI_VCSR_DR | SCC_EPCI_VCSR_AT); | ||
339 | out_be32(reg, val); | ||
340 | |||
341 | /* set base addresses. (no need to set?) */ | ||
342 | |||
343 | /* memory space, bus master enable */ | ||
344 | reg = epci_base + PCI_COMMAND; | ||
345 | val = PCI_COMMAND_MEMORY | PCI_COMMAND_MASTER; | ||
346 | out_be32(reg, val); | ||
347 | |||
348 | /* endian mode setup */ | ||
349 | reg = epci_base + SCC_EPCI_ECMODE; | ||
350 | val = 0x00550155; | ||
351 | out_be32(reg, val); | ||
352 | |||
353 | /* set control option */ | ||
354 | reg = epci_base + SCC_EPCI_CNTOPT; | ||
355 | val = in_be32(reg); | ||
356 | val |= SCC_EPCI_CNTOPT_O2PMB; | ||
357 | out_be32(reg, val); | ||
358 | |||
359 | /* XXX: temporay: set registers for address conversion setup */ | ||
360 | reg = epci_base + SCC_EPCI_CNF10_REG; | ||
361 | out_be32(reg, 0x80000008); | ||
362 | reg = epci_base + SCC_EPCI_CNF14_REG; | ||
363 | out_be32(reg, 0x40000008); | ||
364 | |||
365 | reg = epci_base + SCC_EPCI_BAM0; | ||
366 | out_be32(reg, 0x80000000); | ||
367 | reg = epci_base + SCC_EPCI_BAM1; | ||
368 | out_be32(reg, 0xe0000000); | ||
369 | |||
370 | reg = epci_base + SCC_EPCI_PVBAT; | ||
371 | out_be32(reg, 0x80000000); | ||
372 | |||
373 | if (!hwres) { | ||
374 | /* release external PCI reset */ | ||
375 | reg = epci_base + SCC_EPCI_CLKRST; | ||
376 | val = in_be32(reg); | ||
377 | val |= SCC_EPCI_CLKRST_PCIRST; | ||
378 | out_be32(reg, val); | ||
379 | } | ||
380 | |||
381 | return 0; | ||
382 | } | ||
383 | |||
384 | static int __init celleb_setup_epci(struct device_node *node, | ||
385 | struct pci_controller *hose) | ||
386 | { | ||
387 | struct resource r; | ||
388 | |||
389 | pr_debug("PCI: celleb_setup_epci()\n"); | ||
390 | |||
391 | /* | ||
392 | * Note: | ||
393 | * Celleb epci uses cfg_addr and cfg_data member of | ||
394 | * pci_controller structure in irregular way. | ||
395 | * | ||
396 | * cfg_addr is used to map for control registers of | ||
397 | * celleb epci. | ||
398 | * | ||
399 | * cfg_data is used for configuration area of devices | ||
400 | * on Celleb epci buses. | ||
401 | */ | ||
402 | |||
403 | if (of_address_to_resource(node, 0, &r)) | ||
404 | goto error; | ||
405 | hose->cfg_addr = ioremap(r.start, (r.end - r.start + 1)); | ||
406 | if (!hose->cfg_addr) | ||
407 | goto error; | ||
408 | pr_debug("EPCI: cfg_addr map 0x%016lx->0x%016lx + 0x%016lx\n", | ||
409 | r.start, (unsigned long)hose->cfg_addr, (r.end - r.start + 1)); | ||
410 | |||
411 | if (of_address_to_resource(node, 2, &r)) | ||
412 | goto error; | ||
413 | hose->cfg_data = ioremap(r.start, (r.end - r.start + 1)); | ||
414 | if (!hose->cfg_data) | ||
415 | goto error; | ||
416 | pr_debug("EPCI: cfg_data map 0x%016lx->0x%016lx + 0x%016lx\n", | ||
417 | r.start, (unsigned long)hose->cfg_data, (r.end - r.start + 1)); | ||
418 | |||
419 | hose->ops = &celleb_epci_ops; | ||
420 | celleb_epci_init(hose); | ||
421 | |||
422 | return 0; | ||
423 | |||
424 | error: | ||
425 | if (hose->cfg_addr) | ||
426 | iounmap(hose->cfg_addr); | ||
427 | |||
428 | if (hose->cfg_data) | ||
429 | iounmap(hose->cfg_data); | ||
430 | return 1; | ||
431 | } | ||
432 | |||
433 | struct celleb_phb_spec celleb_epci_spec __initdata = { | ||
434 | .setup = celleb_setup_epci, | ||
435 | .ops = &spiderpci_ops, | ||
436 | .iowa_init = &spiderpci_iowa_init, | ||
437 | .iowa_data = (void *)0, | ||
438 | }; | ||
diff --git a/arch/powerpc/platforms/cell/celleb_scc_sio.c b/arch/powerpc/platforms/cell/celleb_scc_sio.c new file mode 100644 index 00000000000..3a16c5b3c46 --- /dev/null +++ b/arch/powerpc/platforms/cell/celleb_scc_sio.c | |||
@@ -0,0 +1,101 @@ | |||
1 | /* | ||
2 | * setup serial port in SCC | ||
3 | * | ||
4 | * (C) Copyright 2006-2007 TOSHIBA CORPORATION | ||
5 | * | ||
6 | * This program is free software; you can redistribute it and/or modify | ||
7 | * it under the terms of the GNU General Public License as published by | ||
8 | * the Free Software Foundation; either version 2 of the License, or | ||
9 | * (at your option) any later version. | ||
10 | * | ||
11 | * This program is distributed in the hope that it will be useful, | ||
12 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
13 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
14 | * GNU General Public License for more details. | ||
15 | * | ||
16 | * You should have received a copy of the GNU General Public License along | ||
17 | * with this program; if not, write to the Free Software Foundation, Inc., | ||
18 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. | ||
19 | */ | ||
20 | |||
21 | #include <linux/tty.h> | ||
22 | #include <linux/serial.h> | ||
23 | #include <linux/serial_core.h> | ||
24 | #include <linux/console.h> | ||
25 | |||
26 | #include <asm/io.h> | ||
27 | #include <asm/prom.h> | ||
28 | |||
29 | /* sio irq0=0xb00010022 irq0=0xb00010023 irq2=0xb00010024 | ||
30 | mmio=0xfff000-0x1000,0xff2000-0x1000 */ | ||
31 | static int txx9_serial_bitmap __initdata; | ||
32 | |||
33 | static struct { | ||
34 | uint32_t offset; | ||
35 | uint32_t index; | ||
36 | } txx9_scc_tab[3] __initdata = { | ||
37 | { 0x300, 0 }, /* 0xFFF300 */ | ||
38 | { 0x400, 0 }, /* 0xFFF400 */ | ||
39 | { 0x800, 1 } /* 0xFF2800 */ | ||
40 | }; | ||
41 | |||
42 | static int __init txx9_serial_init(void) | ||
43 | { | ||
44 | extern int early_serial_txx9_setup(struct uart_port *port); | ||
45 | struct device_node *node = NULL; | ||
46 | int i; | ||
47 | struct uart_port req; | ||
48 | struct of_irq irq; | ||
49 | struct resource res; | ||
50 | |||
51 | while ((node = of_find_compatible_node(node, | ||
52 | "serial", "toshiba,sio-scc")) != NULL) { | ||
53 | for (i = 0; i < ARRAY_SIZE(txx9_scc_tab); i++) { | ||
54 | if (!(txx9_serial_bitmap & (1<<i))) | ||
55 | continue; | ||
56 | |||
57 | if (of_irq_map_one(node, i, &irq)) | ||
58 | continue; | ||
59 | if (of_address_to_resource(node, | ||
60 | txx9_scc_tab[i].index, &res)) | ||
61 | continue; | ||
62 | |||
63 | memset(&req, 0, sizeof(req)); | ||
64 | req.line = i; | ||
65 | req.iotype = UPIO_MEM; | ||
66 | req.mapbase = res.start + txx9_scc_tab[i].offset; | ||
67 | #ifdef CONFIG_SERIAL_TXX9_CONSOLE | ||
68 | req.membase = ioremap(req.mapbase, 0x24); | ||
69 | #endif | ||
70 | req.irq = irq_create_of_mapping(irq.controller, | ||
71 | irq.specifier, irq.size); | ||
72 | req.flags |= UPF_IOREMAP | UPF_BUGGY_UART | ||
73 | /*HAVE_CTS_LINE*/; | ||
74 | req.uartclk = 83300000; | ||
75 | early_serial_txx9_setup(&req); | ||
76 | } | ||
77 | } | ||
78 | |||
79 | return 0; | ||
80 | } | ||
81 | |||
82 | static int __init txx9_serial_config(char *ptr) | ||
83 | { | ||
84 | int i; | ||
85 | |||
86 | for (;;) { | ||
87 | switch (get_option(&ptr, &i)) { | ||
88 | default: | ||
89 | return 0; | ||
90 | case 2: | ||
91 | txx9_serial_bitmap |= 1 << i; | ||
92 | break; | ||
93 | case 1: | ||
94 | txx9_serial_bitmap |= 1 << i; | ||
95 | return 0; | ||
96 | } | ||
97 | } | ||
98 | } | ||
99 | __setup("txx9_serial=", txx9_serial_config); | ||
100 | |||
101 | console_initcall(txx9_serial_init); | ||
diff --git a/arch/powerpc/platforms/cell/celleb_scc_uhc.c b/arch/powerpc/platforms/cell/celleb_scc_uhc.c new file mode 100644 index 00000000000..d63b720bfe3 --- /dev/null +++ b/arch/powerpc/platforms/cell/celleb_scc_uhc.c | |||
@@ -0,0 +1,95 @@ | |||
1 | /* | ||
2 | * SCC (Super Companion Chip) UHC setup | ||
3 | * | ||
4 | * (C) Copyright 2006-2007 TOSHIBA CORPORATION | ||
5 | * | ||
6 | * This program is free software; you can redistribute it and/or modify | ||
7 | * it under the terms of the GNU General Public License as published by | ||
8 | * the Free Software Foundation; either version 2 of the License, or | ||
9 | * (at your option) any later version. | ||
10 | * | ||
11 | * This program is distributed in the hope that it will be useful, | ||
12 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
13 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
14 | * GNU General Public License for more details. | ||
15 | * | ||
16 | * You should have received a copy of the GNU General Public License along | ||
17 | * with this program; if not, write to the Free Software Foundation, Inc., | ||
18 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. | ||
19 | */ | ||
20 | |||
21 | #include <linux/kernel.h> | ||
22 | #include <linux/pci.h> | ||
23 | |||
24 | #include <asm/delay.h> | ||
25 | #include <asm/io.h> | ||
26 | #include <asm/machdep.h> | ||
27 | |||
28 | #include "celleb_scc.h" | ||
29 | |||
30 | #define UHC_RESET_WAIT_MAX 10000 | ||
31 | |||
32 | static inline int uhc_clkctrl_ready(u32 val) | ||
33 | { | ||
34 | const u32 mask = SCC_UHC_USBCEN | SCC_UHC_USBCEN; | ||
35 | return((val & mask) == mask); | ||
36 | } | ||
37 | |||
38 | /* | ||
39 | * UHC(usb host controller) enable function. | ||
40 | * affect to both of OHCI and EHCI core module. | ||
41 | */ | ||
42 | static void enable_scc_uhc(struct pci_dev *dev) | ||
43 | { | ||
44 | void __iomem *uhc_base; | ||
45 | u32 __iomem *uhc_clkctrl; | ||
46 | u32 __iomem *uhc_ecmode; | ||
47 | u32 val = 0; | ||
48 | int i; | ||
49 | |||
50 | if (!machine_is(celleb_beat) && | ||
51 | !machine_is(celleb_native)) | ||
52 | return; | ||
53 | |||
54 | uhc_base = ioremap(pci_resource_start(dev, 0), | ||
55 | pci_resource_len(dev, 0)); | ||
56 | if (!uhc_base) { | ||
57 | printk(KERN_ERR "failed to map UHC register base.\n"); | ||
58 | return; | ||
59 | } | ||
60 | uhc_clkctrl = uhc_base + SCC_UHC_CKRCTRL; | ||
61 | uhc_ecmode = uhc_base + SCC_UHC_ECMODE; | ||
62 | |||
63 | /* setup for normal mode */ | ||
64 | val |= SCC_UHC_F48MCKLEN; | ||
65 | out_be32(uhc_clkctrl, val); | ||
66 | val |= SCC_UHC_PHY_SUSPEND_SEL; | ||
67 | out_be32(uhc_clkctrl, val); | ||
68 | udelay(10); | ||
69 | val |= SCC_UHC_PHYEN; | ||
70 | out_be32(uhc_clkctrl, val); | ||
71 | udelay(50); | ||
72 | |||
73 | /* disable reset */ | ||
74 | val |= SCC_UHC_HCLKEN; | ||
75 | out_be32(uhc_clkctrl, val); | ||
76 | val |= (SCC_UHC_USBCEN | SCC_UHC_USBEN); | ||
77 | out_be32(uhc_clkctrl, val); | ||
78 | i = 0; | ||
79 | while (!uhc_clkctrl_ready(in_be32(uhc_clkctrl))) { | ||
80 | udelay(10); | ||
81 | if (i++ > UHC_RESET_WAIT_MAX) { | ||
82 | printk(KERN_ERR "Failed to disable UHC reset %x\n", | ||
83 | in_be32(uhc_clkctrl)); | ||
84 | break; | ||
85 | } | ||
86 | } | ||
87 | |||
88 | /* Endian Conversion Mode for Master ALL area */ | ||
89 | out_be32(uhc_ecmode, SCC_UHC_ECMODE_BY_BYTE); | ||
90 | |||
91 | iounmap(uhc_base); | ||
92 | } | ||
93 | |||
94 | DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_TOSHIBA_2, | ||
95 | PCI_DEVICE_ID_TOSHIBA_SCC_USB, enable_scc_uhc); | ||