[SPARC64] PCI: Consolidate PCI access code into pci_common.c
All the sun4u controllers do the same thing to compute the physical I/O address to poke, and we can move the sun4v code into this common location too. This one needs a bit of testing, in particular the Sabre code had some funny stuff that would break up u16 and/or u32 accesses into pieces and I didn't think that was needed any more. If it is we need to find out why and add back code to do it again. Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
@@ -14,6 +14,200 @@
|
|||||||
#include <asm/oplib.h>
|
#include <asm/oplib.h>
|
||||||
|
|
||||||
#include "pci_impl.h"
|
#include "pci_impl.h"
|
||||||
|
#include "pci_sun4v.h"
|
||||||
|
|
||||||
|
static int config_out_of_range(struct pci_pbm_info *pbm,
|
||||||
|
unsigned long bus,
|
||||||
|
unsigned long devfn,
|
||||||
|
unsigned long reg)
|
||||||
|
{
|
||||||
|
if (bus < pbm->pci_first_busno ||
|
||||||
|
bus > pbm->pci_last_busno)
|
||||||
|
return 1;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void *sun4u_config_mkaddr(struct pci_pbm_info *pbm,
|
||||||
|
unsigned long bus,
|
||||||
|
unsigned long devfn,
|
||||||
|
unsigned long reg)
|
||||||
|
{
|
||||||
|
unsigned long rbits = pbm->config_space_reg_bits;
|
||||||
|
|
||||||
|
if (config_out_of_range(pbm, bus, devfn, reg))
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
reg = (reg & ((1 << rbits) - 1));
|
||||||
|
devfn <<= rbits;
|
||||||
|
bus <<= rbits + 8;
|
||||||
|
|
||||||
|
return (void *) (pbm->config_space | bus | devfn | reg);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int sun4u_read_pci_cfg(struct pci_bus *bus_dev, unsigned int devfn,
|
||||||
|
int where, int size, u32 *value)
|
||||||
|
{
|
||||||
|
struct pci_pbm_info *pbm = bus_dev->sysdata;
|
||||||
|
unsigned char bus = bus_dev->number;
|
||||||
|
u32 *addr;
|
||||||
|
u16 tmp16;
|
||||||
|
u8 tmp8;
|
||||||
|
|
||||||
|
if (bus_dev == pbm->pci_bus && devfn == 0x00)
|
||||||
|
return pci_host_bridge_read_pci_cfg(bus_dev, devfn, where,
|
||||||
|
size, value);
|
||||||
|
|
||||||
|
switch (size) {
|
||||||
|
case 1:
|
||||||
|
*value = 0xff;
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
*value = 0xffff;
|
||||||
|
break;
|
||||||
|
case 4:
|
||||||
|
*value = 0xffffffff;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
addr = sun4u_config_mkaddr(pbm, bus, devfn, where);
|
||||||
|
if (!addr)
|
||||||
|
return PCIBIOS_SUCCESSFUL;
|
||||||
|
|
||||||
|
switch (size) {
|
||||||
|
case 1:
|
||||||
|
pci_config_read8((u8 *)addr, &tmp8);
|
||||||
|
*value = (u32) tmp8;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 2:
|
||||||
|
if (where & 0x01) {
|
||||||
|
printk("pci_read_config_word: misaligned reg [%x]\n",
|
||||||
|
where);
|
||||||
|
return PCIBIOS_SUCCESSFUL;
|
||||||
|
}
|
||||||
|
pci_config_read16((u16 *)addr, &tmp16);
|
||||||
|
*value = (u32) tmp16;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 4:
|
||||||
|
if (where & 0x03) {
|
||||||
|
printk("pci_read_config_dword: misaligned reg [%x]\n",
|
||||||
|
where);
|
||||||
|
return PCIBIOS_SUCCESSFUL;
|
||||||
|
}
|
||||||
|
pci_config_read32(addr, value);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return PCIBIOS_SUCCESSFUL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int sun4u_write_pci_cfg(struct pci_bus *bus_dev, unsigned int devfn,
|
||||||
|
int where, int size, u32 value)
|
||||||
|
{
|
||||||
|
struct pci_pbm_info *pbm = bus_dev->sysdata;
|
||||||
|
unsigned char bus = bus_dev->number;
|
||||||
|
u32 *addr;
|
||||||
|
|
||||||
|
if (bus_dev == pbm->pci_bus && devfn == 0x00)
|
||||||
|
return pci_host_bridge_write_pci_cfg(bus_dev, devfn, where,
|
||||||
|
size, value);
|
||||||
|
addr = sun4u_config_mkaddr(pbm, bus, devfn, where);
|
||||||
|
if (!addr)
|
||||||
|
return PCIBIOS_SUCCESSFUL;
|
||||||
|
|
||||||
|
switch (size) {
|
||||||
|
case 1:
|
||||||
|
pci_config_write8((u8 *)addr, value);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 2:
|
||||||
|
if (where & 0x01) {
|
||||||
|
printk("pci_write_config_word: misaligned reg [%x]\n",
|
||||||
|
where);
|
||||||
|
return PCIBIOS_SUCCESSFUL;
|
||||||
|
}
|
||||||
|
pci_config_write16((u16 *)addr, value);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 4:
|
||||||
|
if (where & 0x03) {
|
||||||
|
printk("pci_write_config_dword: misaligned reg [%x]\n",
|
||||||
|
where);
|
||||||
|
return PCIBIOS_SUCCESSFUL;
|
||||||
|
}
|
||||||
|
pci_config_write32(addr, value);
|
||||||
|
}
|
||||||
|
return PCIBIOS_SUCCESSFUL;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct pci_ops sun4u_pci_ops = {
|
||||||
|
.read = sun4u_read_pci_cfg,
|
||||||
|
.write = sun4u_write_pci_cfg,
|
||||||
|
};
|
||||||
|
|
||||||
|
static int sun4v_read_pci_cfg(struct pci_bus *bus_dev, unsigned int devfn,
|
||||||
|
int where, int size, u32 *value)
|
||||||
|
{
|
||||||
|
struct pci_pbm_info *pbm = bus_dev->sysdata;
|
||||||
|
u32 devhandle = pbm->devhandle;
|
||||||
|
unsigned int bus = bus_dev->number;
|
||||||
|
unsigned int device = PCI_SLOT(devfn);
|
||||||
|
unsigned int func = PCI_FUNC(devfn);
|
||||||
|
unsigned long ret;
|
||||||
|
|
||||||
|
if (bus_dev == pbm->pci_bus && devfn == 0x00)
|
||||||
|
return pci_host_bridge_read_pci_cfg(bus_dev, devfn, where,
|
||||||
|
size, value);
|
||||||
|
if (config_out_of_range(pbm, bus, devfn, where)) {
|
||||||
|
ret = ~0UL;
|
||||||
|
} else {
|
||||||
|
ret = pci_sun4v_config_get(devhandle,
|
||||||
|
HV_PCI_DEVICE_BUILD(bus, device, func),
|
||||||
|
where, size);
|
||||||
|
}
|
||||||
|
switch (size) {
|
||||||
|
case 1:
|
||||||
|
*value = ret & 0xff;
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
*value = ret & 0xffff;
|
||||||
|
break;
|
||||||
|
case 4:
|
||||||
|
*value = ret & 0xffffffff;
|
||||||
|
break;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
return PCIBIOS_SUCCESSFUL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int sun4v_write_pci_cfg(struct pci_bus *bus_dev, unsigned int devfn,
|
||||||
|
int where, int size, u32 value)
|
||||||
|
{
|
||||||
|
struct pci_pbm_info *pbm = bus_dev->sysdata;
|
||||||
|
u32 devhandle = pbm->devhandle;
|
||||||
|
unsigned int bus = bus_dev->number;
|
||||||
|
unsigned int device = PCI_SLOT(devfn);
|
||||||
|
unsigned int func = PCI_FUNC(devfn);
|
||||||
|
unsigned long ret;
|
||||||
|
|
||||||
|
if (bus_dev == pbm->pci_bus && devfn == 0x00)
|
||||||
|
return pci_host_bridge_write_pci_cfg(bus_dev, devfn, where,
|
||||||
|
size, value);
|
||||||
|
if (config_out_of_range(pbm, bus, devfn, where)) {
|
||||||
|
/* Do nothing. */
|
||||||
|
} else {
|
||||||
|
ret = pci_sun4v_config_put(devhandle,
|
||||||
|
HV_PCI_DEVICE_BUILD(bus, device, func),
|
||||||
|
where, size, value);
|
||||||
|
}
|
||||||
|
return PCIBIOS_SUCCESSFUL;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct pci_ops sun4v_pci_ops = {
|
||||||
|
.read = sun4v_read_pci_cfg,
|
||||||
|
.write = sun4v_write_pci_cfg,
|
||||||
|
};
|
||||||
|
|
||||||
void pci_get_pbm_props(struct pci_pbm_info *pbm)
|
void pci_get_pbm_props(struct pci_pbm_info *pbm)
|
||||||
{
|
{
|
||||||
|
@@ -27,138 +27,6 @@
|
|||||||
"i" (ASI_PHYS_BYPASS_EC_E) \
|
"i" (ASI_PHYS_BYPASS_EC_E) \
|
||||||
: "memory")
|
: "memory")
|
||||||
|
|
||||||
/* Fire config space address format is nearly identical to
|
|
||||||
* that of SCHIZO and PSYCHO, except that in order to accomodate
|
|
||||||
* PCI-E extended config space the encoding can handle 12 bits
|
|
||||||
* of register address:
|
|
||||||
*
|
|
||||||
* 32 28 27 20 19 15 14 12 11 2 1 0
|
|
||||||
* -------------------------------------------------
|
|
||||||
* |0 0 0 0 0| bus | device | function | reg | 0 0 |
|
|
||||||
* -------------------------------------------------
|
|
||||||
*/
|
|
||||||
#define FIRE_CONFIG_BASE(PBM) ((PBM)->config_space)
|
|
||||||
#define FIRE_CONFIG_ENCODE(BUS, DEVFN, REG) \
|
|
||||||
(((unsigned long)(BUS) << 20) | \
|
|
||||||
((unsigned long)(DEVFN) << 12) | \
|
|
||||||
((unsigned long)(REG)))
|
|
||||||
|
|
||||||
static void *fire_pci_config_mkaddr(struct pci_pbm_info *pbm,
|
|
||||||
unsigned char bus,
|
|
||||||
unsigned int devfn,
|
|
||||||
int where)
|
|
||||||
{
|
|
||||||
if (!pbm)
|
|
||||||
return NULL;
|
|
||||||
return (void *)
|
|
||||||
(FIRE_CONFIG_BASE(pbm) |
|
|
||||||
FIRE_CONFIG_ENCODE(bus, devfn, where));
|
|
||||||
}
|
|
||||||
|
|
||||||
/* FIRE PCI configuration space accessors. */
|
|
||||||
|
|
||||||
static int fire_read_pci_cfg(struct pci_bus *bus_dev, unsigned int devfn,
|
|
||||||
int where, int size, u32 *value)
|
|
||||||
{
|
|
||||||
struct pci_pbm_info *pbm = bus_dev->sysdata;
|
|
||||||
unsigned char bus = bus_dev->number;
|
|
||||||
u32 *addr;
|
|
||||||
u16 tmp16;
|
|
||||||
u8 tmp8;
|
|
||||||
|
|
||||||
if (bus_dev == pbm->pci_bus && devfn == 0x00)
|
|
||||||
return pci_host_bridge_read_pci_cfg(bus_dev, devfn, where,
|
|
||||||
size, value);
|
|
||||||
switch (size) {
|
|
||||||
case 1:
|
|
||||||
*value = 0xff;
|
|
||||||
break;
|
|
||||||
case 2:
|
|
||||||
*value = 0xffff;
|
|
||||||
break;
|
|
||||||
case 4:
|
|
||||||
*value = 0xffffffff;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
addr = fire_pci_config_mkaddr(pbm, bus, devfn, where);
|
|
||||||
if (!addr)
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
|
|
||||||
switch (size) {
|
|
||||||
case 1:
|
|
||||||
pci_config_read8((u8 *)addr, &tmp8);
|
|
||||||
*value = tmp8;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 2:
|
|
||||||
if (where & 0x01) {
|
|
||||||
printk("pci_read_config_word: misaligned reg [%x]\n",
|
|
||||||
where);
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
}
|
|
||||||
pci_config_read16((u16 *)addr, &tmp16);
|
|
||||||
*value = tmp16;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 4:
|
|
||||||
if (where & 0x03) {
|
|
||||||
printk("pci_read_config_dword: misaligned reg [%x]\n",
|
|
||||||
where);
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
}
|
|
||||||
|
|
||||||
pci_config_read32(addr, value);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int fire_write_pci_cfg(struct pci_bus *bus_dev, unsigned int devfn,
|
|
||||||
int where, int size, u32 value)
|
|
||||||
{
|
|
||||||
struct pci_pbm_info *pbm = bus_dev->sysdata;
|
|
||||||
unsigned char bus = bus_dev->number;
|
|
||||||
u32 *addr;
|
|
||||||
|
|
||||||
if (bus_dev == pbm->pci_bus && devfn == 0x00)
|
|
||||||
return pci_host_bridge_write_pci_cfg(bus_dev, devfn, where,
|
|
||||||
size, value);
|
|
||||||
addr = fire_pci_config_mkaddr(pbm, bus, devfn, where);
|
|
||||||
if (!addr)
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
|
|
||||||
switch (size) {
|
|
||||||
case 1:
|
|
||||||
pci_config_write8((u8 *)addr, value);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 2:
|
|
||||||
if (where & 0x01) {
|
|
||||||
printk("pci_write_config_word: misaligned reg [%x]\n",
|
|
||||||
where);
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
}
|
|
||||||
pci_config_write16((u16 *)addr, value);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 4:
|
|
||||||
if (where & 0x03) {
|
|
||||||
printk("pci_write_config_dword: misaligned reg [%x]\n",
|
|
||||||
where);
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
}
|
|
||||||
|
|
||||||
pci_config_write32(addr, value);
|
|
||||||
}
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
}
|
|
||||||
|
|
||||||
static struct pci_ops pci_fire_ops = {
|
|
||||||
.read = fire_read_pci_cfg,
|
|
||||||
.write = fire_write_pci_cfg,
|
|
||||||
};
|
|
||||||
|
|
||||||
static void pci_fire_scan_bus(struct pci_pbm_info *pbm)
|
static void pci_fire_scan_bus(struct pci_pbm_info *pbm)
|
||||||
{
|
{
|
||||||
pbm->pci_bus = pci_scan_one_pbm(pbm);
|
pbm->pci_bus = pci_scan_one_pbm(pbm);
|
||||||
@@ -314,7 +182,8 @@ static void pci_fire_pbm_init(struct pci_controller_info *p,
|
|||||||
pci_pbm_root = pbm;
|
pci_pbm_root = pbm;
|
||||||
|
|
||||||
pbm->scan_bus = pci_fire_scan_bus;
|
pbm->scan_bus = pci_fire_scan_bus;
|
||||||
pbm->pci_ops = &pci_fire_ops;
|
pbm->pci_ops = &sun4u_pci_ops;
|
||||||
|
pbm->config_space_reg_bits = 12;
|
||||||
|
|
||||||
pbm->index = pci_num_pbms++;
|
pbm->index = pci_num_pbms++;
|
||||||
|
|
||||||
|
@@ -77,6 +77,9 @@ struct pci_pbm_info {
|
|||||||
/* Base of PCI Config space, can be per-PBM or shared. */
|
/* Base of PCI Config space, can be per-PBM or shared. */
|
||||||
unsigned long config_space;
|
unsigned long config_space;
|
||||||
|
|
||||||
|
/* This will be 12 on PCI-E controllers, 8 elsewhere. */
|
||||||
|
unsigned long config_space_reg_bits;
|
||||||
|
|
||||||
/* State of 66MHz capabilities on this PBM. */
|
/* State of 66MHz capabilities on this PBM. */
|
||||||
int is_66mhz_capable;
|
int is_66mhz_capable;
|
||||||
int all_devs_66mhz;
|
int all_devs_66mhz;
|
||||||
@@ -156,4 +159,7 @@ extern void pci_config_write8(u8 *addr, u8 val);
|
|||||||
extern void pci_config_write16(u16 *addr, u16 val);
|
extern void pci_config_write16(u16 *addr, u16 val);
|
||||||
extern void pci_config_write32(u32 *addr, u32 val);
|
extern void pci_config_write32(u32 *addr, u32 val);
|
||||||
|
|
||||||
|
extern struct pci_ops sun4u_pci_ops;
|
||||||
|
extern struct pci_ops sun4v_pci_ops;
|
||||||
|
|
||||||
#endif /* !(PCI_IMPL_H) */
|
#endif /* !(PCI_IMPL_H) */
|
||||||
|
@@ -94,122 +94,6 @@ static void *psycho_pci_config_mkaddr(struct pci_pbm_info *pbm,
|
|||||||
PSYCHO_CONFIG_ENCODE(bus, devfn, where));
|
PSYCHO_CONFIG_ENCODE(bus, devfn, where));
|
||||||
}
|
}
|
||||||
|
|
||||||
static int psycho_out_of_range(struct pci_pbm_info *pbm,
|
|
||||||
unsigned char bus,
|
|
||||||
unsigned char devfn)
|
|
||||||
{
|
|
||||||
return ((bus == pbm->pci_first_busno) &&
|
|
||||||
PCI_SLOT(devfn) > 8);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* PSYCHO PCI configuration space accessors. */
|
|
||||||
|
|
||||||
static int psycho_read_pci_cfg(struct pci_bus *bus_dev, unsigned int devfn,
|
|
||||||
int where, int size, u32 *value)
|
|
||||||
{
|
|
||||||
struct pci_pbm_info *pbm = bus_dev->sysdata;
|
|
||||||
unsigned char bus = bus_dev->number;
|
|
||||||
u32 *addr;
|
|
||||||
u16 tmp16;
|
|
||||||
u8 tmp8;
|
|
||||||
|
|
||||||
if (bus_dev == pbm->pci_bus && devfn == 0x00)
|
|
||||||
return pci_host_bridge_read_pci_cfg(bus_dev, devfn, where,
|
|
||||||
size, value);
|
|
||||||
|
|
||||||
switch (size) {
|
|
||||||
case 1:
|
|
||||||
*value = 0xff;
|
|
||||||
break;
|
|
||||||
case 2:
|
|
||||||
*value = 0xffff;
|
|
||||||
break;
|
|
||||||
case 4:
|
|
||||||
*value = 0xffffffff;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
addr = psycho_pci_config_mkaddr(pbm, bus, devfn, where);
|
|
||||||
if (!addr)
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
|
|
||||||
if (psycho_out_of_range(pbm, bus, devfn))
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
switch (size) {
|
|
||||||
case 1:
|
|
||||||
pci_config_read8((u8 *)addr, &tmp8);
|
|
||||||
*value = (u32) tmp8;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 2:
|
|
||||||
if (where & 0x01) {
|
|
||||||
printk("pci_read_config_word: misaligned reg [%x]\n",
|
|
||||||
where);
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
}
|
|
||||||
pci_config_read16((u16 *)addr, &tmp16);
|
|
||||||
*value = (u32) tmp16;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 4:
|
|
||||||
if (where & 0x03) {
|
|
||||||
printk("pci_read_config_dword: misaligned reg [%x]\n",
|
|
||||||
where);
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
}
|
|
||||||
pci_config_read32(addr, value);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int psycho_write_pci_cfg(struct pci_bus *bus_dev, unsigned int devfn,
|
|
||||||
int where, int size, u32 value)
|
|
||||||
{
|
|
||||||
struct pci_pbm_info *pbm = bus_dev->sysdata;
|
|
||||||
unsigned char bus = bus_dev->number;
|
|
||||||
u32 *addr;
|
|
||||||
|
|
||||||
if (bus_dev == pbm->pci_bus && devfn == 0x00)
|
|
||||||
return pci_host_bridge_write_pci_cfg(bus_dev, devfn, where,
|
|
||||||
size, value);
|
|
||||||
addr = psycho_pci_config_mkaddr(pbm, bus, devfn, where);
|
|
||||||
if (!addr)
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
|
|
||||||
if (psycho_out_of_range(pbm, bus, devfn))
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
|
|
||||||
switch (size) {
|
|
||||||
case 1:
|
|
||||||
pci_config_write8((u8 *)addr, value);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 2:
|
|
||||||
if (where & 0x01) {
|
|
||||||
printk("pci_write_config_word: misaligned reg [%x]\n",
|
|
||||||
where);
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
}
|
|
||||||
pci_config_write16((u16 *)addr, value);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 4:
|
|
||||||
if (where & 0x03) {
|
|
||||||
printk("pci_write_config_dword: misaligned reg [%x]\n",
|
|
||||||
where);
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
}
|
|
||||||
pci_config_write32(addr, value);
|
|
||||||
}
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
}
|
|
||||||
|
|
||||||
static struct pci_ops psycho_ops = {
|
|
||||||
.read = psycho_read_pci_cfg,
|
|
||||||
.write = psycho_write_pci_cfg,
|
|
||||||
};
|
|
||||||
|
|
||||||
/* PSYCHO error handling support. */
|
/* PSYCHO error handling support. */
|
||||||
enum psycho_error_type {
|
enum psycho_error_type {
|
||||||
UE_ERR, CE_ERR, PCI_ERR
|
UE_ERR, CE_ERR, PCI_ERR
|
||||||
@@ -1089,7 +973,8 @@ static void psycho_pbm_init(struct pci_controller_info *p,
|
|||||||
pci_pbm_root = pbm;
|
pci_pbm_root = pbm;
|
||||||
|
|
||||||
pbm->scan_bus = psycho_scan_bus;
|
pbm->scan_bus = psycho_scan_bus;
|
||||||
pbm->pci_ops = &psycho_ops;
|
pbm->pci_ops = &sun4u_pci_ops;
|
||||||
|
pbm->config_space_reg_bits = 8;
|
||||||
|
|
||||||
pbm->index = pci_num_pbms++;
|
pbm->index = pci_num_pbms++;
|
||||||
|
|
||||||
|
@@ -205,294 +205,9 @@
|
|||||||
#define SABRE_MEMSPACE 0x100000000UL
|
#define SABRE_MEMSPACE 0x100000000UL
|
||||||
#define SABRE_MEMSPACE_SIZE 0x07fffffffUL
|
#define SABRE_MEMSPACE_SIZE 0x07fffffffUL
|
||||||
|
|
||||||
/* UltraSparc-IIi Programmer's Manual, page 325, PCI
|
|
||||||
* configuration space address format:
|
|
||||||
*
|
|
||||||
* 32 24 23 16 15 11 10 8 7 2 1 0
|
|
||||||
* ---------------------------------------------------------
|
|
||||||
* |0 0 0 0 0 0 0 0 1| bus | device | function | reg | 0 0 |
|
|
||||||
* ---------------------------------------------------------
|
|
||||||
*/
|
|
||||||
#define SABRE_CONFIG_BASE(PBM) \
|
|
||||||
((PBM)->config_space | (1UL << 24))
|
|
||||||
#define SABRE_CONFIG_ENCODE(BUS, DEVFN, REG) \
|
|
||||||
(((unsigned long)(BUS) << 16) | \
|
|
||||||
((unsigned long)(DEVFN) << 8) | \
|
|
||||||
((unsigned long)(REG)))
|
|
||||||
|
|
||||||
static int hummingbird_p;
|
static int hummingbird_p;
|
||||||
static struct pci_bus *sabre_root_bus;
|
static struct pci_bus *sabre_root_bus;
|
||||||
|
|
||||||
static void *sabre_pci_config_mkaddr(struct pci_pbm_info *pbm,
|
|
||||||
unsigned char bus,
|
|
||||||
unsigned int devfn,
|
|
||||||
int where)
|
|
||||||
{
|
|
||||||
if (!pbm)
|
|
||||||
return NULL;
|
|
||||||
return (void *)
|
|
||||||
(SABRE_CONFIG_BASE(pbm) |
|
|
||||||
SABRE_CONFIG_ENCODE(bus, devfn, where));
|
|
||||||
}
|
|
||||||
|
|
||||||
static int sabre_out_of_range(unsigned char devfn)
|
|
||||||
{
|
|
||||||
if (hummingbird_p)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
return (((PCI_SLOT(devfn) == 0) && (PCI_FUNC(devfn) > 0)) ||
|
|
||||||
((PCI_SLOT(devfn) == 1) && (PCI_FUNC(devfn) > 1)) ||
|
|
||||||
(PCI_SLOT(devfn) > 1));
|
|
||||||
}
|
|
||||||
|
|
||||||
static int __sabre_out_of_range(struct pci_pbm_info *pbm,
|
|
||||||
unsigned char bus,
|
|
||||||
unsigned char devfn)
|
|
||||||
{
|
|
||||||
if (hummingbird_p)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
return ((pbm->parent == 0) ||
|
|
||||||
((pbm == &pbm->parent->pbm_A) &&
|
|
||||||
(bus == pbm->pci_first_busno) &&
|
|
||||||
PCI_SLOT(devfn) > 8));
|
|
||||||
}
|
|
||||||
|
|
||||||
static int __sabre_read_pci_cfg(struct pci_bus *bus_dev, unsigned int devfn,
|
|
||||||
int where, int size, u32 *value)
|
|
||||||
{
|
|
||||||
struct pci_pbm_info *pbm = bus_dev->sysdata;
|
|
||||||
unsigned char bus = bus_dev->number;
|
|
||||||
u32 *addr;
|
|
||||||
u16 tmp16;
|
|
||||||
u8 tmp8;
|
|
||||||
|
|
||||||
switch (size) {
|
|
||||||
case 1:
|
|
||||||
*value = 0xff;
|
|
||||||
break;
|
|
||||||
case 2:
|
|
||||||
*value = 0xffff;
|
|
||||||
break;
|
|
||||||
case 4:
|
|
||||||
*value = 0xffffffff;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
addr = sabre_pci_config_mkaddr(pbm, bus, devfn, where);
|
|
||||||
if (!addr)
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
|
|
||||||
if (__sabre_out_of_range(pbm, bus, devfn))
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
|
|
||||||
switch (size) {
|
|
||||||
case 1:
|
|
||||||
pci_config_read8((u8 *) addr, &tmp8);
|
|
||||||
*value = tmp8;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 2:
|
|
||||||
if (where & 0x01) {
|
|
||||||
printk("pci_read_config_word: misaligned reg [%x]\n",
|
|
||||||
where);
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
}
|
|
||||||
pci_config_read16((u16 *) addr, &tmp16);
|
|
||||||
*value = tmp16;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 4:
|
|
||||||
if (where & 0x03) {
|
|
||||||
printk("pci_read_config_dword: misaligned reg [%x]\n",
|
|
||||||
where);
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
}
|
|
||||||
pci_config_read32(addr, value);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int sabre_read_pci_cfg(struct pci_bus *bus, unsigned int devfn,
|
|
||||||
int where, int size, u32 *value)
|
|
||||||
{
|
|
||||||
struct pci_pbm_info *pbm = bus->sysdata;
|
|
||||||
|
|
||||||
if (bus == pbm->pci_bus && devfn == 0x00)
|
|
||||||
return pci_host_bridge_read_pci_cfg(bus, devfn, where,
|
|
||||||
size, value);
|
|
||||||
|
|
||||||
if (!bus->number && sabre_out_of_range(devfn)) {
|
|
||||||
switch (size) {
|
|
||||||
case 1:
|
|
||||||
*value = 0xff;
|
|
||||||
break;
|
|
||||||
case 2:
|
|
||||||
*value = 0xffff;
|
|
||||||
break;
|
|
||||||
case 4:
|
|
||||||
*value = 0xffffffff;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (bus->number || PCI_SLOT(devfn))
|
|
||||||
return __sabre_read_pci_cfg(bus, devfn, where, size, value);
|
|
||||||
|
|
||||||
/* When accessing PCI config space of the PCI controller itself (bus
|
|
||||||
* 0, device slot 0, function 0) there are restrictions. Each
|
|
||||||
* register must be accessed as it's natural size. Thus, for example
|
|
||||||
* the Vendor ID must be accessed as a 16-bit quantity.
|
|
||||||
*/
|
|
||||||
|
|
||||||
switch (size) {
|
|
||||||
case 1:
|
|
||||||
if (where < 8) {
|
|
||||||
u32 tmp32;
|
|
||||||
u16 tmp16;
|
|
||||||
|
|
||||||
__sabre_read_pci_cfg(bus, devfn, where & ~1, 2, &tmp32);
|
|
||||||
tmp16 = (u16) tmp32;
|
|
||||||
if (where & 1)
|
|
||||||
*value = tmp16 >> 8;
|
|
||||||
else
|
|
||||||
*value = tmp16 & 0xff;
|
|
||||||
} else
|
|
||||||
return __sabre_read_pci_cfg(bus, devfn, where, 1, value);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 2:
|
|
||||||
if (where < 8)
|
|
||||||
return __sabre_read_pci_cfg(bus, devfn, where, 2, value);
|
|
||||||
else {
|
|
||||||
u32 tmp32;
|
|
||||||
u8 tmp8;
|
|
||||||
|
|
||||||
__sabre_read_pci_cfg(bus, devfn, where, 1, &tmp32);
|
|
||||||
tmp8 = (u8) tmp32;
|
|
||||||
*value = tmp8;
|
|
||||||
__sabre_read_pci_cfg(bus, devfn, where + 1, 1, &tmp32);
|
|
||||||
tmp8 = (u8) tmp32;
|
|
||||||
*value |= tmp8 << 8;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 4: {
|
|
||||||
u32 tmp32;
|
|
||||||
u16 tmp16;
|
|
||||||
|
|
||||||
sabre_read_pci_cfg(bus, devfn, where, 2, &tmp32);
|
|
||||||
tmp16 = (u16) tmp32;
|
|
||||||
*value = tmp16;
|
|
||||||
sabre_read_pci_cfg(bus, devfn, where + 2, 2, &tmp32);
|
|
||||||
tmp16 = (u16) tmp32;
|
|
||||||
*value |= tmp16 << 16;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int __sabre_write_pci_cfg(struct pci_bus *bus_dev, unsigned int devfn,
|
|
||||||
int where, int size, u32 value)
|
|
||||||
{
|
|
||||||
struct pci_pbm_info *pbm = bus_dev->sysdata;
|
|
||||||
unsigned char bus = bus_dev->number;
|
|
||||||
u32 *addr;
|
|
||||||
|
|
||||||
addr = sabre_pci_config_mkaddr(pbm, bus, devfn, where);
|
|
||||||
if (!addr)
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
|
|
||||||
if (__sabre_out_of_range(pbm, bus, devfn))
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
|
|
||||||
switch (size) {
|
|
||||||
case 1:
|
|
||||||
pci_config_write8((u8 *) addr, value);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 2:
|
|
||||||
if (where & 0x01) {
|
|
||||||
printk("pci_write_config_word: misaligned reg [%x]\n",
|
|
||||||
where);
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
}
|
|
||||||
pci_config_write16((u16 *) addr, value);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 4:
|
|
||||||
if (where & 0x03) {
|
|
||||||
printk("pci_write_config_dword: misaligned reg [%x]\n",
|
|
||||||
where);
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
}
|
|
||||||
pci_config_write32(addr, value);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int sabre_write_pci_cfg(struct pci_bus *bus, unsigned int devfn,
|
|
||||||
int where, int size, u32 value)
|
|
||||||
{
|
|
||||||
struct pci_pbm_info *pbm = bus->sysdata;
|
|
||||||
|
|
||||||
if (bus == pbm->pci_bus && devfn == 0x00)
|
|
||||||
return pci_host_bridge_write_pci_cfg(bus, devfn, where,
|
|
||||||
size, value);
|
|
||||||
|
|
||||||
if (bus->number)
|
|
||||||
return __sabre_write_pci_cfg(bus, devfn, where, size, value);
|
|
||||||
|
|
||||||
if (sabre_out_of_range(devfn))
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
|
|
||||||
switch (size) {
|
|
||||||
case 1:
|
|
||||||
if (where < 8) {
|
|
||||||
u32 tmp32;
|
|
||||||
u16 tmp16;
|
|
||||||
|
|
||||||
__sabre_read_pci_cfg(bus, devfn, where & ~1, 2, &tmp32);
|
|
||||||
tmp16 = (u16) tmp32;
|
|
||||||
if (where & 1) {
|
|
||||||
value &= 0x00ff;
|
|
||||||
value |= tmp16 << 8;
|
|
||||||
} else {
|
|
||||||
value &= 0xff00;
|
|
||||||
value |= tmp16;
|
|
||||||
}
|
|
||||||
tmp32 = (u32) tmp16;
|
|
||||||
return __sabre_write_pci_cfg(bus, devfn, where & ~1, 2, tmp32);
|
|
||||||
} else
|
|
||||||
return __sabre_write_pci_cfg(bus, devfn, where, 1, value);
|
|
||||||
break;
|
|
||||||
case 2:
|
|
||||||
if (where < 8)
|
|
||||||
return __sabre_write_pci_cfg(bus, devfn, where, 2, value);
|
|
||||||
else {
|
|
||||||
__sabre_write_pci_cfg(bus, devfn, where, 1, value & 0xff);
|
|
||||||
__sabre_write_pci_cfg(bus, devfn, where + 1, 1, value >> 8);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case 4:
|
|
||||||
sabre_write_pci_cfg(bus, devfn, where, 2, value & 0xffff);
|
|
||||||
sabre_write_pci_cfg(bus, devfn, where + 2, 2, value >> 16);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
}
|
|
||||||
|
|
||||||
static struct pci_ops sabre_ops = {
|
|
||||||
.read = sabre_read_pci_cfg,
|
|
||||||
.write = sabre_write_pci_cfg,
|
|
||||||
};
|
|
||||||
|
|
||||||
/* SABRE error handling support. */
|
/* SABRE error handling support. */
|
||||||
static void sabre_check_iommu_error(struct pci_pbm_info *pbm,
|
static void sabre_check_iommu_error(struct pci_pbm_info *pbm,
|
||||||
unsigned long afsr,
|
unsigned long afsr,
|
||||||
@@ -1010,7 +725,8 @@ static void sabre_pbm_init(struct pci_controller_info *p, struct pci_pbm_info *p
|
|||||||
printk("%s: SABRE PCI Bus Module\n", pbm->name);
|
printk("%s: SABRE PCI Bus Module\n", pbm->name);
|
||||||
|
|
||||||
pbm->scan_bus = sabre_scan_bus;
|
pbm->scan_bus = sabre_scan_bus;
|
||||||
pbm->pci_ops = &sabre_ops;
|
pbm->pci_ops = &sun4u_pci_ops;
|
||||||
|
pbm->config_space_reg_bits = 8;
|
||||||
|
|
||||||
pbm->index = pci_num_pbms++;
|
pbm->index = pci_num_pbms++;
|
||||||
|
|
||||||
|
@@ -104,125 +104,6 @@ static void *schizo_pci_config_mkaddr(struct pci_pbm_info *pbm,
|
|||||||
SCHIZO_CONFIG_ENCODE(bus, devfn, where));
|
SCHIZO_CONFIG_ENCODE(bus, devfn, where));
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Just make sure the bus number is in range. */
|
|
||||||
static int schizo_out_of_range(struct pci_pbm_info *pbm,
|
|
||||||
unsigned char bus,
|
|
||||||
unsigned char devfn)
|
|
||||||
{
|
|
||||||
if (bus < pbm->pci_first_busno ||
|
|
||||||
bus > pbm->pci_last_busno)
|
|
||||||
return 1;
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* SCHIZO PCI configuration space accessors. */
|
|
||||||
|
|
||||||
static int schizo_read_pci_cfg(struct pci_bus *bus_dev, unsigned int devfn,
|
|
||||||
int where, int size, u32 *value)
|
|
||||||
{
|
|
||||||
struct pci_pbm_info *pbm = bus_dev->sysdata;
|
|
||||||
unsigned char bus = bus_dev->number;
|
|
||||||
u32 *addr;
|
|
||||||
u16 tmp16;
|
|
||||||
u8 tmp8;
|
|
||||||
|
|
||||||
if (bus_dev == pbm->pci_bus && devfn == 0x00)
|
|
||||||
return pci_host_bridge_read_pci_cfg(bus_dev, devfn, where,
|
|
||||||
size, value);
|
|
||||||
switch (size) {
|
|
||||||
case 1:
|
|
||||||
*value = 0xff;
|
|
||||||
break;
|
|
||||||
case 2:
|
|
||||||
*value = 0xffff;
|
|
||||||
break;
|
|
||||||
case 4:
|
|
||||||
*value = 0xffffffff;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
addr = schizo_pci_config_mkaddr(pbm, bus, devfn, where);
|
|
||||||
if (!addr)
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
|
|
||||||
if (schizo_out_of_range(pbm, bus, devfn))
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
switch (size) {
|
|
||||||
case 1:
|
|
||||||
pci_config_read8((u8 *)addr, &tmp8);
|
|
||||||
*value = tmp8;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 2:
|
|
||||||
if (where & 0x01) {
|
|
||||||
printk("pci_read_config_word: misaligned reg [%x]\n",
|
|
||||||
where);
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
}
|
|
||||||
pci_config_read16((u16 *)addr, &tmp16);
|
|
||||||
*value = tmp16;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 4:
|
|
||||||
if (where & 0x03) {
|
|
||||||
printk("pci_read_config_dword: misaligned reg [%x]\n",
|
|
||||||
where);
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
}
|
|
||||||
pci_config_read32(addr, value);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int schizo_write_pci_cfg(struct pci_bus *bus_dev, unsigned int devfn,
|
|
||||||
int where, int size, u32 value)
|
|
||||||
{
|
|
||||||
struct pci_pbm_info *pbm = bus_dev->sysdata;
|
|
||||||
unsigned char bus = bus_dev->number;
|
|
||||||
u32 *addr;
|
|
||||||
|
|
||||||
if (bus_dev == pbm->pci_bus && devfn == 0x00)
|
|
||||||
return pci_host_bridge_write_pci_cfg(bus_dev, devfn, where,
|
|
||||||
size, value);
|
|
||||||
addr = schizo_pci_config_mkaddr(pbm, bus, devfn, where);
|
|
||||||
if (!addr)
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
|
|
||||||
if (schizo_out_of_range(pbm, bus, devfn))
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
|
|
||||||
switch (size) {
|
|
||||||
case 1:
|
|
||||||
pci_config_write8((u8 *)addr, value);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 2:
|
|
||||||
if (where & 0x01) {
|
|
||||||
printk("pci_write_config_word: misaligned reg [%x]\n",
|
|
||||||
where);
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
}
|
|
||||||
pci_config_write16((u16 *)addr, value);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 4:
|
|
||||||
if (where & 0x03) {
|
|
||||||
printk("pci_write_config_dword: misaligned reg [%x]\n",
|
|
||||||
where);
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
}
|
|
||||||
|
|
||||||
pci_config_write32(addr, value);
|
|
||||||
}
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
}
|
|
||||||
|
|
||||||
static struct pci_ops schizo_ops = {
|
|
||||||
.read = schizo_read_pci_cfg,
|
|
||||||
.write = schizo_write_pci_cfg,
|
|
||||||
};
|
|
||||||
|
|
||||||
/* SCHIZO error handling support. */
|
/* SCHIZO error handling support. */
|
||||||
enum schizo_error_type {
|
enum schizo_error_type {
|
||||||
UE_ERR, CE_ERR, PCI_ERR, SAFARI_ERR
|
UE_ERR, CE_ERR, PCI_ERR, SAFARI_ERR
|
||||||
@@ -1494,7 +1375,8 @@ static void schizo_pbm_init(struct pci_controller_info *p,
|
|||||||
pci_pbm_root = pbm;
|
pci_pbm_root = pbm;
|
||||||
|
|
||||||
pbm->scan_bus = schizo_scan_bus;
|
pbm->scan_bus = schizo_scan_bus;
|
||||||
pbm->pci_ops = &schizo_ops;
|
pbm->pci_ops = &sun4u_pci_ops;
|
||||||
|
pbm->config_space_reg_bits = 8;
|
||||||
|
|
||||||
pbm->index = pci_num_pbms++;
|
pbm->index = pci_num_pbms++;
|
||||||
|
|
||||||
|
@@ -593,89 +593,6 @@ const struct pci_iommu_ops pci_sun4v_iommu_ops = {
|
|||||||
.dma_sync_sg_for_cpu = pci_4v_dma_sync_sg_for_cpu,
|
.dma_sync_sg_for_cpu = pci_4v_dma_sync_sg_for_cpu,
|
||||||
};
|
};
|
||||||
|
|
||||||
static inline int pci_sun4v_out_of_range(struct pci_pbm_info *pbm, unsigned int bus, unsigned int device, unsigned int func)
|
|
||||||
{
|
|
||||||
if (bus < pbm->pci_first_busno ||
|
|
||||||
bus > pbm->pci_last_busno)
|
|
||||||
return 1;
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int pci_sun4v_read_pci_cfg(struct pci_bus *bus_dev, unsigned int devfn,
|
|
||||||
int where, int size, u32 *value)
|
|
||||||
{
|
|
||||||
struct pci_pbm_info *pbm = bus_dev->sysdata;
|
|
||||||
u32 devhandle = pbm->devhandle;
|
|
||||||
unsigned int bus = bus_dev->number;
|
|
||||||
unsigned int device = PCI_SLOT(devfn);
|
|
||||||
unsigned int func = PCI_FUNC(devfn);
|
|
||||||
unsigned long ret;
|
|
||||||
|
|
||||||
if (bus_dev == pbm->pci_bus && devfn == 0x00)
|
|
||||||
return pci_host_bridge_read_pci_cfg(bus_dev, devfn, where,
|
|
||||||
size, value);
|
|
||||||
if (pci_sun4v_out_of_range(pbm, bus, device, func)) {
|
|
||||||
ret = ~0UL;
|
|
||||||
} else {
|
|
||||||
ret = pci_sun4v_config_get(devhandle,
|
|
||||||
HV_PCI_DEVICE_BUILD(bus, device, func),
|
|
||||||
where, size);
|
|
||||||
#if 0
|
|
||||||
printk("rcfg: [%x:%x:%x:%d]=[%lx]\n",
|
|
||||||
devhandle, HV_PCI_DEVICE_BUILD(bus, device, func),
|
|
||||||
where, size, ret);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
switch (size) {
|
|
||||||
case 1:
|
|
||||||
*value = ret & 0xff;
|
|
||||||
break;
|
|
||||||
case 2:
|
|
||||||
*value = ret & 0xffff;
|
|
||||||
break;
|
|
||||||
case 4:
|
|
||||||
*value = ret & 0xffffffff;
|
|
||||||
break;
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int pci_sun4v_write_pci_cfg(struct pci_bus *bus_dev, unsigned int devfn,
|
|
||||||
int where, int size, u32 value)
|
|
||||||
{
|
|
||||||
struct pci_pbm_info *pbm = bus_dev->sysdata;
|
|
||||||
u32 devhandle = pbm->devhandle;
|
|
||||||
unsigned int bus = bus_dev->number;
|
|
||||||
unsigned int device = PCI_SLOT(devfn);
|
|
||||||
unsigned int func = PCI_FUNC(devfn);
|
|
||||||
unsigned long ret;
|
|
||||||
|
|
||||||
if (bus_dev == pbm->pci_bus && devfn == 0x00)
|
|
||||||
return pci_host_bridge_write_pci_cfg(bus_dev, devfn, where,
|
|
||||||
size, value);
|
|
||||||
if (pci_sun4v_out_of_range(pbm, bus, device, func)) {
|
|
||||||
/* Do nothing. */
|
|
||||||
} else {
|
|
||||||
ret = pci_sun4v_config_put(devhandle,
|
|
||||||
HV_PCI_DEVICE_BUILD(bus, device, func),
|
|
||||||
where, size, value);
|
|
||||||
#if 0
|
|
||||||
printk("wcfg: [%x:%x:%x:%d] v[%x] == [%lx]\n",
|
|
||||||
devhandle, HV_PCI_DEVICE_BUILD(bus, device, func),
|
|
||||||
where, size, value, ret);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
return PCIBIOS_SUCCESSFUL;
|
|
||||||
}
|
|
||||||
|
|
||||||
static struct pci_ops pci_sun4v_ops = {
|
|
||||||
.read = pci_sun4v_read_pci_cfg,
|
|
||||||
.write = pci_sun4v_write_pci_cfg,
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
static void pci_sun4v_scan_bus(struct pci_pbm_info *pbm)
|
static void pci_sun4v_scan_bus(struct pci_pbm_info *pbm)
|
||||||
{
|
{
|
||||||
struct property *prop;
|
struct property *prop;
|
||||||
@@ -1238,7 +1155,8 @@ static void pci_sun4v_pbm_init(struct pci_controller_info *p, struct device_node
|
|||||||
pci_pbm_root = pbm;
|
pci_pbm_root = pbm;
|
||||||
|
|
||||||
pbm->scan_bus = pci_sun4v_scan_bus;
|
pbm->scan_bus = pci_sun4v_scan_bus;
|
||||||
pbm->pci_ops = &pci_sun4v_ops;
|
pbm->pci_ops = &sun4v_pci_ops;
|
||||||
|
pbm->config_space_reg_bits = 12;
|
||||||
|
|
||||||
pbm->index = pci_num_pbms++;
|
pbm->index = pci_num_pbms++;
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user