dm: x86: spi: Convert ICH SPI driver to driver model PCI API

At present this SPI driver works by searching the PCI buses for its
peripheral. It also uses the legacy PCI API.

In addition the driver has code to determine the type of Intel PCH that is
used (version 7 or version 9). Now that we have proper PCH drivers we can
use those to obtain the information we need.

While the device tree has a node for the SPI peripheral it is not in the
right place. It should be on the PCI bus as a sub-peripheral of the LPC
device.

Update the device tree files to show the SPI controller within the PCH, so
that PCI access works as expected.

This patch includes Bin's fix-up patch from here:

   https://patchwork.ozlabs.org/patch/569478/

Signed-off-by: Simon Glass <sjg@chromium.org>
Signed-off-by: Bin Meng <bmeng.cn@gmail.com>
Reviewed-by: Bin Meng <bmeng.cn@gmail.com>
Tested-by: Bin Meng <bmeng.cn@gmail.com>
master
Simon Glass 9 years ago committed by Bin Meng
parent fffe25db04
commit f2b85ab5e6
  1. 3
      arch/x86/cpu/coreboot/pci.c
  2. 7
      arch/x86/cpu/irq.c
  3. 47
      arch/x86/cpu/ivybridge/bd82x6x.c
  4. 160
      arch/x86/dts/bayleybay.dts
  5. 24
      arch/x86/dts/broadwell_som-6896.dts
  6. 5
      arch/x86/dts/chromebook_link.dts
  7. 34
      arch/x86/dts/chromebox_panther.dts
  8. 150
      arch/x86/dts/crownbay.dts
  9. 99
      arch/x86/dts/galileo.dts
  10. 158
      arch/x86/dts/minnowmax.dts
  11. 26
      arch/x86/dts/qemu-x86_i440fx.dts
  12. 38
      arch/x86/dts/qemu-x86_q35.dts
  13. 152
      drivers/spi/ich.c

@ -14,7 +14,8 @@
#include <pci.h> #include <pci.h>
static const struct udevice_id generic_pch_ids[] = { static const struct udevice_id generic_pch_ids[] = {
{ .compatible = "intel,pch" }, { .compatible = "intel,pch7" },
{ .compatible = "intel,pch9" },
{ } { }
}; };

@ -97,6 +97,7 @@ static int create_pirq_routing_table(void)
struct irq_routing_table *rt; struct irq_routing_table *rt;
struct irq_info *slot, *slot_base; struct irq_info *slot, *slot_base;
int irq_entries = 0; int irq_entries = 0;
int parent;
int i; int i;
int ret; int ret;
@ -106,7 +107,11 @@ static int create_pirq_routing_table(void)
return -EINVAL; return -EINVAL;
} }
ret = fdtdec_get_pci_addr(blob, node, FDT_PCI_SPACE_CONFIG, /* TODO(sjg@chromium.org): Drop this when PIRQ is a driver */
parent = fdt_parent_offset(blob, node);
if (parent < 0)
return -EINVAL;
ret = fdtdec_get_pci_addr(blob, parent, FDT_PCI_SPACE_CONFIG,
"reg", &addr); "reg", &addr);
if (ret) if (ret)
return ret; return ret;

