diff options
author | Huacai Chen <chenhc@lemote.com> | 2014-03-21 06:44:03 -0400 |
---|---|---|
committer | Ralf Baechle <ralf@linux-mips.org> | 2014-03-31 12:17:12 -0400 |
commit | c7d3555ac07503d471d0ef75495c7370f7ec7aa1 (patch) | |
tree | bcf2c94a7194f42bd0ab34bf4b452888d092b400 /arch | |
parent | 1a08f1524d2ee4d4239e56ee1b3f6da0df929563 (diff) |
MIPS: Loongson 3: Add HT-linked PCI support
Loongson family machines use Hyper-Transport bus for inter-core
connection and device connection. The PCI bus is a subordinate
linked at HT1.
With LEFI firmware interface, We don't need fixup for PCI irq routing
(except providing a VBIOS of the integrated GPU).
Signed-off-by: Huacai Chen <chenhc@lemote.com>
Signed-off-by: Hongliang Tao <taohl@lemote.com>
Signed-off-by: Hua Yan <yanh@lemote.com>
Tested-by: Alex Smith <alex.smith@imgtec.com>
Reviewed-by: Alex Smith <alex.smith@imgtec.com>
Cc: John Crispin <john@phrozen.org>
Cc: Steven J. Hill <Steven.Hill@imgtec.com>
Cc: Aurelien Jarno <aurelien@aurel32.net>
Cc: linux-mips@linux-mips.org
Cc: Fuxin Zhang <zhangfx@lemote.com>
Cc: Zhangjin Wu <wuzhangjin@gmail.com>
Patchwork: https://patchwork.linux-mips.org/patch/6633
Signed-off-by: Ralf Baechle <ralf@linux-mips.org>
Diffstat (limited to 'arch')
-rw-r--r-- | arch/mips/include/asm/mach-loongson/loongson.h | 7 | ||||
-rw-r--r-- | arch/mips/include/asm/mach-loongson/pci.h | 5 | ||||
-rw-r--r-- | arch/mips/pci/Makefile | 1 | ||||
-rw-r--r-- | arch/mips/pci/fixup-loongson3.c | 66 | ||||
-rw-r--r-- | arch/mips/pci/ops-loongson3.c | 101 |
5 files changed, 180 insertions, 0 deletions
diff --git a/arch/mips/include/asm/mach-loongson/loongson.h b/arch/mips/include/asm/mach-loongson/loongson.h index 5913ea064ebe..f0367ffbf8c1 100644 --- a/arch/mips/include/asm/mach-loongson/loongson.h +++ b/arch/mips/include/asm/mach-loongson/loongson.h | |||
@@ -15,6 +15,7 @@ | |||
15 | #include <linux/init.h> | 15 | #include <linux/init.h> |
16 | #include <linux/irq.h> | 16 | #include <linux/irq.h> |
17 | #include <linux/kconfig.h> | 17 | #include <linux/kconfig.h> |
18 | #include <boot_param.h> | ||
18 | 19 | ||
19 | /* loongson internal northbridge initialization */ | 20 | /* loongson internal northbridge initialization */ |
20 | extern void bonito_irq_init(void); | 21 | extern void bonito_irq_init(void); |
@@ -101,7 +102,13 @@ static inline void do_perfcnt_IRQ(void) | |||
101 | #define LOONGSON_PCICFG_BASE 0x1fe80000 | 102 | #define LOONGSON_PCICFG_BASE 0x1fe80000 |
102 | #define LOONGSON_PCICFG_SIZE 0x00000800 /* 2K */ | 103 | #define LOONGSON_PCICFG_SIZE 0x00000800 /* 2K */ |
103 | #define LOONGSON_PCICFG_TOP (LOONGSON_PCICFG_BASE+LOONGSON_PCICFG_SIZE-1) | 104 | #define LOONGSON_PCICFG_TOP (LOONGSON_PCICFG_BASE+LOONGSON_PCICFG_SIZE-1) |
105 | |||
106 | #if defined(CONFIG_HT_PCI) | ||
107 | #define LOONGSON_PCIIO_BASE loongson_sysconf.pci_io_base | ||
108 | #else | ||
104 | #define LOONGSON_PCIIO_BASE 0x1fd00000 | 109 | #define LOONGSON_PCIIO_BASE 0x1fd00000 |
110 | #endif | ||
111 | |||
105 | #define LOONGSON_PCIIO_SIZE 0x00100000 /* 1M */ | 112 | #define LOONGSON_PCIIO_SIZE 0x00100000 /* 1M */ |
106 | #define LOONGSON_PCIIO_TOP (LOONGSON_PCIIO_BASE+LOONGSON_PCIIO_SIZE-1) | 113 | #define LOONGSON_PCIIO_TOP (LOONGSON_PCIIO_BASE+LOONGSON_PCIIO_SIZE-1) |
107 | 114 | ||
diff --git a/arch/mips/include/asm/mach-loongson/pci.h b/arch/mips/include/asm/mach-loongson/pci.h index bc99dab4ef63..1212774f66ef 100644 --- a/arch/mips/include/asm/mach-loongson/pci.h +++ b/arch/mips/include/asm/mach-loongson/pci.h | |||
@@ -40,8 +40,13 @@ extern struct pci_ops loongson_pci_ops; | |||
40 | #else /* loongson2f/32bit & loongson2e */ | 40 | #else /* loongson2f/32bit & loongson2e */ |
41 | 41 | ||
42 | /* this pci memory space is mapped by pcimap in pci.c */ | 42 | /* this pci memory space is mapped by pcimap in pci.c */ |
43 | #ifdef CONFIG_CPU_LOONGSON3 | ||
44 | #define LOONGSON_PCI_MEM_START 0x40000000UL | ||
45 | #define LOONGSON_PCI_MEM_END 0x7effffffUL | ||
46 | #else | ||
43 | #define LOONGSON_PCI_MEM_START LOONGSON_PCILO1_BASE | 47 | #define LOONGSON_PCI_MEM_START LOONGSON_PCILO1_BASE |
44 | #define LOONGSON_PCI_MEM_END (LOONGSON_PCILO1_BASE + 0x04000000 * 2) | 48 | #define LOONGSON_PCI_MEM_END (LOONGSON_PCILO1_BASE + 0x04000000 * 2) |
49 | #endif | ||
45 | /* this is an offset from mips_io_port_base */ | 50 | /* this is an offset from mips_io_port_base */ |
46 | #define LOONGSON_PCI_IO_START 0x00004000UL | 51 | #define LOONGSON_PCI_IO_START 0x00004000UL |
47 | 52 | ||
diff --git a/arch/mips/pci/Makefile b/arch/mips/pci/Makefile index 137f2a6feb25..d61138a177cc 100644 --- a/arch/mips/pci/Makefile +++ b/arch/mips/pci/Makefile | |||
@@ -29,6 +29,7 @@ obj-$(CONFIG_LASAT) += pci-lasat.o | |||
29 | obj-$(CONFIG_MIPS_COBALT) += fixup-cobalt.o | 29 | obj-$(CONFIG_MIPS_COBALT) += fixup-cobalt.o |
30 | obj-$(CONFIG_LEMOTE_FULOONG2E) += fixup-fuloong2e.o ops-loongson2.o | 30 | obj-$(CONFIG_LEMOTE_FULOONG2E) += fixup-fuloong2e.o ops-loongson2.o |
31 | obj-$(CONFIG_LEMOTE_MACH2F) += fixup-lemote2f.o ops-loongson2.o | 31 | obj-$(CONFIG_LEMOTE_MACH2F) += fixup-lemote2f.o ops-loongson2.o |
32 | obj-$(CONFIG_LEMOTE_MACH3A) += fixup-loongson3.o ops-loongson3.o | ||
32 | obj-$(CONFIG_MIPS_MALTA) += fixup-malta.o pci-malta.o | 33 | obj-$(CONFIG_MIPS_MALTA) += fixup-malta.o pci-malta.o |
33 | obj-$(CONFIG_PMC_MSP7120_GW) += fixup-pmcmsp.o ops-pmcmsp.o | 34 | obj-$(CONFIG_PMC_MSP7120_GW) += fixup-pmcmsp.o ops-pmcmsp.o |
34 | obj-$(CONFIG_PMC_MSP7120_EVAL) += fixup-pmcmsp.o ops-pmcmsp.o | 35 | obj-$(CONFIG_PMC_MSP7120_EVAL) += fixup-pmcmsp.o ops-pmcmsp.o |
diff --git a/arch/mips/pci/fixup-loongson3.c b/arch/mips/pci/fixup-loongson3.c new file mode 100644 index 000000000000..d708ae46d325 --- /dev/null +++ b/arch/mips/pci/fixup-loongson3.c | |||
@@ -0,0 +1,66 @@ | |||
1 | /* | ||
2 | * fixup-loongson3.c | ||
3 | * | ||
4 | * Copyright (C) 2012 Lemote, Inc. | ||
5 | * Author: Xiang Yu, xiangy@lemote.com | ||
6 | * Chen Huacai, chenhc@lemote.com | ||
7 | * | ||
8 | * This program is free software; you can redistribute it and/or modify it | ||
9 | * under the terms of the GNU General Public License as published by the | ||
10 | * Free Software Foundation; either version 2 of the License, or (at your | ||
11 | * option) any later version. | ||
12 | * | ||
13 | * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESS OR IMPLIED | ||
14 | * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF | ||
15 | * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN | ||
16 | * NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, | ||
17 | * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT | ||
18 | * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF | ||
19 | * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON | ||
20 | * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
21 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF | ||
22 | * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
23 | * | ||
24 | */ | ||
25 | |||
26 | #include <linux/pci.h> | ||
27 | #include <boot_param.h> | ||
28 | |||
29 | static void print_fixup_info(const struct pci_dev *pdev) | ||
30 | { | ||
31 | dev_info(&pdev->dev, "Device %x:%x, irq %d\n", | ||
32 | pdev->vendor, pdev->device, pdev->irq); | ||
33 | } | ||
34 | |||
35 | int __init pcibios_map_irq(const struct pci_dev *dev, u8 slot, u8 pin) | ||
36 | { | ||
37 | print_fixup_info(dev); | ||
38 | return dev->irq; | ||
39 | } | ||
40 | |||
41 | static void pci_fixup_radeon(struct pci_dev *pdev) | ||
42 | { | ||
43 | if (pdev->resource[PCI_ROM_RESOURCE].start) | ||
44 | return; | ||
45 | |||
46 | if (!loongson_sysconf.vgabios_addr) | ||
47 | return; | ||
48 | |||
49 | pdev->resource[PCI_ROM_RESOURCE].start = | ||
50 | loongson_sysconf.vgabios_addr; | ||
51 | pdev->resource[PCI_ROM_RESOURCE].end = | ||
52 | loongson_sysconf.vgabios_addr + 256*1024 - 1; | ||
53 | pdev->resource[PCI_ROM_RESOURCE].flags |= IORESOURCE_ROM_COPY; | ||
54 | |||
55 | dev_info(&pdev->dev, "BAR %d: assigned %pR for Radeon ROM\n", | ||
56 | PCI_ROM_RESOURCE, &pdev->resource[PCI_ROM_RESOURCE]); | ||
57 | } | ||
58 | |||
59 | DECLARE_PCI_FIXUP_CLASS_FINAL(PCI_VENDOR_ID_ATI, PCI_ANY_ID, | ||
60 | PCI_CLASS_DISPLAY_VGA, 8, pci_fixup_radeon); | ||
61 | |||
62 | /* Do platform specific device initialization at pci_enable_device() time */ | ||
63 | int pcibios_plat_dev_init(struct pci_dev *dev) | ||
64 | { | ||
65 | return 0; | ||
66 | } | ||
diff --git a/arch/mips/pci/ops-loongson3.c b/arch/mips/pci/ops-loongson3.c new file mode 100644 index 000000000000..46ed541a3ec7 --- /dev/null +++ b/arch/mips/pci/ops-loongson3.c | |||
@@ -0,0 +1,101 @@ | |||
1 | #include <linux/types.h> | ||
2 | #include <linux/pci.h> | ||
3 | #include <linux/kernel.h> | ||
4 | |||
5 | #include <asm/mips-boards/bonito64.h> | ||
6 | |||
7 | #include <loongson.h> | ||
8 | |||
9 | #define PCI_ACCESS_READ 0 | ||
10 | #define PCI_ACCESS_WRITE 1 | ||
11 | |||
12 | #define HT1LO_PCICFG_BASE 0x1a000000 | ||
13 | #define HT1LO_PCICFG_BASE_TP1 0x1b000000 | ||
14 | |||
15 | static int loongson3_pci_config_access(unsigned char access_type, | ||
16 | struct pci_bus *bus, unsigned int devfn, | ||
17 | int where, u32 *data) | ||
18 | { | ||
19 | unsigned char busnum = bus->number; | ||
20 | u_int64_t addr, type; | ||
21 | void *addrp; | ||
22 | int device = PCI_SLOT(devfn); | ||
23 | int function = PCI_FUNC(devfn); | ||
24 | int reg = where & ~3; | ||
25 | |||
26 | addr = (busnum << 16) | (device << 11) | (function << 8) | reg; | ||
27 | if (busnum == 0) { | ||
28 | if (device > 31) | ||
29 | return PCIBIOS_DEVICE_NOT_FOUND; | ||
30 | addrp = (void *)(TO_UNCAC(HT1LO_PCICFG_BASE) | (addr & 0xffff)); | ||
31 | type = 0; | ||
32 | |||
33 | } else { | ||
34 | addrp = (void *)(TO_UNCAC(HT1LO_PCICFG_BASE_TP1) | (addr)); | ||
35 | type = 0x10000; | ||
36 | } | ||
37 | |||
38 | if (access_type == PCI_ACCESS_WRITE) | ||
39 | writel(*data, addrp); | ||
40 | else { | ||
41 | *data = readl(addrp); | ||
42 | if (*data == 0xffffffff) { | ||
43 | *data = -1; | ||
44 | return PCIBIOS_DEVICE_NOT_FOUND; | ||
45 | } | ||
46 | } | ||
47 | return PCIBIOS_SUCCESSFUL; | ||
48 | } | ||
49 | |||
50 | static int loongson3_pci_pcibios_read(struct pci_bus *bus, unsigned int devfn, | ||
51 | int where, int size, u32 *val) | ||
52 | { | ||
53 | u32 data = 0; | ||
54 | int ret = loongson3_pci_config_access(PCI_ACCESS_READ, | ||
55 | bus, devfn, where, &data); | ||
56 | |||
57 | if (ret != PCIBIOS_SUCCESSFUL) | ||
58 | return ret; | ||
59 | |||
60 | if (size == 1) | ||
61 | *val = (data >> ((where & 3) << 3)) & 0xff; | ||
62 | else if (size == 2) | ||
63 | *val = (data >> ((where & 3) << 3)) & 0xffff; | ||
64 | else | ||
65 | *val = data; | ||
66 | |||
67 | return PCIBIOS_SUCCESSFUL; | ||
68 | } | ||
69 | |||
70 | static int loongson3_pci_pcibios_write(struct pci_bus *bus, unsigned int devfn, | ||
71 | int where, int size, u32 val) | ||
72 | { | ||
73 | u32 data = 0; | ||
74 | int ret; | ||
75 | |||
76 | if (size == 4) | ||
77 | data = val; | ||
78 | else { | ||
79 | ret = loongson3_pci_config_access(PCI_ACCESS_READ, | ||
80 | bus, devfn, where, &data); | ||
81 | if (ret != PCIBIOS_SUCCESSFUL) | ||
82 | return ret; | ||
83 | |||
84 | if (size == 1) | ||
85 | data = (data & ~(0xff << ((where & 3) << 3))) | | ||
86 | (val << ((where & 3) << 3)); | ||
87 | else if (size == 2) | ||
88 | data = (data & ~(0xffff << ((where & 3) << 3))) | | ||
89 | (val << ((where & 3) << 3)); | ||
90 | } | ||
91 | |||
92 | ret = loongson3_pci_config_access(PCI_ACCESS_WRITE, | ||
93 | bus, devfn, where, &data); | ||
94 | |||
95 | return ret; | ||
96 | } | ||
97 | |||
98 | struct pci_ops loongson_pci_ops = { | ||
99 | .read = loongson3_pci_pcibios_read, | ||
100 | .write = loongson3_pci_pcibios_write | ||
101 | }; | ||