@ -3,12 +3,12 @@
* *
* SPDX-License-Identifier: GPL-2.0+ * SPDX-License-Identifier: GPL-2.0+
*/ */
#include <common.h> #include <common.h>
#include <dm.h> #include <dm.h>
#include <errno.h> #include <errno.h>
#include <fdtdec.h> #include <fdtdec.h>
#include <malloc.h> #include <malloc.h>
#include <pch.h>
#include <asm/lapic.h> #include <asm/lapic.h>
#include <asm/pci.h> #include <asm/pci.h>
#include <asm/arch/bd82x6x.h> #include <asm/arch/bd82x6x.h>
@ -16,6 +16,8 @@
#include <asm/arch/pch.h> #include <asm/arch/pch.h>
#include <asm/arch/sandybridge.h> #include <asm/arch/sandybridge.h>
#define BIOS_CTRL 0xdc
void bd82x6x_pci_init(pci_dev_t dev) void bd82x6x_pci_init(pci_dev_t dev)
{ {
u16 reg16; u16 reg16;
@ -96,6 +98,7 @@ static int bd82x6x_probe(struct udevice *dev)
return 0; return 0;
} }
/* TODO(sjg@chromium.org): Move this to the PCH init() method */
int bd82x6x_init(void) int bd82x6x_init(void)
{ {
const void *blob = gd->fdt_blob; const void *blob = gd->fdt_blob;
@ -116,6 +119,47 @@ int bd82x6x_init(void)
return 0; return 0;
} }
static int bd82x6x_pch_get_sbase(struct udevice *dev, ulong *sbasep)
{
u32 rcba;
dm_pci_read_config32(dev, PCH_RCBA, &rcba);
/* Bits 31-14 are the base address, 13-1 are reserved, 0 is enable */
rcba = rcba & 0xffffc000;
*sbasep = rcba + 0x3800;
return 0;
}
static enum pch_version bd82x6x_pch_get_version(struct udevice *dev)
{
return PCHV_9;
}
static int bd82x6x_set_spi_protect(struct udevice *dev, bool protect)
{
uint8_t bios_cntl;
/* Adjust the BIOS write protect and SMM BIOS Write Protect Disable */
dm_pci_read_config8(dev, BIOS_CTRL, &bios_cntl);
if (protect) {
bios_cntl &= ~BIOS_CTRL_BIOSWE;
bios_cntl |= BIT(5);
} else {
bios_cntl |= BIOS_CTRL_BIOSWE;
bios_cntl &= ~BIT(5);
}
dm_pci_write_config8(dev, BIOS_CTRL, bios_cntl);
return 0;
}
static const struct pch_ops bd82x6x_pch_ops = {
.get_sbase = bd82x6x_pch_get_sbase,
.get_version = bd82x6x_pch_get_version,
.set_spi_protect = bd82x6x_set_spi_protect,
};
static const struct udevice_id bd82x6x_ids[] = { static const struct udevice_id bd82x6x_ids[] = {
{ .compatible = "intel,bd82x6x" }, { .compatible = "intel,bd82x6x" },
{ } { }
@ -126,4 +170,5 @@ U_BOOT_DRIVER(bd82x6x_drv) = {
.id = UCLASS_PCH, .id = UCLASS_PCH,
.of_match = bd82x6x_ids, .of_match = bd82x6x_ids,
.probe = bd82x6x_probe, .probe = bd82x6x_probe,
.ops = &bd82x6x_pch_ops,
}; };

@ -65,23 +65,6 @@
}; };
}; };
spi {
#address-cells = <1>;
#size-cells = <0>;
compatible = "intel,ich-spi";
spi-flash@0 {
#address-cells = <1>;
#size-cells = <1>;
reg = <0>;
compatible = "winbond,w25q64dw", "spi-flash";
memory-map = <0xff800000 0x00800000>;
rw-mrc-cache {
label = "rw-mrc-cache";
reg = <0x006e0000 0x00010000>;
};
};
};
gpioa { gpioa {
compatible = "intel,ich6-gpio"; compatible = "intel,ich6-gpio";
u-boot,dm-pre-reloc; u-boot,dm-pre-reloc;
@ -133,66 +116,91 @@
0x42000000 0x0 0xc0000000 0xc0000000 0 0x20000000 0x42000000 0x0 0xc0000000 0xc0000000 0 0x20000000
0x01000000 0x0 0x2000 0x2000 0 0xe000>; 0x01000000 0x0 0x2000 0x2000 0 0xe000>;
irq-router@1f,0 { pch@1f,0 {
reg = <0x0000f800 0 0 0 0>; reg = <0x0000f800 0 0 0 0>;
compatible = "intel,irq-router"; compatible = "intel,pch9";
intel,pirq-config = "ibase";
intel,ibase-offset = <0x50>; irq-router {
intel,pirq-link = <8 8>; compatible = "intel,irq-router";
intel,pirq-mask = <0xdee0>; intel,pirq-config = "ibase";
intel,pirq-routing = < intel,ibase-offset = <0x50>;
/* BayTrail PCI devices */ intel,pirq-link = <8 8>;
PCI_BDF(0, 2, 0) INTA PIRQA intel,pirq-mask = <0xdee0>;
PCI_BDF(0, 3, 0) INTA PIRQA intel,pirq-routing = <
PCI_BDF(0, 16, 0) INTA PIRQA /* BayTrail PCI devices */
PCI_BDF(0, 17, 0) INTA PIRQA PCI_BDF(0, 2, 0) INTA PIRQA
PCI_BDF(0, 18, 0) INTA PIRQA PCI_BDF(0, 3, 0) INTA PIRQA
PCI_BDF(0, 19, 0) INTA PIRQA PCI_BDF(0, 16, 0) INTA PIRQA
PCI_BDF(0, 20, 0) INTA PIRQA PCI_BDF(0, 17, 0) INTA PIRQA
PCI_BDF(0, 21, 0) INTA PIRQA PCI_BDF(0, 18, 0) INTA PIRQA
PCI_BDF(0, 22, 0) INTA PIRQA PCI_BDF(0, 19, 0) INTA PIRQA
PCI_BDF(0, 23, 0) INTA PIRQA PCI_BDF(0, 20, 0) INTA PIRQA
PCI_BDF(0, 24, 0) INTA PIRQA PCI_BDF(0, 21, 0) INTA PIRQA
PCI_BDF(0, 24, 1) INTC PIRQC PCI_BDF(0, 22, 0) INTA PIRQA
PCI_BDF(0, 24, 2) INTD PIRQD PCI_BDF(0, 23, 0) INTA PIRQA
PCI_BDF(0, 24, 3) INTB PIRQB PCI_BDF(0, 24, 0) INTA PIRQA
PCI_BDF(0, 24, 4) INTA PIRQA PCI_BDF(0, 24, 1) INTC PIRQC
PCI_BDF(0, 24, 5) INTC PIRQC PCI_BDF(0, 24, 2) INTD PIRQD
PCI_BDF(0, 24, 6) INTD PIRQD PCI_BDF(0, 24, 3) INTB PIRQB
PCI_BDF(0, 24, 7) INTB PIRQB PCI_BDF(0, 24, 4) INTA PIRQA
PCI_BDF(0, 26, 0) INTA PIRQA PCI_BDF(0, 24, 5) INTC PIRQC
PCI_BDF(0, 27, 0) INTA PIRQA PCI_BDF(0, 24, 6) INTD PIRQD
PCI_BDF(0, 28, 0) INTA PIRQA PCI_BDF(0, 24, 7) INTB PIRQB
PCI_BDF(0, 28, 1) INTB PIRQB PCI_BDF(0, 26, 0) INTA PIRQA
PCI_BDF(0, 28, 2) INTC PIRQC PCI_BDF(0, 27, 0) INTA PIRQA
PCI_BDF(0, 28, 3) INTD PIRQD PCI_BDF(0, 28, 0) INTA PIRQA
PCI_BDF(0, 29, 0) INTA PIRQA PCI_BDF(0, 28, 1) INTB PIRQB
PCI_BDF(0, 30, 0) INTA PIRQA PCI_BDF(0, 28, 2) INTC PIRQC
PCI_BDF(0, 30, 1) INTD PIRQD PCI_BDF(0, 28, 3) INTD PIRQD
PCI_BDF(0, 30, 2) INTB PIRQB PCI_BDF(0, 29, 0) INTA PIRQA
PCI_BDF(0, 30, 3) INTC PIRQC PCI_BDF(0, 30, 0) INTA PIRQA
PCI_BDF(0, 30, 4) INTD PIRQD PCI_BDF(0, 30, 1) INTD PIRQD
PCI_BDF(0, 30, 5) INTB PIRQB PCI_BDF(0, 30, 2) INTB PIRQB
PCI_BDF(0, 31, 3) INTB PIRQB PCI_BDF(0, 30, 3) INTC PIRQC
PCI_BDF(0, 30, 4) INTD PIRQD
/* PCIe root ports downstream interrupts */ PCI_BDF(0, 30, 5) INTB PIRQB
PCI_BDF(1, 0, 0) INTA PIRQA PCI_BDF(0, 31, 3) INTB PIRQB
PCI_BDF(1, 0, 0) INTB PIRQB
PCI_BDF(1, 0, 0) INTC PIRQC /*
PCI_BDF(1, 0, 0) INTD PIRQD * PCIe root ports downstream
PCI_BDF(2, 0, 0) INTA PIRQB * interrupts
PCI_BDF(2, 0, 0) INTB PIRQC */
PCI_BDF(2, 0, 0) INTC PIRQD PCI_BDF(1, 0, 0) INTA PIRQA
PCI_BDF(2, 0, 0) INTD PIRQA PCI_BDF(1, 0, 0) INTB PIRQB
PCI_BDF(3, 0, 0) INTA PIRQC PCI_BDF(1, 0, 0) INTC PIRQC
PCI_BDF(3, 0, 0) INTB PIRQD PCI_BDF(1, 0, 0) INTD PIRQD
PCI_BDF(3, 0, 0) INTC PIRQA PCI_BDF(2, 0, 0) INTA PIRQB
PCI_BDF(3, 0, 0) INTD PIRQB PCI_BDF(2, 0, 0) INTB PIRQC
PCI_BDF(4, 0, 0) INTA PIRQD PCI_BDF(2, 0, 0) INTC PIRQD
PCI_BDF(4, 0, 0) INTB PIRQA PCI_BDF(2, 0, 0) INTD PIRQA
PCI_BDF(4, 0, 0) INTC PIRQB PCI_BDF(3, 0, 0) INTA PIRQC
PCI_BDF(4, 0, 0) INTD PIRQC PCI_BDF(3, 0, 0) INTB PIRQD
>; PCI_BDF(3, 0, 0) INTC PIRQA
PCI_BDF(3, 0, 0) INTD PIRQB
PCI_BDF(4, 0, 0) INTA PIRQD
PCI_BDF(4, 0, 0) INTB PIRQA
PCI_BDF(4, 0, 0) INTC PIRQB
PCI_BDF(4, 0, 0) INTD PIRQC
>;
};
spi {
#address-cells = <1>;
#size-cells = <0>;
compatible = "intel,ich-spi";
spi-flash@0 {
#address-cells = <1>;
#size-cells = <1>;
reg = <0>;
compatible = "winbond,w25q64dw",
"spi-flash";
memory-map = <0xff800000 0x00800000>;
rw-mrc-cache {
label = "rw-mrc-cache";
reg = <0x006e0000 0x00010000>;
};
};
};
}; };
}; };

@ -29,16 +29,22 @@
ranges = <0x02000000 0x0 0xe0000000 0xe0000000 0 0x10000000 ranges = <0x02000000 0x0 0xe0000000 0xe0000000 0 0x10000000
0x42000000 0x0 0xd0000000 0xd0000000 0 0x10000000 0x42000000 0x0 0xd0000000 0xd0000000 0 0x10000000
0x01000000 0x0 0x2000 0x2000 0 0xe000>; 0x01000000 0x0 0x2000 0x2000 0 0xe000>;
};
spi { pch@1f,0 {
#address-cells = <1>; reg = <0x0000f800 0 0 0 0>;
#size-cells = <0>; compatible = "intel,pch9";
compatible = "intel,ich-spi";
spi-flash@0 { spi {
reg = <0>; #address-cells = <1>;
compatible = "winbond,w25q128", "spi-flash"; #size-cells = <0>;
memory-map = <0xff000000 0x01000000>; compatible = "intel,ich-spi";
spi-flash@0 {
reg = <0>;
compatible = "winbond,w25q128", "spi-flash";
memory-map = <0xff000000 0x01000000>;
};
};
}; };
}; };
}; };

@ -186,9 +186,9 @@
intel,pch-backlight = <0x04000000>; intel,pch-backlight = <0x04000000>;
}; };
pch { pch@1f,0 {
reg = <0x0000f800 0 0 0 0>; reg = <0x0000f800 0 0 0 0>;
compatible = "intel,bd82x6x", "intel,pch"; compatible = "intel,bd82x6x", "intel,pch9";
u-boot,dm-pre-reloc; u-boot,dm-pre-reloc;
#address-cells = <1>; #address-cells = <1>;
#size-cells = <1>; #size-cells = <1>;
@ -200,6 +200,7 @@
1 0 0 0 0 0 0 0>; 1 0 0 0 0 0 0 0>;
/* Enable EC SMI source */ /* Enable EC SMI source */
intel,alt-gp-smi-enable = <0x0100>; intel,alt-gp-smi-enable = <0x0100>;
spi { spi {
#address-cells = <1>; #address-cells = <1>;
#size-cells = <0>; #size-cells = <0>;

@ -51,21 +51,27 @@
ranges = <0x02000000 0x0 0xe0000000 0xe0000000 0 0x10000000 ranges = <0x02000000 0x0 0xe0000000 0xe0000000 0 0x10000000
0x42000000 0x0 0xd0000000 0xd0000000 0 0x10000000 0x42000000 0x0 0xd0000000 0xd0000000 0 0x10000000
0x01000000 0x0 0x1000 0x1000 0 0xf000>; 0x01000000 0x0 0x1000 0x1000 0 0xf000>;
};
spi { pch@1f,0 {
#address-cells = <1>; reg = <0x0000f800 0 0 0 0>;
#size-cells = <0>; compatible = "intel,pch9";
compatible = "intel,ich-spi";
spi-flash@0 { spi {
#size-cells = <1>; #address-cells = <1>;
#address-cells = <1>; #size-cells = <0>;
reg = <0>; compatible = "intel,ich-spi";
compatible = "winbond,w25q64", "spi-flash"; spi-flash@0 {
memory-map = <0xff800000 0x00800000>; #size-cells = <1>;
rw-mrc-cache { #address-cells = <1>;
label = "rw-mrc-cache"; reg = <0>;
reg = <0x003e0000 0x00010000>; compatible = "winbond,w25q64",
"spi-flash";
memory-map = <0xff800000 0x00800000>;
rw-mrc-cache {
label = "rw-mrc-cache";
reg = <0x003e0000 0x00010000>;
};
};
}; };
}; };
}; };

@ -72,17 +72,6 @@
stdout-path = "/serial"; stdout-path = "/serial";
}; };
spi {
#address-cells = <1>;
#size-cells = <0>;
compatible = "intel,ich-spi";
spi-flash@0 {
reg = <0>;
compatible = "sst,25vf016b", "spi-flash";
memory-map = <0xffe00000 0x00200000>;
};
};
microcode { microcode {
update@0 { update@0 {
#include "microcode/m0220661105_cv.dtsi" #include "microcode/m0220661105_cv.dtsi"
@ -170,68 +159,85 @@
}; };
}; };
irq-router@1f,0 { pch@1f,0 {
reg = <0x0000f800 0 0 0 0>; reg = <0x0000f800 0 0 0 0>;
compatible = "intel,irq-router"; compatible = "intel,pch7";
intel,pirq-config = "pci";
intel,pirq-link = <0x60 8>; irq-router {
intel,pirq-mask = <0xcee0>; compatible = "intel,irq-router";
intel,pirq-routing = < intel,pirq-config = "pci";
/* TunnelCreek PCI devices */ intel,pirq-link = <0x60 8>;
PCI_BDF(0, 2, 0) INTA PIRQE intel,pirq-mask = <0xcee0>;
PCI_BDF(0, 3, 0) INTA PIRQF intel,pirq-routing = <
PCI_BDF(0, 23, 0) INTA PIRQA /* TunnelCreek PCI devices */
PCI_BDF(0, 23, 0) INTB PIRQB PCI_BDF(0, 2, 0) INTA PIRQE
PCI_BDF(0, 23, 0) INTC PIRQC PCI_BDF(0, 3, 0) INTA PIRQF
PCI_BDF(0, 23, 0) INTD PIRQD PCI_BDF(0, 23, 0) INTA PIRQA
PCI_BDF(0, 24, 0) INTA PIRQB PCI_BDF(0, 23, 0) INTB PIRQB
PCI_BDF(0, 24, 0) INTB PIRQC PCI_BDF(0, 23, 0) INTC PIRQC
PCI_BDF(0, 24, 0) INTC PIRQD PCI_BDF(0, 23, 0) INTD PIRQD
PCI_BDF(0, 24, 0) INTD PIRQA PCI_BDF(0, 24, 0) INTA PIRQB
PCI_BDF(0, 25, 0) INTA PIRQC PCI_BDF(0, 24, 0) INTB PIRQC
PCI_BDF(0, 25, 0) INTB PIRQD PCI_BDF(0, 24, 0) INTC PIRQD
PCI_BDF(0, 25, 0) INTC PIRQA PCI_BDF(0, 24, 0) INTD PIRQA
PCI_BDF(0, 25, 0) INTD PIRQB PCI_BDF(0, 25, 0) INTA PIRQC
PCI_BDF(0, 26, 0) INTA PIRQD PCI_BDF(0, 25, 0) INTB PIRQD
PCI_BDF(0, 26, 0) INTB PIRQA PCI_BDF(0, 25, 0) INTC PIRQA
PCI_BDF(0, 26, 0) INTC PIRQB PCI_BDF(0, 25, 0) INTD PIRQB
PCI_BDF(0, 26, 0) INTD PIRQC PCI_BDF(0, 26, 0) INTA PIRQD
PCI_BDF(0, 27, 0) INTA PIRQG PCI_BDF(0, 26, 0) INTB PIRQA
/* PCI_BDF(0, 26, 0) INTC PIRQB
* Topcliff PCI devices PCI_BDF(0, 26, 0) INTD PIRQC
* PCI_BDF(0, 27, 0) INTA PIRQG
* Note on the Crown Bay board, Topcliff chipset /*
* is connected to TunnelCreek PCIe port 0, so * Topcliff PCI devices
* its bus number is 1 for its PCIe port and 2 *
* for its PCI devices per U-Boot current PCI * Note on the Crown Bay board, Topcliff
* bus enumeration algorithm. * chipset is connected to TunnelCreek
*/ * PCIe port 0, so its bus number is 1
PCI_BDF(1, 0, 0) INTA PIRQA * for its PCIe port and 2 for its PCI
PCI_BDF(2, 0, 1) INTA PIRQA * devices per U-Boot current PCI bus
PCI_BDF(2, 0, 2) INTA PIRQA * enumeration algorithm.
PCI_BDF(2, 2, 0) INTB PIRQD */
PCI_BDF(2, 2, 1) INTB PIRQD PCI_BDF(1, 0, 0) INTA PIRQA
PCI_BDF(2, 2, 2) INTB PIRQD PCI_BDF(2, 0, 1) INTA PIRQA
PCI_BDF(2, 2, 3) INTB PIRQD PCI_BDF(2, 0, 2) INTA PIRQA
PCI_BDF(2, 2, 4) INTB PIRQD PCI_BDF(2, 2, 0) INTB PIRQD
PCI_BDF(2, 4, 0) INTC PIRQC PCI_BDF(2, 2, 1) INTB PIRQD
PCI_BDF(2, 4, 1) INTC PIRQC PCI_BDF(2, 2, 2) INTB PIRQD
PCI_BDF(2, 6, 0) INTD PIRQB PCI_BDF(2, 2, 3) INTB PIRQD
PCI_BDF(2, 8, 0) INTA PIRQA PCI_BDF(2, 2, 4) INTB PIRQD
PCI_BDF(2, 8, 1) INTA PIRQA PCI_BDF(2, 4, 0) INTC PIRQC
PCI_BDF(2, 8, 2) INTA PIRQA PCI_BDF(2, 4, 1) INTC PIRQC
PCI_BDF(2, 8, 3) INTA PIRQA PCI_BDF(2, 6, 0) INTD PIRQB
PCI_BDF(2, 10, 0) INTB PIRQD PCI_BDF(2, 8, 0) INTA PIRQA
PCI_BDF(2, 10, 1) INTB PIRQD PCI_BDF(2, 8, 1) INTA PIRQA
PCI_BDF(2, 10, 2) INTB PIRQD PCI_BDF(2, 8, 2) INTA PIRQA
PCI_BDF(2, 10, 3) INTB PIRQD PCI_BDF(2, 8, 3) INTA PIRQA
PCI_BDF(2, 10, 4) INTB PIRQD PCI_BDF(2, 10, 0) INTB PIRQD
PCI_BDF(2, 12, 0) INTC PIRQC PCI_BDF(2, 10, 1) INTB PIRQD
PCI_BDF(2, 12, 1) INTC PIRQC PCI_BDF(2, 10, 2) INTB PIRQD
PCI_BDF(2, 12, 2) INTC PIRQC PCI_BDF(2, 10, 3) INTB PIRQD
PCI_BDF(2, 12, 3) INTC PIRQC PCI_BDF(2, 10, 4) INTB PIRQD
PCI_BDF(2, 12, 4) INTC PIRQC PCI_BDF(2, 12, 0) INTC PIRQC
>; PCI_BDF(2, 12, 1) INTC PIRQC
PCI_BDF(2, 12, 2) INTC PIRQC
PCI_BDF(2, 12, 3) INTC PIRQC
PCI_BDF(2, 12, 4) INTC PIRQC
>;
};
spi {
#address-cells = <1>;
#size-cells = <0>;
compatible = "intel,ich-spi";
spi-flash@0 {
reg = <0>;
compatible = "sst,25vf016b",
"spi-flash";
memory-map = <0xffe00000 0x00200000>;
};
};
}; };
}; };

@ -79,37 +79,59 @@
current-speed = <115200>; current-speed = <115200>;
}; };
irq-router@1f,0 { pch@1f,0 {
reg = <0x0000f800 0 0 0 0>; reg = <0x0000f800 0 0 0 0>;
compatible = "intel,irq-router"; compatible = "intel,pch7";
intel,pirq-config = "pci";
intel,pirq-link = <0x60 8>; irq-router {
intel,pirq-mask = <0xdef8>; compatible = "intel,irq-router";
intel,pirq-routing = < intel,pirq-config = "pci";
PCI_BDF(0, 20, 0) INTA PIRQE intel,pirq-link = <0x60 8>;
PCI_BDF(0, 20, 1) INTB PIRQF intel,pirq-mask = <0xdef8>;
PCI_BDF(0, 20, 2) INTC PIRQG intel,pirq-routing = <
PCI_BDF(0, 20, 3) INTD PIRQH PCI_BDF(0, 20, 0) INTA PIRQE
PCI_BDF(0, 20, 4) INTA PIRQE PCI_BDF(0, 20, 1) INTB PIRQF
PCI_BDF(0, 20, 5) INTB PIRQF PCI_BDF(0, 20, 2) INTC PIRQG
PCI_BDF(0, 20, 6) INTC PIRQG PCI_BDF(0, 20, 3) INTD PIRQH
PCI_BDF(0, 20, 7) INTD PIRQH PCI_BDF(0, 20, 4) INTA PIRQE
PCI_BDF(0, 21, 0) INTA PIRQE PCI_BDF(0, 20, 5) INTB PIRQF
PCI_BDF(0, 21, 1) INTB PIRQF PCI_BDF(0, 20, 6) INTC PIRQG
PCI_BDF(0, 21, 2) INTC PIRQG PCI_BDF(0, 20, 7) INTD PIRQH
PCI_BDF(0, 23, 0) INTA PIRQA PCI_BDF(0, 21, 0) INTA PIRQE
PCI_BDF(0, 23, 1) INTB PIRQB PCI_BDF(0, 21, 1) INTB PIRQF
PCI_BDF(0, 21, 2) INTC PIRQG
/* PCIe root ports downstream interrupts */ PCI_BDF(0, 23, 0) INTA PIRQA
PCI_BDF(1, 0, 0) INTA PIRQA PCI_BDF(0, 23, 1) INTB PIRQB
PCI_BDF(1, 0, 0) INTB PIRQB
PCI_BDF(1, 0, 0) INTC PIRQC /* PCIe root ports downstream interrupts */
PCI_BDF(1, 0, 0) INTD PIRQD PCI_BDF(1, 0, 0) INTA PIRQA
PCI_BDF(2, 0, 0) INTA PIRQB PCI_BDF(1, 0, 0) INTB PIRQB
PCI_BDF(2, 0, 0) INTB PIRQC PCI_BDF(1, 0, 0) INTC PIRQC
PCI_BDF(2, 0, 0) INTC PIRQD PCI_BDF(1, 0, 0) INTD PIRQD
PCI_BDF(2, 0, 0) INTD PIRQA PCI_BDF(2, 0, 0) INTA PIRQB
>; PCI_BDF(2, 0, 0) INTB PIRQC
PCI_BDF(2, 0, 0) INTC PIRQD
PCI_BDF(2, 0, 0) INTD PIRQA
>;
};
spi {
#address-cells = <1>;
#size-cells = <0>;
compatible = "intel,ich-spi";
spi-flash@0 {
#size-cells = <1>;
#address-cells = <1>;
reg = <0>;
compatible = "winbond,w25q64",
"spi-flash";
memory-map = <0xff800000 0x00800000>;
rw-mrc-cache {
label = "rw-mrc-cache";
reg = <0x00010000 0x00010000>;
};
};
};
}; };
}; };
@ -127,21 +149,4 @@
bank-name = "B"; bank-name = "B";
}; };
spi {
#address-cells = <1>;
#size-cells = <0>;
compatible = "intel,ich-spi";
spi-flash@0 {
#size-cells = <1>;
#address-cells = <1>;
reg = <0>;
compatible = "winbond,w25q64", "spi-flash";
memory-map = <0xff800000 0x00800000>;
rw-mrc-cache {
label = "rw-mrc-cache";
reg = <0x00010000 0x00010000>;
};
};
};
}; };

@ -150,66 +150,91 @@
0x42000000 0x0 0xc0000000 0xc0000000 0 0x20000000 0x42000000 0x0 0xc0000000 0xc0000000 0 0x20000000
0x01000000 0x0 0x2000 0x2000 0 0xe000>; 0x01000000 0x0 0x2000 0x2000 0 0xe000>;
irq-router@1f,0 { pch@1f,0 {
reg = <0x0000f800 0 0 0 0>; reg = <0x0000f800 0 0 0 0>;
compatible = "intel,irq-router"; compatible = "pci8086,0f1c", "intel,pch9";
intel,pirq-config = "ibase";
intel,ibase-offset = <0x50>;
intel,pirq-link = <8 8>;
intel,pirq-mask = <0xdee0>;
intel,pirq-routing = <
/* BayTrail PCI devices */
PCI_BDF(0, 2, 0) INTA PIRQA
PCI_BDF(0, 3, 0) INTA PIRQA
PCI_BDF(0, 16, 0) INTA PIRQA
PCI_BDF(0, 17, 0) INTA PIRQA
PCI_BDF(0, 18, 0) INTA PIRQA
PCI_BDF(0, 19, 0) INTA PIRQA
PCI_BDF(0, 20, 0) INTA PIRQA
PCI_BDF(0, 21, 0) INTA PIRQA
PCI_BDF(0, 22, 0) INTA PIRQA
PCI_BDF(0, 23, 0) INTA PIRQA
PCI_BDF(0, 24, 0) INTA PIRQA
PCI_BDF(0, 24, 1) INTC PIRQC
PCI_BDF(0, 24, 2) INTD PIRQD
PCI_BDF(0, 24, 3) INTB PIRQB
PCI_BDF(0, 24, 4) INTA PIRQA
PCI_BDF(0, 24, 5) INTC PIRQC
PCI_BDF(0, 24, 6) INTD PIRQD
PCI_BDF(0, 24, 7) INTB PIRQB
PCI_BDF(0, 26, 0) INTA PIRQA
PCI_BDF(0, 27, 0) INTA PIRQA
PCI_BDF(0, 28, 0) INTA PIRQA
PCI_BDF(0, 28, 1) INTB PIRQB
PCI_BDF(0, 28, 2) INTC PIRQC
PCI_BDF(0, 28, 3) INTD PIRQD
PCI_BDF(0, 29, 0) INTA PIRQA
PCI_BDF(0, 30, 0) INTA PIRQA
PCI_BDF(0, 30, 1) INTD PIRQD
PCI_BDF(0, 30, 2) INTB PIRQB
PCI_BDF(0, 30, 3) INTC PIRQC
PCI_BDF(0, 30, 4) INTD PIRQD
PCI_BDF(0, 30, 5) INTB PIRQB
PCI_BDF(0, 31, 3) INTB PIRQB
/* PCIe root ports downstream interrupts */ irq-router {
PCI_BDF(1, 0, 0) INTA PIRQA compatible = "intel,irq-router";
PCI_BDF(1, 0, 0) INTB PIRQB intel,pirq-config = "ibase";
PCI_BDF(1, 0, 0) INTC PIRQC intel,ibase-offset = <0x50>;
PCI_BDF(1, 0, 0) INTD PIRQD intel,pirq-link = <8 8>;
PCI_BDF(2, 0, 0) INTA PIRQB intel,pirq-mask = <0xdee0>;
PCI_BDF(2, 0, 0) INTB PIRQC intel,pirq-routing = <
PCI_BDF(2, 0, 0) INTC PIRQD /* BayTrail PCI devices */
PCI_BDF(2, 0, 0) INTD PIRQA PCI_BDF(0, 2, 0) INTA PIRQA
PCI_BDF(3, 0, 0) INTA PIRQC PCI_BDF(0, 3, 0) INTA PIRQA
PCI_BDF(3, 0, 0) INTB PIRQD PCI_BDF(0, 16, 0) INTA PIRQA
PCI_BDF(3, 0, 0) INTC PIRQA PCI_BDF(0, 17, 0) INTA PIRQA
PCI_BDF(3, 0, 0) INTD PIRQB PCI_BDF(0, 18, 0) INTA PIRQA
PCI_BDF(4, 0, 0) INTA PIRQD PCI_BDF(0, 19, 0) INTA PIRQA
PCI_BDF(4, 0, 0) INTB PIRQA PCI_BDF(0, 20, 0) INTA PIRQA
PCI_BDF(4, 0, 0) INTC PIRQB PCI_BDF(0, 21, 0) INTA PIRQA
PCI_BDF(4, 0, 0) INTD PIRQC PCI_BDF(0, 22, 0) INTA PIRQA
>; PCI_BDF(0, 23, 0) INTA PIRQA
PCI_BDF(0, 24, 0) INTA PIRQA
PCI_BDF(0, 24, 1) INTC PIRQC
PCI_BDF(0, 24, 2) INTD PIRQD
PCI_BDF(0, 24, 3) INTB PIRQB
PCI_BDF(0, 24, 4) INTA PIRQA
PCI_BDF(0, 24, 5) INTC PIRQC
PCI_BDF(0, 24, 6) INTD PIRQD
PCI_BDF(0, 24, 7) INTB PIRQB
PCI_BDF(0, 26, 0) INTA PIRQA
PCI_BDF(0, 27, 0) INTA PIRQA
PCI_BDF(0, 28, 0) INTA PIRQA
PCI_BDF(0, 28, 1) INTB PIRQB
PCI_BDF(0, 28, 2) INTC PIRQC
PCI_BDF(0, 28, 3) INTD PIRQD
PCI_BDF(0, 29, 0) INTA PIRQA
PCI_BDF(0, 30, 0) INTA PIRQA
PCI_BDF(0, 30, 1) INTD PIRQD
PCI_BDF(0, 30, 2) INTB PIRQB
PCI_BDF(0, 30, 3) INTC PIRQC
PCI_BDF(0, 30, 4) INTD PIRQD
PCI_BDF(0, 30, 5) INTB PIRQB
PCI_BDF(0, 31, 3) INTB PIRQB
/*
* PCIe root ports downstream
* interrupts
*/
PCI_BDF(1, 0, 0) INTA PIRQA
PCI_BDF(1, 0, 0) INTB PIRQB
PCI_BDF(1, 0, 0) INTC PIRQC
PCI_BDF(1, 0, 0) INTD PIRQD
PCI_BDF(2, 0, 0) INTA PIRQB
PCI_BDF(2, 0, 0) INTB PIRQC
PCI_BDF(2, 0, 0) INTC PIRQD
PCI_BDF(2, 0, 0) INTD PIRQA
PCI_BDF(3, 0, 0) INTA PIRQC
PCI_BDF(3, 0, 0) INTB PIRQD
PCI_BDF(3, 0, 0) INTC PIRQA
PCI_BDF(3, 0, 0) INTD PIRQB
PCI_BDF(4, 0, 0) INTA PIRQD
PCI_BDF(4, 0, 0) INTB PIRQA
PCI_BDF(4, 0, 0) INTC PIRQB
PCI_BDF(4, 0, 0) INTD PIRQC
>;
};
spi {
#address-cells = <1>;
#size-cells = <0>;
compatible = "intel,ich-spi";
spi-flash@0 {
#address-cells = <1>;
#size-cells = <1>;
reg = <0>;
compatible = "stmicro,n25q064a",
"spi-flash";
memory-map = <0xff800000 0x00800000>;
rw-mrc-cache {
label = "rw-mrc-cache";
reg = <0x006f0000 0x00010000>;
};
};
};
}; };
}; };
@ -269,23 +294,6 @@
}; };
}; };
spi {
#address-cells = <1>;
#size-cells = <0>;
compatible = "intel,ich-spi";
spi-flash@0 {
#address-cells = <1>;
#size-cells = <1>;
reg = <0>;
compatible = "stmicro,n25q064a", "spi-flash";
memory-map = <0xff800000 0x00800000>;
rw-mrc-cache {
label = "rw-mrc-cache";
reg = <0x006f0000 0x00010000>;
};
};
};
microcode { microcode {
update@0 { update@0 {
#include "microcode/m0130673322.dtsi" #include "microcode/m0130673322.dtsi"

@ -51,18 +51,22 @@
0x42000000 0x0 0xd0000000 0xd0000000 0 0x10000000 0x42000000 0x0 0xd0000000 0xd0000000 0 0x10000000
0x01000000 0x0 0x2000 0x2000 0 0xe000>; 0x01000000 0x0 0x2000 0x2000 0 0xe000>;
irq-router@1,0 { pch@1,0 {
reg = <0x00000800 0 0 0 0>; reg = <0x00000800 0 0 0 0>;
compatible = "intel,irq-router"; compatible = "intel,pch7";
intel,pirq-config = "pci";
intel,pirq-link = <0x60 4>; irq-router {
intel,pirq-mask = <0x0e40>; compatible = "intel,irq-router";
intel,pirq-routing = < intel,pirq-config = "pci";
/* PIIX UHCI */ intel,pirq-link = <0x60 4>;
PCI_BDF(0, 1, 2) INTD PIRQD intel,pirq-mask = <0x0e40>;
/* e1000 NIC */ intel,pirq-routing = <
PCI_BDF(0, 3, 0) INTA PIRQC /* PIIX UHCI */
>; PCI_BDF(0, 1, 2) INTD PIRQD
/* e1000 NIC */
PCI_BDF(0, 3, 0) INTA PIRQC
>;
};
}; };
}; };

@ -62,24 +62,28 @@
0x42000000 0x0 0xd0000000 0xd0000000 0 0x10000000 0x42000000 0x0 0xd0000000 0xd0000000 0 0x10000000
0x01000000 0x0 0x2000 0x2000 0 0xe000>; 0x01000000 0x0 0x2000 0x2000 0 0xe000>;
irq-router@1f,0 { pch@1f,0 {
reg = <0x0000f800 0 0 0 0>; reg = <0x0000f800 0 0 0 0>;
compatible = "intel,irq-router"; compatible = "intel,pch9";
intel,pirq-config = "pci";
intel,pirq-link = <0x60 8>; irq-router {
intel,pirq-mask = <0x0e40>; compatible = "intel,irq-router";
intel,pirq-routing = < intel,pirq-config = "pci";
/* e1000 NIC */ intel,pirq-link = <0x60 8>;
PCI_BDF(0, 2, 0) INTA PIRQG intel,pirq-mask = <0x0e40>;
/* ICH9 UHCI */ intel,pirq-routing = <
PCI_BDF(0, 29, 0) INTA PIRQA /* e1000 NIC */
PCI_BDF(0, 29, 1) INTB PIRQB PCI_BDF(0, 2, 0) INTA PIRQG
PCI_BDF(0, 29, 2) INTC PIRQC /* ICH9 UHCI */
/* ICH9 EHCI */ PCI_BDF(0, 29, 0) INTA PIRQA
PCI_BDF(0, 29, 7) INTD PIRQD PCI_BDF(0, 29, 1) INTB PIRQB
/* ICH9 SATA */ PCI_BDF(0, 29, 2) INTC PIRQC
PCI_BDF(0, 31, 2) INTA PIRQA /* ICH9 EHCI */
>; PCI_BDF(0, 29, 7) INTD PIRQD
/* ICH9 SATA */
PCI_BDF(0, 31, 2) INTA PIRQA
>;
};
}; };
}; };

@ -5,14 +5,14 @@
* *
* This file is derived from the flashrom project. * This file is derived from the flashrom project.
*/ */
#include <common.h> #include <common.h>
#include <dm.h> #include <dm.h>
#include <errno.h> #include <errno.h>
#include <malloc.h> #include <malloc.h>
#include <spi.h> #include <pch.h>
#include <pci.h> #include <pci.h>
#include <pci_ids.h> #include <pci_ids.h>
#include <spi.h>
#include <asm/io.h> #include <asm/io.h>
#include "ich.h" #include "ich.h"
@ -27,9 +27,7 @@
#endif #endif
struct ich_spi_platdata { struct ich_spi_platdata {
pci_dev_t dev; /* PCI device number */ enum pch_version ich_version; /* Controller version, 7 or 9 */
int ich_version; /* Controller version, 7 or 9 */
bool use_sbase; /* Use SBASE instead of RCB */
}; };
struct ich_spi_priv { struct ich_spi_priv {
@ -122,40 +120,16 @@ static void ich_set_bbar(struct ich_spi_priv *ctlr, uint32_t minaddr)
ich_writel(ctlr, ichspi_bbar, ctlr->bbar); ich_writel(ctlr, ichspi_bbar, ctlr->bbar);
} }
/*
* Check if this device ID matches one of supported Intel PCH devices.
*
* Return the ICH version if there is a match, or zero otherwise.
*/
static int get_ich_version(uint16_t device_id)
{
if (device_id == PCI_DEVICE_ID_INTEL_TGP_LPC ||
device_id == PCI_DEVICE_ID_INTEL_ITC_LPC ||
device_id == PCI_DEVICE_ID_INTEL_QRK_ILB)
return 7;
if ((device_id >= PCI_DEVICE_ID_INTEL_COUGARPOINT_LPC_MIN &&
device_id <= PCI_DEVICE_ID_INTEL_COUGARPOINT_LPC_MAX) ||
(device_id >= PCI_DEVICE_ID_INTEL_PANTHERPOINT_LPC_MIN &&
device_id <= PCI_DEVICE_ID_INTEL_PANTHERPOINT_LPC_MAX) ||
device_id == PCI_DEVICE_ID_INTEL_VALLEYVIEW_LPC ||
device_id == PCI_DEVICE_ID_INTEL_LYNXPOINT_LPC ||
device_id == PCI_DEVICE_ID_INTEL_WILDCATPOINT_LPC)
return 9;
return 0;
}
/* @return 1 if the SPI flash supports the 33MHz speed */ /* @return 1 if the SPI flash supports the 33MHz speed */
static int ich9_can_do_33mhz(pci_dev_t dev) static int ich9_can_do_33mhz(struct udevice *dev)
{ {
u32 fdod, speed; u32 fdod, speed;
/* Observe SPI Descriptor Component Section 0 */ /* Observe SPI Descriptor Component Section 0 */
pci_write_config_dword(dev, 0xb0, 0x1000); dm_pci_write_config32(dev->parent, 0xb0, 0x1000);
/* Extract the Write/Erase SPI Frequency from descriptor */ /* Extract the Write/Erase SPI Frequency from descriptor */
pci_read_config_dword(dev, 0xb4, &fdod); dm_pci_read_config32(dev->parent, 0xb4, &fdod);
/* Bits 23:21 have the fast read clock frequency, 0=20MHz, 1=33MHz */ /* Bits 23:21 have the fast read clock frequency, 0=20MHz, 1=33MHz */
speed = (fdod >> 21) & 7; speed = (fdod >> 21) & 7;
@ -163,59 +137,22 @@ static int ich9_can_do_33mhz(pci_dev_t dev)
return speed == 1; return speed == 1;
} }
static int ich_find_spi_controller(struct ich_spi_platdata *ich) static int ich_init_controller(struct udevice *dev,
{ struct ich_spi_platdata *plat,
int last_bus = pci_last_busno();
int bus;
if (last_bus == -1) {
debug("No PCI busses?\n");
return -ENODEV;
}
for (bus = 0; bus <= last_bus; bus++) {
uint16_t vendor_id, device_id;
uint32_t ids;
pci_dev_t dev;
dev = PCI_BDF(bus, 31, 0);
pci_read_config_dword(dev, 0, &ids);
vendor_id = ids;
device_id = ids >> 16;
if (vendor_id == PCI_VENDOR_ID_INTEL) {
ich->dev = dev;
ich->ich_version = get_ich_version(device_id);
if (device_id == PCI_DEVICE_ID_INTEL_VALLEYVIEW_LPC)
ich->use_sbase = true;
return ich->ich_version == 0 ? -ENODEV : 0;
}
}
debug("ICH SPI: No ICH found.\n");
return -ENODEV;
}
static int ich_init_controller(struct ich_spi_platdata *plat,
struct ich_spi_priv *ctlr) struct ich_spi_priv *ctlr)
{ {
uint8_t *rcrb; /* Root Complex Register Block */ ulong sbase_addr;
uint32_t rcba; /* Root Complex Base Address */ void *sbase;
uint32_t sbase_addr;
uint8_t *sbase;
pci_read_config_dword(plat->dev, 0xf0, &rcba);
/* Bits 31-14 are the base address, 13-1 are reserved, 0 is enable. */
rcrb = (uint8_t *)(rcba & 0xffffc000);
/* SBASE is similar */ /* SBASE is similar */
pci_read_config_dword(plat->dev, 0x54, &sbase_addr); pch_get_sbase(dev->parent, &sbase_addr);
sbase = (uint8_t *)(sbase_addr & 0xfffffe00); sbase = (void *)sbase_addr;
debug("%s: sbase=%p\n", __func__, sbase);
if (plat->ich_version == 7) { if (plat->ich_version == PCHV_7) {
struct ich7_spi_regs *ich7_spi; struct ich7_spi_regs *ich7_spi = sbase;
ich7_spi = (struct ich7_spi_regs *)(rcrb + 0x3020); ich7_spi = (struct ich7_spi_regs *)sbase;
ctlr->ichspi_lock = readw(&ich7_spi->spis) & SPIS_LOCK; ctlr->ichspi_lock = readw(&ich7_spi->spis) & SPIS_LOCK;
ctlr->opmenu = offsetof(struct ich7_spi_regs, opmenu); ctlr->opmenu = offsetof(struct ich7_spi_regs, opmenu);
ctlr->menubytes = sizeof(ich7_spi->opmenu); ctlr->menubytes = sizeof(ich7_spi->opmenu);
@ -228,13 +165,9 @@ static int ich_init_controller(struct ich_spi_platdata *plat,
ctlr->bbar = offsetof(struct ich7_spi_regs, bbar); ctlr->bbar = offsetof(struct ich7_spi_regs, bbar);
ctlr->preop = offsetof(struct ich7_spi_regs, preop); ctlr->preop = offsetof(struct ich7_spi_regs, preop);
ctlr->base = ich7_spi; ctlr->base = ich7_spi;
} else if (plat->ich_version == 9) { } else if (plat->ich_version == PCHV_9) {
struct ich9_spi_regs *ich9_spi; struct ich9_spi_regs *ich9_spi = sbase;
if (plat->use_sbase)
ich9_spi = (struct ich9_spi_regs *)sbase;
else
ich9_spi = (struct ich9_spi_regs *)(rcrb + 0x3800);
ctlr->ichspi_lock = readw(&ich9_spi->hsfs) & HSFS_FLOCKDN; ctlr->ichspi_lock = readw(&ich9_spi->hsfs) & HSFS_FLOCKDN;
ctlr->opmenu = offsetof(struct ich9_spi_regs, opmenu); ctlr->opmenu = offsetof(struct ich9_spi_regs, opmenu);
ctlr->menubytes = sizeof(ich9_spi->opmenu); ctlr->menubytes = sizeof(ich9_spi->opmenu);
@ -258,9 +191,9 @@ static int ich_init_controller(struct ich_spi_platdata *plat,
/* Work out the maximum speed we can support */ /* Work out the maximum speed we can support */
ctlr->max_speed = 20000000; ctlr->max_speed = 20000000;
if (plat->ich_version == 9 && ich9_can_do_33mhz(plat->dev)) if (plat->ich_version == PCHV_9 && ich9_can_do_33mhz(dev))
ctlr->max_speed = 33000000; ctlr->max_speed = 33000000;
debug("ICH SPI: Version %d detected at %p, speed %ld\n", debug("ICH SPI: Version ID %d detected at %p, speed %ld\n",
plat->ich_version, ctlr->base, ctlr->max_speed); plat->ich_version, ctlr->base, ctlr->max_speed);
ich_set_bbar(ctlr, 0); ich_set_bbar(ctlr, 0);
@ -487,7 +420,7 @@ static int ich_spi_xfer(struct udevice *dev, unsigned int bitlen,
if (ret < 0) if (ret < 0)
return ret; return ret;
if (plat->ich_version == 7) if (plat->ich_version == PCHV_7)
ich_writew(ctlr, SPIS_CDS | SPIS_FCERR, ctlr->status); ich_writew(ctlr, SPIS_CDS | SPIS_FCERR, ctlr->status);
else else
ich_writeb(ctlr, SPIS_CDS | SPIS_FCERR, ctlr->status); ich_writeb(ctlr, SPIS_CDS | SPIS_FCERR, ctlr->status);
@ -682,30 +615,30 @@ int spi_write_protect_region(struct udevice *dev, uint32_t lower_limit,
return 0; return 0;
} }
static int ich_spi_probe(struct udevice *bus) static int ich_spi_probe(struct udevice *dev)
{ {
struct ich_spi_platdata *plat = dev_get_platdata(bus); struct ich_spi_platdata *plat = dev_get_platdata(dev);
struct ich_spi_priv *priv = dev_get_priv(bus); struct ich_spi_priv *priv = dev_get_priv(dev);
uint8_t bios_cntl; uint8_t bios_cntl;
int ret; int ret;
ret = ich_init_controller(plat, priv); /* Check the ICH version */
plat->ich_version = pch_get_version(dev->parent);
ret = ich_init_controller(dev, plat, priv);
if (ret) if (ret)
return ret; return ret;
/* /* Disable the BIOS write protect so write commands are allowed */
* Disable the BIOS write protect so write commands are allowed. On ret = pch_set_spi_protect(dev->parent, false);
* v9, deassert SMM BIOS Write Protect Disable. if (ret == -ENOSYS) {
*/
if (plat->use_sbase) {
bios_cntl = ich_readb(priv, priv->bcr); bios_cntl = ich_readb(priv, priv->bcr);
bios_cntl &= ~BIT(5); /* clear Enable InSMM_STS (EISS) */ bios_cntl &= ~BIT(5); /* clear Enable InSMM_STS (EISS) */
bios_cntl |= 1; /* Write Protect Disable (WPD) */ bios_cntl |= 1; /* Write Protect Disable (WPD) */
ich_writeb(priv, bios_cntl, priv->bcr); ich_writeb(priv, bios_cntl, priv->bcr);
} else { } else if (ret) {
pci_read_config_byte(plat->dev, 0xdc, &bios_cntl); debug("%s: Failed to disable write-protect: err=%d\n",
if (plat->ich_version == 9) __func__, ret);
bios_cntl &= ~BIT(5); return ret;
pci_write_config_byte(plat->dev, 0xdc, bios_cntl | 0x1);
} }
priv->cur_speed = priv->max_speed; priv->cur_speed = priv->max_speed;
@ -713,18 +646,6 @@ static int ich_spi_probe(struct udevice *bus)
return 0; return 0;
} }
static int ich_spi_ofdata_to_platdata(struct udevice *bus)
{
struct ich_spi_platdata *plat = dev_get_platdata(bus);
int ret;
ret = ich_find_spi_controller(plat);
if (ret)
return ret;
return 0;
}
static int ich_spi_set_speed(struct udevice *bus, uint speed) static int ich_spi_set_speed(struct udevice *bus, uint speed)
{ {
struct ich_spi_priv *priv = dev_get_priv(bus); struct ich_spi_priv *priv = dev_get_priv(bus);
@ -757,7 +678,7 @@ static int ich_spi_child_pre_probe(struct udevice *dev)
* ICH 7 SPI controller only supports array read command * ICH 7 SPI controller only supports array read command
* and byte program command for SST flash * and byte program command for SST flash
*/ */
if (plat->ich_version == 7) { if (plat->ich_version == PCHV_7) {
slave->mode_rx = SPI_RX_SLOW; slave->mode_rx = SPI_RX_SLOW;
slave->mode = SPI_TX_BYTE; slave->mode = SPI_TX_BYTE;
} }
@ -785,7 +706,6 @@ U_BOOT_DRIVER(ich_spi) = {
.id = UCLASS_SPI, .id = UCLASS_SPI,
.of_match = ich_spi_ids, .of_match = ich_spi_ids,
.ops = &ich_spi_ops, .ops = &ich_spi_ops,
.ofdata_to_platdata = ich_spi_ofdata_to_platdata,
.platdata_auto_alloc_size = sizeof(struct ich_spi_platdata), .platdata_auto_alloc_size = sizeof(struct ich_spi_platdata),
.priv_auto_alloc_size = sizeof(struct ich_spi_priv), .priv_auto_alloc_size = sizeof(struct ich_spi_priv),
.child_pre_probe = ich_spi_child_pre_probe, .child_pre_probe = ich_spi_child_pre_probe,

Loading…
Cancel
Save