Commit 3d0f8bc7 authored by Martyn Welch's avatar Martyn Welch Committed by Greg Kroah-Hartman

Staging: vme: Update support for the Universe II VME driver

This patch starts the reworking of the vme_ca91cx42.c for the VME core. The
driver seems to have only supported the Universe II for a long time, I have no
hardware to check to see if it would work on the Universe I.

This patch is compile tested with minimal run-time testing. It enables basic
support for slave and master windows and interrupts. The master windows have
undergone basic run testing.
Signed-off-by: default avatarMartyn Welch <martyn.welch@gefanuc.com>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@suse.de>
parent 42fb5031
comment "VME Bridge Drivers" comment "VME Bridge Drivers"
config VME_CA91CX42 config VME_CA91CX42
tristate "Universe I/II" tristate "Universe II"
depends on BROKEN
help help
If you say Y here you get support for the Tundra CA91C042 (Universe I) If you say Y here you get support for the Tundra CA91C142
and CA91C142 (Universe II) VME bridge chips. (Universe II) VME bridge chip.
config VME_TSI148 config VME_TSI148
tristate "Tempe" tristate "Tempe"
......
...@@ -20,7 +20,6 @@ ...@@ -20,7 +20,6 @@
#include <linux/mm.h> #include <linux/mm.h>
#include <linux/types.h> #include <linux/types.h>
#include <linux/errno.h> #include <linux/errno.h>
#include <linux/proc_fs.h>
#include <linux/pci.h> #include <linux/pci.h>
#include <linux/dma-mapping.h> #include <linux/dma-mapping.h>
#include <linux/poll.h> #include <linux/poll.h>
...@@ -34,1212 +33,1333 @@ ...@@ -34,1212 +33,1333 @@
#include "../vme_bridge.h" #include "../vme_bridge.h"
#include "vme_ca91cx42.h" #include "vme_ca91cx42.h"
extern struct vmeSharedData *vmechip_interboard_data; static int __init ca91cx42_init(void);
extern dma_addr_t vmechip_interboard_datap; static int ca91cx42_probe(struct pci_dev *, const struct pci_device_id *);
extern const int vmechip_revision; static void ca91cx42_remove(struct pci_dev *);
extern const int vmechip_devid; static void __exit ca91cx42_exit(void);
extern const int vmechip_irq;
extern int vmechip_irq_overhead_ticks; struct vme_bridge *ca91cx42_bridge;
extern char *vmechip_baseaddr; wait_queue_head_t dma_queue;
extern const int vme_slotnum; wait_queue_head_t iack_queue;
extern int vme_syscon; wait_queue_head_t lm_queue;
extern unsigned int out_image_va[]; wait_queue_head_t mbox_queue;
extern unsigned int vme_irqlog[8][0x100];
void (*lm_callback[4])(int); /* Called in interrupt handler, be careful! */
static int outCTL[] = { LSI0_CTL, LSI1_CTL, LSI2_CTL, LSI3_CTL, void *crcsr_kernel;
LSI4_CTL, LSI5_CTL, LSI6_CTL, LSI7_CTL dma_addr_t crcsr_bus;
};
struct mutex vme_rmw; /* Only one RMW cycle at a time */
static int outBS[] = { LSI0_BS, LSI1_BS, LSI2_BS, LSI3_BS, struct mutex vme_int; /*
LSI4_BS, LSI5_BS, LSI6_BS, LSI7_BS * Only one VME interrupt can be
}; * generated at a time, provide locking
*/
static int outBD[] = { LSI0_BD, LSI1_BD, LSI2_BD, LSI3_BD, struct mutex vme_irq; /* Locking for VME irq callback configuration */
LSI4_BD, LSI5_BD, LSI6_BD, LSI7_BD
};
static int outTO[] = { LSI0_TO, LSI1_TO, LSI2_TO, LSI3_TO,
LSI4_TO, LSI5_TO, LSI6_TO, LSI7_TO
};
static int inCTL[] = { VSI0_CTL, VSI1_CTL, VSI2_CTL, VSI3_CTL,
VSI4_CTL, VSI5_CTL, VSI6_CTL, VSI7_CTL
};
static int inBS[] = { VSI0_BS, VSI1_BS, VSI2_BS, VSI3_BS,
VSI4_BS, VSI5_BS, VSI6_BS, VSI7_BS
};
static int inBD[] = { VSI0_BD, VSI1_BD, VSI2_BD, VSI3_BD, static char driver_name[] = "vme_ca91cx42";
VSI4_BD, VSI5_BD, VSI6_BD, VSI7_BD
};
static int inTO[] = { VSI0_TO, VSI1_TO, VSI2_TO, VSI3_TO, static struct pci_device_id ca91cx42_ids[] = {
VSI4_TO, VSI5_TO, VSI6_TO, VSI7_TO { PCI_DEVICE(PCI_VENDOR_ID_TUNDRA, PCI_DEVICE_ID_TUNDRA_CA91C142) },
}; { },
static int vmevec[7] = { V1_STATID, V2_STATID, V3_STATID, V4_STATID,
V5_STATID, V6_STATID, V7_STATID
}; };
struct interrupt_counters { static struct pci_driver ca91cx42_driver = {
unsigned int acfail; .name = driver_name,
unsigned int sysfail; .id_table = ca91cx42_ids,
unsigned int sw_int; .probe = ca91cx42_probe,
unsigned int sw_iack; .remove = ca91cx42_remove,
unsigned int verr;
unsigned int lerr;
unsigned int lm;
unsigned int mbox;
unsigned int dma;
unsigned int virq[7];
unsigned int vown;
}; };
extern wait_queue_head_t dma_queue[]; static u32 ca91cx42_DMA_irqhandler(void)
extern wait_queue_head_t lm_queue;
extern wait_queue_head_t mbox_queue;
extern int tb_speed;
unsigned int uni_irq_time;
unsigned int uni_dma_irq_time;
unsigned int uni_lm_event;
static spinlock_t lm_lock = SPIN_LOCK_UNLOCKED;
static struct interrupt_counters Interrupt_counters = { 0, 0,
0, 0, 0, 0,
0, 0, 0,
{0, 0, 0, 0, 0, 0, 0},
0
};
#define read_register(offset) readl(vmechip_baseaddr + offset)
#define write_register(value,offset) writel(value, vmechip_baseaddr + offset)
#define read_register_word(offset) readw(vmechip_baseaddr + offset)
#define write_register_word(value,offset) writew(value, vmechip_baseaddr + offset)
int uni_procinfo(char *buf)
{ {
char *p; wake_up(&dma_queue);
p = buf;
p += sprintf(p, "\n");
{
unsigned long misc_ctl;
misc_ctl = read_register(MISC_CTL);
p += sprintf(p, "MISC_CTL:\t\t\t0x%08lx\n", misc_ctl);
p += sprintf(p, "VME Bus Time Out:\t\t");
switch ((misc_ctl & UNIV_BM_MISC_CTL_VBTO) >>
UNIV_OF_MISC_CTL_VBTO) {
case 0x0:
p += sprintf(p, "Disabled\n");
break;
case 0x1:
p += sprintf(p, "16 us\n");
break;
case 0x2:
p += sprintf(p, "32 us\n");
break;
case 0x3:
p += sprintf(p, "64 us\n");
break;
case 0x4:
p += sprintf(p, "128 us\n");
break;
case 0x5:
p += sprintf(p, "256 us\n");
break;
case 0x6:
p += sprintf(p, "512 us\n");
break;
case 0x7:
p += sprintf(p, "1024 us\n");
break;
default:
p += sprintf(p, "Reserved Value, Undefined\n");
}
p += sprintf(p, "VME Arbitration Time Out:\t");
switch ((misc_ctl & UNIV_BM_MISC_CTL_VARBTO) >>
UNIV_OF_MISC_CTL_VARBTO) {
case 0x0:
p += sprintf(p, "Disabled");
break;
case 0x1:
p += sprintf(p, "16 us");
break;
case 0x2:
p += sprintf(p, "256 us");
break;
default:
p += sprintf(p, "Reserved Value, Undefined");
}
if (misc_ctl & UNIV_BM_MISC_CTL_VARB)
p += sprintf(p, ", Priority Arbitration\n");
else
p += sprintf(p, ", Round Robin Arbitration\n");
p += sprintf(p, "\n");
}
{
unsigned int lmisc;
unsigned int crt;
unsigned int cwt;
lmisc = read_register(LMISC);
p += sprintf(p, "LMISC:\t\t\t\t0x%08x\n", lmisc);
crt = (lmisc & UNIV_BM_LMISC_CRT) >> UNIV_OF_LMISC_CRT;
cwt = (lmisc & UNIV_BM_LMISC_CWT) >> UNIV_OF_LMISC_CWT;
p += sprintf(p, "Coupled Request Timer:\t\t");
switch (crt) {
case 0x0:
p += sprintf(p, "Disabled\n");
break;
case 0x1:
p += sprintf(p, "128 us\n");
break;
case 0x2:
p += sprintf(p, "256 us\n");
break;
case 0x3:
p += sprintf(p, "512 us\n");
break;
case 0x4:
p += sprintf(p, "1024 us\n");
break;
case 0x5:
p += sprintf(p, "2048 us\n");
break;
case 0x6:
p += sprintf(p, "4096 us\n");
break;
default:
p += sprintf(p, "Reserved\n");
}
p += sprintf(p, "Coupled Window Timer:\t\t");
switch (cwt) {
case 0x0:
p += sprintf(p, "Disabled\n");
break;
case 0x1:
p += sprintf(p, "16 PCI Clocks\n");
break;
case 0x2:
p += sprintf(p, "32 PCI Clocks\n");
break;
case 0x3:
p += sprintf(p, "64 PCI Clocks\n");
break;
case 0x4:
p += sprintf(p, "128 PCI Clocks\n");
break;
case 0x5:
p += sprintf(p, "256 PCI Clocks\n");
break;
case 0x6:
p += sprintf(p, "512 PCI Clocks\n");
break;
default:
p += sprintf(p, "Reserved\n");
}
p += sprintf(p, "\n");
}
{
unsigned int mast_ctl;
mast_ctl = read_register(MAST_CTL);
p += sprintf(p, "MAST_CTL:\t\t\t0x%08x\n", mast_ctl);
{
int retries;
retries = ((mast_ctl & UNIV_BM_MAST_CTL_MAXRTRY) return CA91CX42_LINT_DMA;
>> UNIV_OF_MAST_CTL_MAXRTRY) * 64; }
p += sprintf(p, "Max PCI Master Retries:\t\t");
if (retries)
p += sprintf(p, "%d\n", retries);
else
p += sprintf(p, "Forever\n");
}
p += sprintf(p, "Posted Write Transfer Count:\t"); static u32 ca91cx42_LM_irqhandler(u32 stat)
switch ((mast_ctl & UNIV_BM_MAST_CTL_PWON) >> {
UNIV_OF_MAST_CTL_PWON) { int i;
case 0x0: u32 serviced = 0;
p += sprintf(p, "128 Bytes\n");
break;
case 0x1:
p += sprintf(p, "256 Bytes\n");
break;
case 0x2:
p += sprintf(p, "512 Bytes\n");
break;
case 0x3:
p += sprintf(p, "1024 Bytes\n");
break;
case 0x4:
p += sprintf(p, "2048 Bytes\n");
break;
case 0x5:
p += sprintf(p, "4096 Bytes\n");
break;
default:
p += sprintf(p, "Undefined\n");
}
p += sprintf(p, "VMEbus Request Level:\t\t");
switch ((mast_ctl & UNIV_BM_MAST_CTL_VRL) >>
UNIV_OF_MAST_CTL_VRL) {
case 0x0:
p += sprintf(p, "Level 0\n");
case 0x1:
p += sprintf(p, "Level 1\n");
case 0x2:
p += sprintf(p, "Level 2\n");
case 0x3:
p += sprintf(p, "Level 3\n");
}
p += sprintf(p, "VMEbus Request Mode:\t\t");
if (mast_ctl & UNIV_BM_MAST_CTL_VRM)
p += sprintf(p, "Fair Request Mode\n");
else
p += sprintf(p, "Demand Request Mode\n");
p += sprintf(p, "VMEbus Release Mode:\t\t");
if (mast_ctl & UNIV_BM_MAST_CTL_VREL)
p += sprintf(p, "Release on Request\n");
else
p += sprintf(p, "Release when Done\n");
p += sprintf(p, "VMEbus Ownership Bit:\t\t");
if (mast_ctl & UNIV_BM_MAST_CTL_VOWN)
p += sprintf(p, "Acquire and hold VMEbus\n");
else
p += sprintf(p, "Release VMEbus\n");
p += sprintf(p, "VMEbus Ownership Bit Ack:\t");
if (mast_ctl & UNIV_BM_MAST_CTL_VOWN_ACK)
p += sprintf(p, "Owning VMEbus\n");
else
p += sprintf(p, "Not Owning VMEbus\n");
p += sprintf(p, "\n");
}
{
unsigned int misc_stat;
misc_stat = read_register(MISC_STAT); for (i = 0; i < 4; i++) {
p += sprintf(p, "MISC_STAT:\t\t\t0x%08x\n", misc_stat); if (stat & CA91CX42_LINT_LM[i]) {
p += sprintf(p, "Universe BBSY:\t\t\t"); /* We only enable interrupts if the callback is set */
if (misc_stat & UNIV_BM_MISC_STAT_MYBBSY) lm_callback[i](i);
p += sprintf(p, "Negated\n"); serviced |= CA91CX42_LINT_LM[i];
else
p += sprintf(p, "Asserted\n");
p += sprintf(p, "Transmit FIFO:\t\t\t");
if (misc_stat & UNIV_BM_MISC_STAT_TXFE)
p += sprintf(p, "Empty\n");
else
p += sprintf(p, "Not empty\n");
p += sprintf(p, "Receive FIFO:\t\t\t");
if (misc_stat & UNIV_BM_MISC_STAT_RXFE)
p += sprintf(p, "Empty\n");
else
p += sprintf(p, "Not Empty\n");
p += sprintf(p, "\n");
}
p += sprintf(p, "Latency Timer:\t\t\t%02d Clocks\n\n",
(read_register(UNIV_PCI_MISC0) &
UNIV_BM_PCI_MISC0_LTIMER) >> UNIV_OF_PCI_MISC0_LTIMER);
{
unsigned int lint_en;
unsigned int lint_stat;
lint_en = read_register(LINT_EN);
lint_stat = read_register(LINT_STAT);
#define REPORT_IRQ(name,field) \
p += sprintf(p, (lint_en & UNIV_BM_LINT_##name) ? "Enabled" : "Masked"); \
p += sprintf(p, ", triggered %d times", Interrupt_counters.field); \
p += sprintf(p, (lint_stat & UNIV_BM_LINT_##name) ? ", irq now active\n" : "\n");
p += sprintf(p, "ACFAIL Interrupt:\t\t");
REPORT_IRQ(ACFAIL, acfail);
p += sprintf(p, "SYSFAIL Interrupt:\t\t");
REPORT_IRQ(SYSFAIL, sysfail);
p += sprintf(p, "SW_INT Interrupt:\t\t");
REPORT_IRQ(SW_INT, sw_int);
p += sprintf(p, "SW_IACK Interrupt:\t\t");
REPORT_IRQ(SW_IACK, sw_iack);
p += sprintf(p, "VERR Interrupt:\t\t\t");
REPORT_IRQ(VERR, verr);
p += sprintf(p, "LERR Interrupt:\t\t\t");
REPORT_IRQ(LERR, lerr);
p += sprintf(p, "LM Interrupt:\t\t\t");
REPORT_IRQ(LM, lm);
p += sprintf(p, "MBOX Interrupt:\t\t\t");
REPORT_IRQ(MBOX, mbox);
p += sprintf(p, "DMA Interrupt:\t\t\t");
REPORT_IRQ(DMA, dma);
p += sprintf(p, "VIRQ7 Interrupt:\t\t");
REPORT_IRQ(VIRQ7, virq[7 - 1]);
p += sprintf(p, "VIRQ6 Interrupt:\t\t");
REPORT_IRQ(VIRQ6, virq[6 - 1]);
p += sprintf(p, "VIRQ5 Interrupt:\t\t");
REPORT_IRQ(VIRQ5, virq[5 - 1]);
p += sprintf(p, "VIRQ4 Interrupt:\t\t");
REPORT_IRQ(VIRQ4, virq[4 - 1]);
p += sprintf(p, "VIRQ3 Interrupt:\t\t");
REPORT_IRQ(VIRQ3, virq[3 - 1]);
p += sprintf(p, "VIRQ2 Interrupt:\t\t");
REPORT_IRQ(VIRQ2, virq[2 - 1]);
p += sprintf(p, "VIRQ1 Interrupt:\t\t");
REPORT_IRQ(VIRQ1, virq[1 - 1]);
p += sprintf(p, "VOWN Interrupt:\t\t\t");
REPORT_IRQ(VOWN, vown);
p += sprintf(p, "\n");
#undef REPORT_IRQ
}
{
unsigned long vrai_ctl;
vrai_ctl = read_register(VRAI_CTL);
if (vrai_ctl & UNIV_BM_VRAI_CTL_EN) {
unsigned int vrai_bs;
vrai_bs = read_register(VRAI_BS);
p += sprintf(p,
"VME Register Image:\t\tEnabled at VME-Address 0x%x\n",
vrai_bs);
} else
p += sprintf(p, "VME Register Image:\t\tDisabled\n");
}
{
unsigned int slsi;
slsi = read_register(SLSI);
if (slsi & UNIV_BM_SLSI_EN) {
/* Not implemented */
} else {
p += sprintf(p, "Special PCI Slave Image:\tDisabled\n");
} }
} }
{
int i;
for (i = 0; i < (vmechip_revision > 0 ? 8 : 4); i++) { return serviced;
unsigned int ctl, bs, bd, to, vstart, vend; }
ctl = readl(vmechip_baseaddr + outCTL[i]); /* XXX This needs to be split into 4 queues */
bs = readl(vmechip_baseaddr + outBS[i]); static u32 ca91cx42_MB_irqhandler(int mbox_mask)
bd = readl(vmechip_baseaddr + outBD[i]); {
to = readl(vmechip_baseaddr + outTO[i]); wake_up(&mbox_queue);
vstart = bs + to; return CA91CX42_LINT_MBOX;
vend = bd + to; }
p += sprintf(p, "PCI Slave Image %d:\t\t", i); static u32 ca91cx42_IACK_irqhandler(void)
if (ctl & UNIV_BM_LSI_CTL_EN) { {
p += sprintf(p, "Enabled"); wake_up(&iack_queue);
if (ctl & UNIV_BM_LSI_CTL_PWEN)
p += sprintf(p,
", Posted Write Enabled\n");
else
p += sprintf(p, "\n");
p += sprintf(p,
"\t\t\t\tPCI Addresses from 0x%x to 0x%x\n",
bs, bd);
p += sprintf(p,
"\t\t\t\tVME Addresses from 0x%x to 0x%x\n",
vstart, vend);
} else
p += sprintf(p, "Disabled\n");
}
p += sprintf(p, "\n");
}
{
int i;
for (i = 0; i < (vmechip_revision > 0 ? 8 : 4); i++) {
unsigned int ctl, bs, bd, to, vstart, vend;
ctl = readl(vmechip_baseaddr + inCTL[i]);
bs = readl(vmechip_baseaddr + inBS[i]);
bd = readl(vmechip_baseaddr + inBD[i]);
to = readl(vmechip_baseaddr + inTO[i]);
vstart = bs + to;
vend = bd + to;
p += sprintf(p, "VME Slave Image %d:\t\t", i);
if (ctl & UNIV_BM_LSI_CTL_EN) {
p += sprintf(p, "Enabled");
if (ctl & UNIV_BM_LSI_CTL_PWEN)
p += sprintf(p,
", Posted Write Enabled\n");
else
p += sprintf(p, "\n");
p += sprintf(p,
"\t\t\t\tVME Addresses from 0x%x to 0x%x\n",
bs, bd);
p += sprintf(p,
"\t\t\t\tPCI Addresses from 0x%x to 0x%x\n",
vstart, vend);
} else
p += sprintf(p, "Disabled\n");
}
}
return p - buf; return CA91CX42_LINT_SW_IACK;
} }
//---------------------------------------------------------------------------- #if 0
// uni_bus_error_chk() int ca91cx42_bus_error_chk(int clrflag)
//----------------------------------------------------------------------------
int uni_bus_error_chk(int clrflag)
{ {
int tmp; int tmp;
tmp = readl(vmechip_baseaddr + PCI_COMMAND); tmp = ioread32(ca91cx42_bridge->base + PCI_COMMAND);
if (tmp & 0x08000000) { // S_TA is Set if (tmp & 0x08000000) { /* S_TA is Set */
if (clrflag) if (clrflag)
writel(tmp | 0x08000000, iowrite32(tmp | 0x08000000,
vmechip_baseaddr + PCI_COMMAND); ca91cx42_bridge->base + PCI_COMMAND);
return (1); return 1;
} }
return (0); return 0;
}
//-----------------------------------------------------------------------------
// Function : DMA_uni_irqhandler
// Inputs : void
// Outputs : void
// Description: Saves DMA completion timestamp and then wakes up DMA queue
//-----------------------------------------------------------------------------
static void DMA_uni_irqhandler(void)
{
uni_dma_irq_time = uni_irq_time;
wake_up(&dma_queue[0]);
} }
#endif
//----------------------------------------------------------------------------- static u32 ca91cx42_VERR_irqhandler(void)
// Function : LERR_uni_irqhandler
// Inputs : void
// Outputs : void
// Description:
//-----------------------------------------------------------------------------
static void LERR_uni_irqhandler(void)
{ {
int val; int val;
val = readl(vmechip_baseaddr + DGCS); val = ioread32(ca91cx42_bridge->base + DGCS);
if (!(val & 0x00000800)) { if (!(val & 0x00000800)) {
printk(KERN_ERR printk(KERN_ERR "ca91c042: ca91cx42_VERR_irqhandler DMA Read "
"ca91c042: LERR_uni_irqhandler DMA Read Error DGCS=%08X\n", "Error DGCS=%08X\n", val);
val);
} }
return CA91CX42_LINT_VERR;
} }
//----------------------------------------------------------------------------- static u32 ca91cx42_LERR_irqhandler(void)
// Function : VERR_uni_irqhandler
// Inputs : void
// Outputs : void
// Description:
//-----------------------------------------------------------------------------
static void VERR_uni_irqhandler(void)
{ {
int val; int val;
val = readl(vmechip_baseaddr + DGCS); val = ioread32(ca91cx42_bridge->base + DGCS);
if (!(val & 0x00000800)) { if (!(val & 0x00000800)) {
printk(KERN_ERR printk(KERN_ERR "ca91c042: ca91cx42_LERR_irqhandler DMA Read "
"ca91c042: VERR_uni_irqhandler DMA Read Error DGCS=%08X\n", "Error DGCS=%08X\n", val);
val);
}
}
//-----------------------------------------------------------------------------
// Function : MB_uni_irqhandler
// Inputs : void
// Outputs : void
// Description:
//-----------------------------------------------------------------------------
static void MB_uni_irqhandler(int mbox_mask)
{
if (vmechip_irq_overhead_ticks != 0) {
wake_up(&mbox_queue);
} }
}
//----------------------------------------------------------------------------- return CA91CX42_LINT_LERR;
// Function : LM_uni_irqhandler
// Inputs : void
// Outputs : void
// Description:
//-----------------------------------------------------------------------------
static void LM_uni_irqhandler(int lm_mask)
{
uni_lm_event = lm_mask;
wake_up(&lm_queue);
} }
//-----------------------------------------------------------------------------
// Function : VIRQ_uni_irqhandler static u32 ca91cx42_VIRQ_irqhandler(int stat)
// Inputs : void
// Outputs : void
// Description:
//-----------------------------------------------------------------------------
static void VIRQ_uni_irqhandler(int virq_mask)
{ {
int iackvec, i; int vec, i, serviced = 0;
void (*call)(int, int, void *);
void *priv_data;
for (i = 7; i > 0; i--) { for (i = 7; i > 0; i--) {
if (virq_mask & (1 << i)) { if (stat & (1 << i)) {
Interrupt_counters.virq[i - 1]++; vec = ioread32(ca91cx42_bridge->base +
iackvec = readl(vmechip_baseaddr + vmevec[i - 1]); CA91CX42_V_STATID[i]) & 0xff;
vme_irqlog[i][iackvec]++;
call = ca91cx42_bridge->irq[i - 1].callback[vec].func;
priv_data =
ca91cx42_bridge->irq[i - 1].callback[vec].priv_data;
if (call != NULL)
call(i, vec, priv_data);
else
printk("Spurilous VME interrupt, level:%x, "
"vector:%x\n", i, vec);
serviced |= (1 << i);
} }
} }
return serviced;
} }
//----------------------------------------------------------------------------- static irqreturn_t ca91cx42_irqhandler(int irq, void *dev_id)
// Function : uni_irqhandler
// Inputs : int irq, void *dev_id, struct pt_regs *regs
// Outputs : void
// Description:
//-----------------------------------------------------------------------------
static irqreturn_t uni_irqhandler(int irq, void *dev_id)
{ {
long stat, enable; u32 stat, enable, serviced = 0;
if (dev_id != vmechip_baseaddr) if (dev_id != ca91cx42_bridge->base)
return IRQ_NONE; return IRQ_NONE;
uni_irq_time = get_tbl(); enable = ioread32(ca91cx42_bridge->base + LINT_EN);
stat = ioread32(ca91cx42_bridge->base + LINT_STAT);
stat = readl(vmechip_baseaddr + LINT_STAT); /* Only look at unmasked interrupts */
writel(stat, vmechip_baseaddr + LINT_STAT); // Clear all pending ints stat &= enable;
enable = readl(vmechip_baseaddr + LINT_EN);
stat = stat & enable; if (unlikely(!stat))
if (stat & 0x0100) { return IRQ_NONE;
Interrupt_counters.dma++;
DMA_uni_irqhandler(); if (stat & CA91CX42_LINT_DMA)
} serviced |= ca91cx42_DMA_irqhandler();
if (stat & 0x0200) { if (stat & (CA91CX42_LINT_LM0 | CA91CX42_LINT_LM1 | CA91CX42_LINT_LM2 |
Interrupt_counters.lerr++; CA91CX42_LINT_LM3))
LERR_uni_irqhandler(); serviced |= ca91cx42_LM_irqhandler(stat);
} if (stat & CA91CX42_LINT_MBOX)
if (stat & 0x0400) { serviced |= ca91cx42_MB_irqhandler(stat);
Interrupt_counters.verr++; if (stat & CA91CX42_LINT_SW_IACK)
VERR_uni_irqhandler(); serviced |= ca91cx42_IACK_irqhandler();
} if (stat & CA91CX42_LINT_VERR)
if (stat & 0xF0000) { serviced |= ca91cx42_VERR_irqhandler();
Interrupt_counters.mbox++; if (stat & CA91CX42_LINT_LERR)
MB_uni_irqhandler((stat & 0xF0000) >> 16); serviced |= ca91cx42_LERR_irqhandler();
} if (stat & (CA91CX42_LINT_VIRQ1 | CA91CX42_LINT_VIRQ2 |
if (stat & 0xF00000) { CA91CX42_LINT_VIRQ3 | CA91CX42_LINT_VIRQ4 |
Interrupt_counters.lm++; CA91CX42_LINT_VIRQ5 | CA91CX42_LINT_VIRQ6 |
LM_uni_irqhandler((stat & 0xF00000) >> 20); CA91CX42_LINT_VIRQ7))
} serviced |= ca91cx42_VIRQ_irqhandler(stat);
if (stat & 0x0000FE) {
VIRQ_uni_irqhandler(stat & 0x0000FE); /* Clear serviced interrupts */
} iowrite32(stat, ca91cx42_bridge->base + LINT_STAT);
if (stat & UNIV_BM_LINT_ACFAIL) {
Interrupt_counters.acfail++;
}
if (stat & UNIV_BM_LINT_SYSFAIL) {
Interrupt_counters.sysfail++;
}
if (stat & UNIV_BM_LINT_SW_INT) {
Interrupt_counters.sw_int++;
}
if (stat & UNIV_BM_LINT_SW_IACK) {
Interrupt_counters.sw_iack++;
}
if (stat & UNIV_BM_LINT_VOWN) {
Interrupt_counters.vown++;
}
return IRQ_HANDLED; return IRQ_HANDLED;
} }
//----------------------------------------------------------------------------- static int ca91cx42_irq_init(struct vme_bridge *bridge)
// Function : uni_generate_irq
// Description:
//-----------------------------------------------------------------------------
int uni_generate_irq(virqInfo_t * vmeIrq)
{ {
int timeout; int result, tmp;
int looptimeout; struct pci_dev *pdev;
timeout = vmeIrq->waitTime; /* Need pdev */
if (timeout == 0) { pdev = container_of(bridge->parent, struct pci_dev, dev);
timeout++; // Wait at least 1 tick...
}
looptimeout = HZ / 20; // try for 1/20 second
vmeIrq->timeOutFlag = 0; /* Initialise list for VME bus errors */
INIT_LIST_HEAD(&(bridge->vme_errors));
// Validate & setup vector register. /* Disable interrupts from PCI to VME */
if (vmeIrq->vector & 1) { // Universe can only generate even vectors iowrite32(0, bridge->base + VINT_EN);
return (-EINVAL);
}
writel(vmeIrq->vector << 24, vmechip_baseaddr + STATID);
// Assert VMEbus IRQ /* Disable PCI interrupts */
writel(1 << (vmeIrq->level + 24), vmechip_baseaddr + VINT_EN); iowrite32(0, bridge->base + LINT_EN);
/* Clear Any Pending PCI Interrupts */
iowrite32(0x00FFFFFF, bridge->base + LINT_STAT);
// Wait for syscon to do iack result = request_irq(pdev->irq, ca91cx42_irqhandler, IRQF_SHARED,
while (readl(vmechip_baseaddr + VINT_STAT) & driver_name, pdev);
(1 << (vmeIrq->level + 24))) { if (result) {
set_current_state(TASK_INTERRUPTIBLE); dev_err(&pdev->dev, "Can't get assigned pci irq vector %02X\n",
schedule_timeout(looptimeout); pdev->irq);
timeout = timeout - looptimeout; return result;
if (timeout <= 0) {
vmeIrq->timeOutFlag = 1;
break;
}
} }
// Clear VMEbus IRQ bit /* Ensure all interrupts are mapped to PCI Interrupt 0 */
writel(0, vmechip_baseaddr + VINT_EN); iowrite32(0, bridge->base + LINT_MAP0);
iowrite32(0, bridge->base + LINT_MAP1);
iowrite32(0, bridge->base + LINT_MAP2);
/* Enable DMA, mailbox & LM Interrupts */
tmp = CA91CX42_LINT_MBOX3 | CA91CX42_LINT_MBOX2 | CA91CX42_LINT_MBOX1 |
CA91CX42_LINT_MBOX0 | CA91CX42_LINT_SW_IACK |
CA91CX42_LINT_VERR | CA91CX42_LINT_LERR | CA91CX42_LINT_DMA;
return (0); iowrite32(tmp, bridge->base + LINT_EN);
return 0;
} }
//----------------------------------------------------------------------------- static void ca91cx42_irq_exit(struct pci_dev *pdev)
// Function : uni_set_arbiter
// Description:
//-----------------------------------------------------------------------------
int uni_set_arbiter(vmeArbiterCfg_t * vmeArb)
{ {
int temp_ctl = 0; /* Disable interrupts from PCI to VME */
int vbto = 0; iowrite32(0, ca91cx42_bridge->base + VINT_EN);
temp_ctl = readl(vmechip_baseaddr + MISC_CTL); /* Disable PCI interrupts */
temp_ctl &= 0x00FFFFFF; iowrite32(0, ca91cx42_bridge->base + LINT_EN);
/* Clear Any Pending PCI Interrupts */
iowrite32(0x00FFFFFF, ca91cx42_bridge->base + LINT_STAT);
if (vmeArb->globalTimeoutTimer == 0xFFFFFFFF) { free_irq(pdev->irq, pdev);
vbto = 7; }
} else if (vmeArb->globalTimeoutTimer > 1024) {
return (-EINVAL);
} else if (vmeArb->globalTimeoutTimer == 0) {
vbto = 0;
} else {
vbto = 1;
while ((16 * (1 << (vbto - 1))) < vmeArb->globalTimeoutTimer) {
vbto += 1;
}
}
temp_ctl |= (vbto << 28);
if (vmeArb->arbiterMode == VME_PRIORITY_MODE) { /*
temp_ctl |= 1 << 26; * Set up an VME interrupt
} */
int ca91cx42_request_irq(int level, int statid,
void (*callback)(int level, int vector, void *priv_data),
void *priv_data)
{
u32 tmp;
if (vmeArb->arbiterTimeoutFlag) { mutex_lock(&(vme_irq));
temp_ctl |= 2 << 24;
if (ca91cx42_bridge->irq[level - 1].callback[statid].func) {
mutex_unlock(&(vme_irq));
printk("VME Interrupt already taken\n");
return -EBUSY;
} }
writel(temp_ctl, vmechip_baseaddr + MISC_CTL);
return (0); ca91cx42_bridge->irq[level - 1].count++;
ca91cx42_bridge->irq[level - 1].callback[statid].priv_data = priv_data;
ca91cx42_bridge->irq[level - 1].callback[statid].func = callback;
/* Enable IRQ level */
tmp = ioread32(ca91cx42_bridge->base + LINT_EN);
tmp |= CA91CX42_LINT_VIRQ[level];
iowrite32(tmp, ca91cx42_bridge->base + LINT_EN);
mutex_unlock(&(vme_irq));
return 0;
} }
//----------------------------------------------------------------------------- /*
// Function : uni_get_arbiter * Free VME interrupt
// Description: */
//----------------------------------------------------------------------------- void ca91cx42_free_irq(int level, int statid)
int uni_get_arbiter(vmeArbiterCfg_t * vmeArb)
{ {
int temp_ctl = 0; u32 tmp;
int vbto = 0; struct pci_dev *pdev;
temp_ctl = readl(vmechip_baseaddr + MISC_CTL); mutex_lock(&(vme_irq));
vbto = (temp_ctl >> 28) & 0xF; ca91cx42_bridge->irq[level - 1].count--;
if (vbto != 0) {
vmeArb->globalTimeoutTimer = (16 * (1 << (vbto - 1)));
}
if (temp_ctl & (1 << 26)) { /* Disable IRQ level if no more interrupts attached at this level*/
vmeArb->arbiterMode = VME_PRIORITY_MODE; if (ca91cx42_bridge->irq[level - 1].count == 0) {
} else { tmp = ioread32(ca91cx42_bridge->base + LINT_EN);
vmeArb->arbiterMode = VME_R_ROBIN_MODE; tmp &= ~CA91CX42_LINT_VIRQ[level];
} iowrite32(tmp, ca91cx42_bridge->base + LINT_EN);
if (temp_ctl & (3 << 24)) { pdev = container_of(ca91cx42_bridge->parent, struct pci_dev,
vmeArb->arbiterTimeoutFlag = 1; dev);
synchronize_irq(pdev->irq);
} }
return (0);
ca91cx42_bridge->irq[level - 1].callback[statid].func = NULL;
ca91cx42_bridge->irq[level - 1].callback[statid].priv_data = NULL;
mutex_unlock(&(vme_irq));
} }
//----------------------------------------------------------------------------- int ca91cx42_generate_irq(int level, int statid)
// Function : uni_set_requestor
// Description:
//-----------------------------------------------------------------------------
int uni_set_requestor(vmeRequesterCfg_t * vmeReq)
{ {
int temp_ctl = 0; u32 tmp;
temp_ctl = readl(vmechip_baseaddr + MAST_CTL); /* Universe can only generate even vectors */
temp_ctl &= 0xFF0FFFFF; if (statid & 1)
return -EINVAL;
if (vmeReq->releaseMode == 1) { mutex_lock(&(vme_int));
temp_ctl |= (1 << 20);
}
if (vmeReq->fairMode == 1) { tmp = ioread32(ca91cx42_bridge->base + VINT_EN);
temp_ctl |= (1 << 21);
}
temp_ctl |= (vmeReq->requestLevel << 22); /* Set Status/ID */
iowrite32(statid << 24, ca91cx42_bridge->base + STATID);
writel(temp_ctl, vmechip_baseaddr + MAST_CTL); /* Assert VMEbus IRQ */
return (0); tmp = tmp | (1 << (level + 24));
iowrite32(tmp, ca91cx42_bridge->base + VINT_EN);
/* Wait for IACK */
wait_event_interruptible(iack_queue, 0);
/* Return interrupt to low state */
tmp = ioread32(ca91cx42_bridge->base + VINT_EN);
tmp = tmp & ~(1 << (level + 24));
iowrite32(tmp, ca91cx42_bridge->base + VINT_EN);
mutex_unlock(&(vme_int));
return 0;
} }
//----------------------------------------------------------------------------- int ca91cx42_slave_set(struct vme_slave_resource *image, int enabled,
// Function : uni_get_requestor unsigned long long vme_base, unsigned long long size,
// Description: dma_addr_t pci_base, vme_address_t aspace, vme_cycle_t cycle)
//-----------------------------------------------------------------------------
int uni_get_requestor(vmeRequesterCfg_t * vmeReq)
{ {
int temp_ctl = 0; unsigned int i, addr = 0, granularity = 0;
unsigned int temp_ctl = 0;
unsigned int vme_bound, pci_offset;
temp_ctl = readl(vmechip_baseaddr + MAST_CTL); i = image->number;
if (temp_ctl & (1 << 20)) { switch (aspace) {
vmeReq->releaseMode = 1; case VME_A16:
addr |= CA91CX42_VSI_CTL_VAS_A16;
break;
case VME_A24:
addr |= CA91CX42_VSI_CTL_VAS_A24;
break;
case VME_A32:
addr |= CA91CX42_VSI_CTL_VAS_A32;
break;
case VME_USER1:
addr |= CA91CX42_VSI_CTL_VAS_USER1;
break;
case VME_USER2:
addr |= CA91CX42_VSI_CTL_VAS_USER2;
break;
case VME_A64:
case VME_CRCSR:
case VME_USER3:
case VME_USER4:
default:
printk(KERN_ERR "Invalid address space\n");
return -EINVAL;
break;
} }
if (temp_ctl & (1 << 21)) { /*
vmeReq->fairMode = 1; * Bound address is a valid address for the window, adjust
* accordingly
*/
vme_bound = vme_base + size - granularity;
pci_offset = pci_base - vme_base;
/* XXX Need to check that vme_base, vme_bound and pci_offset aren't
* too big for registers
*/
if ((i == 0) || (i == 4))
granularity = 0x1000;
else
granularity = 0x10000;
if (vme_base & (granularity - 1)) {
printk(KERN_ERR "Invalid VME base alignment\n");
return -EINVAL;
}
if (vme_bound & (granularity - 1)) {
printk(KERN_ERR "Invalid VME bound alignment\n");
return -EINVAL;
}
if (pci_offset & (granularity - 1)) {
printk(KERN_ERR "Invalid PCI Offset alignment\n");
return -EINVAL;
} }
vmeReq->requestLevel = (temp_ctl & 0xC00000) >> 22; /* Disable while we are mucking around */
temp_ctl = ioread32(ca91cx42_bridge->base + CA91CX42_VSI_CTL[i]);
temp_ctl &= ~CA91CX42_VSI_CTL_EN;
iowrite32(temp_ctl, ca91cx42_bridge->base + CA91CX42_VSI_CTL[i]);
/* Setup mapping */
iowrite32(vme_base, ca91cx42_bridge->base + CA91CX42_VSI_BS[i]);
iowrite32(vme_bound, ca91cx42_bridge->base + CA91CX42_VSI_BD[i]);
iowrite32(pci_offset, ca91cx42_bridge->base + CA91CX42_VSI_TO[i]);
/* XXX Prefetch stuff currently unsupported */
#if 0
if (vmeIn->wrPostEnable)
temp_ctl |= CA91CX42_VSI_CTL_PWEN;
if (vmeIn->prefetchEnable)
temp_ctl |= CA91CX42_VSI_CTL_PREN;
if (vmeIn->rmwLock)
temp_ctl |= CA91CX42_VSI_CTL_LLRMW;
if (vmeIn->data64BitCapable)
temp_ctl |= CA91CX42_VSI_CTL_LD64EN;
#endif
return (0); /* Setup address space */
temp_ctl &= ~CA91CX42_VSI_CTL_VAS_M;
temp_ctl |= addr;
/* Setup cycle types */
temp_ctl &= ~(CA91CX42_VSI_CTL_PGM_M | CA91CX42_VSI_CTL_SUPER_M);
if (cycle & VME_SUPER)
temp_ctl |= CA91CX42_VSI_CTL_SUPER_SUPR;
if (cycle & VME_USER)
temp_ctl |= CA91CX42_VSI_CTL_SUPER_NPRIV;
if (cycle & VME_PROG)
temp_ctl |= CA91CX42_VSI_CTL_PGM_PGM;
if (cycle & VME_DATA)
temp_ctl |= CA91CX42_VSI_CTL_PGM_DATA;
/* Write ctl reg without enable */
iowrite32(temp_ctl, ca91cx42_bridge->base + CA91CX42_VSI_CTL[i]);
if (enabled)
temp_ctl |= CA91CX42_VSI_CTL_EN;
iowrite32(temp_ctl, ca91cx42_bridge->base + CA91CX42_VSI_CTL[i]);
return 0;
} }
//----------------------------------------------------------------------------- int ca91cx42_slave_get(struct vme_slave_resource *image, int *enabled,
// Function : uni_set_in_bound unsigned long long *vme_base, unsigned long long *size,
// Description: dma_addr_t *pci_base, vme_address_t *aspace, vme_cycle_t *cycle)
//-----------------------------------------------------------------------------
int uni_set_in_bound(vmeInWindowCfg_t * vmeIn)
{ {
int temp_ctl = 0; unsigned int i, granularity = 0, ctl = 0;
unsigned long long vme_bound, pci_offset;
i = image->number;
if ((i == 0) || (i == 4))
granularity = 0x1000;
else
granularity = 0x10000;
/* Read Registers */
ctl = ioread32(ca91cx42_bridge->base + CA91CX42_VSI_CTL[i]);
*vme_base = ioread32(ca91cx42_bridge->base + CA91CX42_VSI_BS[i]);
vme_bound = ioread32(ca91cx42_bridge->base + CA91CX42_VSI_BD[i]);
pci_offset = ioread32(ca91cx42_bridge->base + CA91CX42_VSI_TO[i]);
*pci_base = (dma_addr_t)vme_base + pci_offset;
*size = (unsigned long long)((vme_bound - *vme_base) + granularity);
*enabled = 0;
*aspace = 0;
*cycle = 0;
if (ctl & CA91CX42_VSI_CTL_EN)
*enabled = 1;
if ((ctl & CA91CX42_VSI_CTL_VAS_M) == CA91CX42_VSI_CTL_VAS_A16)
*aspace = VME_A16;
if ((ctl & CA91CX42_VSI_CTL_VAS_M) == CA91CX42_VSI_CTL_VAS_A24)
*aspace = VME_A24;
if ((ctl & CA91CX42_VSI_CTL_VAS_M) == CA91CX42_VSI_CTL_VAS_A32)
*aspace = VME_A32;
if ((ctl & CA91CX42_VSI_CTL_VAS_M) == CA91CX42_VSI_CTL_VAS_USER1)
*aspace = VME_USER1;
if ((ctl & CA91CX42_VSI_CTL_VAS_M) == CA91CX42_VSI_CTL_VAS_USER2)
*aspace = VME_USER2;
if (ctl & CA91CX42_VSI_CTL_SUPER_SUPR)
*cycle |= VME_SUPER;
if (ctl & CA91CX42_VSI_CTL_SUPER_NPRIV)
*cycle |= VME_USER;
if (ctl & CA91CX42_VSI_CTL_PGM_PGM)
*cycle |= VME_PROG;
if (ctl & CA91CX42_VSI_CTL_PGM_DATA)
*cycle |= VME_DATA;
return 0;
}
/*
* Allocate and map PCI Resource
*/
static int ca91cx42_alloc_resource(struct vme_master_resource *image,
unsigned long long size)
{
unsigned long long existing_size;
int retval = 0;
struct pci_dev *pdev;
/* Find pci_dev container of dev */
if (ca91cx42_bridge->parent == NULL) {
printk(KERN_ERR "Dev entry NULL\n");
return -EINVAL;
}
pdev = container_of(ca91cx42_bridge->parent, struct pci_dev, dev);
existing_size = (unsigned long long)(image->pci_resource.end -
image->pci_resource.start);
/* If the existing size is OK, return */
if (existing_size == (size - 1))
return 0;
if (existing_size != 0) {
iounmap(image->kern_base);
image->kern_base = NULL;
if (image->pci_resource.name != NULL)
kfree(image->pci_resource.name);
release_resource(&(image->pci_resource));
memset(&(image->pci_resource), 0, sizeof(struct resource));
}
if (image->pci_resource.name == NULL) {
image->pci_resource.name = kmalloc(VMENAMSIZ+3, GFP_KERNEL);
if (image->pci_resource.name == NULL) {
printk(KERN_ERR "Unable to allocate memory for resource"
" name\n");
retval = -ENOMEM;
goto err_name;
}
}
sprintf((char *)image->pci_resource.name, "%s.%d",
ca91cx42_bridge->name, image->number);
image->pci_resource.start = 0;
image->pci_resource.end = (unsigned long)size;
image->pci_resource.flags = IORESOURCE_MEM;
retval = pci_bus_alloc_resource(pdev->bus,
&(image->pci_resource), size, size, PCIBIOS_MIN_MEM,
0, NULL, NULL);
if (retval) {
printk(KERN_ERR "Failed to allocate mem resource for "
"window %d size 0x%lx start 0x%lx\n",
image->number, (unsigned long)size,
(unsigned long)image->pci_resource.start);
goto err_resource;
}
image->kern_base = ioremap_nocache(
image->pci_resource.start, size);
if (image->kern_base == NULL) {
printk(KERN_ERR "Failed to remap resource\n");
retval = -ENOMEM;
goto err_remap;
}
return 0;
iounmap(image->kern_base);
image->kern_base = NULL;
err_remap:
release_resource(&(image->pci_resource));
err_resource:
kfree(image->pci_resource.name);
memset(&(image->pci_resource), 0, sizeof(struct resource));
err_name:
return retval;
}
/*
* * Free and unmap PCI Resource
* */
static void ca91cx42_free_resource(struct vme_master_resource *image)
{
iounmap(image->kern_base);
image->kern_base = NULL;
release_resource(&(image->pci_resource));
kfree(image->pci_resource.name);
memset(&(image->pci_resource), 0, sizeof(struct resource));
}
// Verify input data int ca91cx42_master_set(struct vme_master_resource *image, int enabled,
if (vmeIn->windowNbr > 7) { unsigned long long vme_base, unsigned long long size,
return (-EINVAL); vme_address_t aspace, vme_cycle_t cycle, vme_width_t dwidth)
{
int retval = 0;
unsigned int i;
unsigned int temp_ctl = 0;
unsigned long long pci_bound, vme_offset, pci_base;
/* Verify input data */
if (vme_base & 0xFFF) {
printk(KERN_ERR "Invalid VME Window alignment\n");
retval = -EINVAL;
goto err_window;
} }
if ((vmeIn->vmeAddrU) || (vmeIn->windowSizeU) || (vmeIn->pciAddrU)) { if (size & 0xFFF) {
return (-EINVAL); printk(KERN_ERR "Invalid VME Window alignment\n");
retval = -EINVAL;
goto err_window;
} }
if ((vmeIn->vmeAddrL & 0xFFF) ||
(vmeIn->windowSizeL & 0xFFF) || (vmeIn->pciAddrL & 0xFFF)) { spin_lock(&(image->lock));
return (-EINVAL);
/* XXX We should do this much later, so that we can exit without
* needing to redo the mapping...
*/
/*
* Let's allocate the resource here rather than further up the stack as
* it avoids pushing loads of bus dependant stuff up the stack
*/
retval = ca91cx42_alloc_resource(image, size);
if (retval) {
spin_unlock(&(image->lock));
printk(KERN_ERR "Unable to allocate memory for resource "
"name\n");
retval = -ENOMEM;
goto err_res;
} }
if (vmeIn->bcastRespond2esst) { pci_base = (unsigned long long)image->pci_resource.start;
return (-EINVAL);
/*
* Bound address is a valid address for the window, adjust
* according to window granularity.
*/
pci_bound = pci_base + (size - 0x1000);
vme_offset = vme_base - pci_base;
i = image->number;
/* Disable while we are mucking around */
temp_ctl = ioread32(ca91cx42_bridge->base + CA91CX42_LSI_CTL[i]);
temp_ctl &= ~CA91CX42_LSI_CTL_EN;
iowrite32(temp_ctl, ca91cx42_bridge->base + CA91CX42_LSI_CTL[i]);
/* XXX Prefetch stuff currently unsupported */
#if 0
if (vmeOut->wrPostEnable)
temp_ctl |= 0x40000000;
#endif
/* Setup cycle types */
temp_ctl &= ~CA91CX42_LSI_CTL_VCT_M;
if (cycle & VME_BLT)
temp_ctl |= CA91CX42_LSI_CTL_VCT_BLT;
if (cycle & VME_MBLT)
temp_ctl |= CA91CX42_LSI_CTL_VCT_MBLT;
/* Setup data width */
temp_ctl &= ~CA91CX42_LSI_CTL_VDW_M;
switch (dwidth) {
case VME_D8:
temp_ctl |= CA91CX42_LSI_CTL_VDW_D8;
break;
case VME_D16:
temp_ctl |= CA91CX42_LSI_CTL_VDW_D16;
break;
case VME_D32:
temp_ctl |= CA91CX42_LSI_CTL_VDW_D32;
break;
case VME_D64:
temp_ctl |= CA91CX42_LSI_CTL_VDW_D64;
break;
default:
spin_unlock(&(image->lock));
printk(KERN_ERR "Invalid data width\n");
retval = -EINVAL;
goto err_dwidth;
break;
} }
switch (vmeIn->addrSpace) {
case VME_A64: /* Setup address space */
case VME_CRCSR: temp_ctl &= ~CA91CX42_LSI_CTL_VAS_M;
case VME_USER3: switch (aspace) {
case VME_USER4:
return (-EINVAL);
case VME_A16: case VME_A16:
temp_ctl |= 0x00000; temp_ctl |= CA91CX42_LSI_CTL_VAS_A16;
break; break;
case VME_A24: case VME_A24:
temp_ctl |= 0x10000; temp_ctl |= CA91CX42_LSI_CTL_VAS_A24;
break; break;
case VME_A32: case VME_A32:
temp_ctl |= 0x20000; temp_ctl |= CA91CX42_LSI_CTL_VAS_A32;
break;
case VME_CRCSR:
temp_ctl |= CA91CX42_LSI_CTL_VAS_CRCSR;
break; break;
case VME_USER1: case VME_USER1:
temp_ctl |= 0x60000; temp_ctl |= CA91CX42_LSI_CTL_VAS_USER1;
break; break;
case VME_USER2: case VME_USER2:
temp_ctl |= 0x70000; temp_ctl |= CA91CX42_LSI_CTL_VAS_USER2;
break;
case VME_A64:
case VME_USER3:
case VME_USER4:
default:
spin_unlock(&(image->lock));
printk(KERN_ERR "Invalid address space\n");
retval = -EINVAL;
goto err_aspace;
break; break;
} }
// Disable while we are mucking around temp_ctl &= ~(CA91CX42_LSI_CTL_PGM_M | CA91CX42_LSI_CTL_SUPER_M);
writel(0x00000000, vmechip_baseaddr + inCTL[vmeIn->windowNbr]); if (cycle & VME_SUPER)
writel(vmeIn->vmeAddrL, vmechip_baseaddr + inBS[vmeIn->windowNbr]); temp_ctl |= CA91CX42_LSI_CTL_SUPER_SUPR;
writel(vmeIn->vmeAddrL + vmeIn->windowSizeL, if (cycle & VME_PROG)
vmechip_baseaddr + inBD[vmeIn->windowNbr]); temp_ctl |= CA91CX42_LSI_CTL_PGM_PGM;
writel(vmeIn->pciAddrL - vmeIn->vmeAddrL,
vmechip_baseaddr + inTO[vmeIn->windowNbr]);
// Setup CTL register. /* Setup mapping */
if (vmeIn->wrPostEnable) iowrite32(pci_base, ca91cx42_bridge->base + CA91CX42_LSI_BS[i]);
temp_ctl |= 0x40000000; iowrite32(pci_bound, ca91cx42_bridge->base + CA91CX42_LSI_BD[i]);
if (vmeIn->prefetchEnable) iowrite32(vme_offset, ca91cx42_bridge->base + CA91CX42_LSI_TO[i]);
temp_ctl |= 0x20000000;
if (vmeIn->rmwLock) /* Write ctl reg without enable */
temp_ctl |= 0x00000040; iowrite32(temp_ctl, ca91cx42_bridge->base + CA91CX42_LSI_CTL[i]);
if (vmeIn->data64BitCapable)
temp_ctl |= 0x00000080; if (enabled)
if (vmeIn->userAccessType & VME_USER) temp_ctl |= CA91CX42_LSI_CTL_EN;
temp_ctl |= 0x00100000;
if (vmeIn->userAccessType & VME_SUPER) iowrite32(temp_ctl, ca91cx42_bridge->base + CA91CX42_LSI_CTL[i]);
temp_ctl |= 0x00200000;
if (vmeIn->dataAccessType & VME_DATA) spin_unlock(&(image->lock));
temp_ctl |= 0x00400000; return 0;
if (vmeIn->dataAccessType & VME_PROG)
temp_ctl |= 0x00800000; err_aspace:
err_dwidth:
ca91cx42_free_resource(image);
err_res:
err_window:
return retval;
}
int __ca91cx42_master_get(struct vme_master_resource *image, int *enabled,
unsigned long long *vme_base, unsigned long long *size,
vme_address_t *aspace, vme_cycle_t *cycle, vme_width_t *dwidth)
{
unsigned int i, ctl;
unsigned long long pci_base, pci_bound, vme_offset;
i = image->number;
ctl = ioread32(ca91cx42_bridge->base + CA91CX42_LSI_CTL[i]);
pci_base = ioread32(ca91cx42_bridge->base + CA91CX42_LSI_BS[i]);
vme_offset = ioread32(ca91cx42_bridge->base + CA91CX42_LSI_TO[i]);
pci_bound = ioread32(ca91cx42_bridge->base + CA91CX42_LSI_BD[i]);
*vme_base = pci_base + vme_offset;
*size = (pci_bound - pci_base) + 0x1000;
*enabled = 0;
*aspace = 0;
*cycle = 0;
*dwidth = 0;
if (ctl & CA91CX42_LSI_CTL_EN)
*enabled = 1;
/* Setup address space */
switch (ctl & CA91CX42_LSI_CTL_VAS_M) {
case CA91CX42_LSI_CTL_VAS_A16:
*aspace = VME_A16;
break;
case CA91CX42_LSI_CTL_VAS_A24:
*aspace = VME_A24;
break;
case CA91CX42_LSI_CTL_VAS_A32:
*aspace = VME_A32;
break;
case CA91CX42_LSI_CTL_VAS_CRCSR:
*aspace = VME_CRCSR;
break;
case CA91CX42_LSI_CTL_VAS_USER1:
*aspace = VME_USER1;
break;
case CA91CX42_LSI_CTL_VAS_USER2:
*aspace = VME_USER2;
break;
}
/* XXX Not sure howto check for MBLT */
/* Setup cycle types */
if (ctl & CA91CX42_LSI_CTL_VCT_BLT)
*cycle |= VME_BLT;
else
*cycle |= VME_SCT;
if (ctl & CA91CX42_LSI_CTL_SUPER_SUPR)
*cycle |= VME_SUPER;
else
*cycle |= VME_USER;
if (ctl & CA91CX42_LSI_CTL_PGM_PGM)
*cycle = VME_PROG;
else
*cycle = VME_DATA;
/* Setup data width */
switch (ctl & CA91CX42_LSI_CTL_VDW_M) {
case CA91CX42_LSI_CTL_VDW_D8:
*dwidth = VME_D8;
break;
case CA91CX42_LSI_CTL_VDW_D16:
*dwidth = VME_D16;
break;
case CA91CX42_LSI_CTL_VDW_D32:
*dwidth = VME_D32;
break;
case CA91CX42_LSI_CTL_VDW_D64:
*dwidth = VME_D64;
break;
}
/* XXX Prefetch stuff currently unsupported */
#if 0
if (ctl & 0x40000000)
vmeOut->wrPostEnable = 1;
#endif
return 0;
}
int ca91cx42_master_get(struct vme_master_resource *image, int *enabled,
unsigned long long *vme_base, unsigned long long *size,
vme_address_t *aspace, vme_cycle_t *cycle, vme_width_t *dwidth)
{
int retval;
spin_lock(&(image->lock));
retval = __ca91cx42_master_get(image, enabled, vme_base, size, aspace,
cycle, dwidth);
spin_unlock(&(image->lock));
return retval;
}
ssize_t ca91cx42_master_read(struct vme_master_resource *image, void *buf,
size_t count, loff_t offset)
{
int retval;
spin_lock(&(image->lock));
memcpy_fromio(buf, image->kern_base + offset, (unsigned int)count);
retval = count;
spin_unlock(&(image->lock));
return retval;
}
ssize_t ca91cx42_master_write(struct vme_master_resource *image, void *buf,
size_t count, loff_t offset)
{
int retval = 0;
spin_lock(&(image->lock));
memcpy_toio(image->kern_base + offset, buf, (unsigned int)count);
retval = count;
spin_unlock(&(image->lock));
return retval;
}
int ca91cx42_slot_get(void)
{
u32 slot = 0;
slot = ioread32(ca91cx42_bridge->base + VCSR_BS);
slot = ((slot & CA91CX42_VCSR_BS_SLOT_M) >> 27);
return (int)slot;
}
static int __init ca91cx42_init(void)
{
return pci_register_driver(&ca91cx42_driver);
}
/*
* Configure CR/CSR space
*
* Access to the CR/CSR can be configured at power-up. The location of the
* CR/CSR registers in the CR/CSR address space is determined by the boards
* Auto-ID or Geographic address. This function ensures that the window is
* enabled at an offset consistent with the boards geopgraphic address.
*/
static int ca91cx42_crcsr_init(struct pci_dev *pdev)
{
unsigned int crcsr_addr;
int tmp, slot;
// Write ctl reg without enable /* XXX We may need to set this somehow as the Universe II does not support
writel(temp_ctl, vmechip_baseaddr + inCTL[vmeIn->windowNbr]); * geographical addressing.
*/
#if 0
if (vme_slotnum != -1)
iowrite32(vme_slotnum << 27, ca91cx42_bridge->base + VCSR_BS);
#endif
slot = ca91cx42_slot_get();
dev_info(&pdev->dev, "CR/CSR Offset: %d\n", slot);
if (slot == 0) {
dev_err(&pdev->dev, "Slot number is unset, not configuring "
"CR/CSR space\n");
return -EINVAL;
}
if (vmeIn->windowEnable) /* Allocate mem for CR/CSR image */
temp_ctl |= 0x80000000; crcsr_kernel = pci_alloc_consistent(pdev, VME_CRCSR_BUF_SIZE,
&crcsr_bus);
if (crcsr_kernel == NULL) {
dev_err(&pdev->dev, "Failed to allocate memory for CR/CSR "
"image\n");
return -ENOMEM;
}
writel(temp_ctl, vmechip_baseaddr + inCTL[vmeIn->windowNbr]); memset(crcsr_kernel, 0, VME_CRCSR_BUF_SIZE);
return (0);
}
//----------------------------------------------------------------------------- crcsr_addr = slot * (512 * 1024);
// Function : uni_get_in_bound iowrite32(crcsr_bus - crcsr_addr, ca91cx42_bridge->base + VCSR_TO);
// Description:
//-----------------------------------------------------------------------------
int uni_get_in_bound(vmeInWindowCfg_t * vmeIn)
{
int temp_ctl = 0;
// Verify input data tmp = ioread32(ca91cx42_bridge->base + VCSR_CTL);
if (vmeIn->windowNbr > 7) { tmp |= CA91CX42_VCSR_CTL_EN;
return (-EINVAL); iowrite32(tmp, ca91cx42_bridge->base + VCSR_CTL);
}
// Get Window mappings.
vmeIn->vmeAddrL = readl(vmechip_baseaddr + inBS[vmeIn->windowNbr]);
vmeIn->pciAddrL = vmeIn->vmeAddrL +
readl(vmechip_baseaddr + inTO[vmeIn->windowNbr]);
vmeIn->windowSizeL = readl(vmechip_baseaddr + inBD[vmeIn->windowNbr]) -
vmeIn->vmeAddrL;
temp_ctl = readl(vmechip_baseaddr + inCTL[vmeIn->windowNbr]);
// Get Control & BUS attributes
if (temp_ctl & 0x40000000)
vmeIn->wrPostEnable = 1;
if (temp_ctl & 0x20000000)
vmeIn->prefetchEnable = 1;
if (temp_ctl & 0x00000040)
vmeIn->rmwLock = 1;
if (temp_ctl & 0x00000080)
vmeIn->data64BitCapable = 1;
if (temp_ctl & 0x00100000)
vmeIn->userAccessType |= VME_USER;
if (temp_ctl & 0x00200000)
vmeIn->userAccessType |= VME_SUPER;
if (temp_ctl & 0x00400000)
vmeIn->dataAccessType |= VME_DATA;
if (temp_ctl & 0x00800000)
vmeIn->dataAccessType |= VME_PROG;
if (temp_ctl & 0x80000000)
vmeIn->windowEnable = 1;
switch ((temp_ctl & 0x70000) >> 16) {
case 0x0:
vmeIn->addrSpace = VME_A16;
break;
case 0x1:
vmeIn->addrSpace = VME_A24;
break;
case 0x2:
vmeIn->addrSpace = VME_A32;
break;
case 0x6:
vmeIn->addrSpace = VME_USER1;
break;
case 0x7:
vmeIn->addrSpace = VME_USER2;
break;
}
return (0); return 0;
} }
//----------------------------------------------------------------------------- static void ca91cx42_crcsr_exit(struct pci_dev *pdev)
// Function : uni_set_out_bound
// Description:
//-----------------------------------------------------------------------------
int uni_set_out_bound(vmeOutWindowCfg_t * vmeOut)
{ {
int temp_ctl = 0; u32 tmp;
// Verify input data
if (vmeOut->windowNbr > 7) {
return (-EINVAL);
}
if ((vmeOut->xlatedAddrU) || (vmeOut->windowSizeU)
|| (vmeOut->pciBusAddrU)) {
return (-EINVAL);
}
if ((vmeOut->xlatedAddrL & 0xFFF) ||
(vmeOut->windowSizeL & 0xFFF) || (vmeOut->pciBusAddrL & 0xFFF)) {
return (-EINVAL);
}
if (vmeOut->bcastSelect2esst) {
return (-EINVAL);
}
switch (vmeOut->addrSpace) {
case VME_A64:
case VME_USER3:
case VME_USER4:
return (-EINVAL);
case VME_A16:
temp_ctl |= 0x00000;
break;
case VME_A24:
temp_ctl |= 0x10000;
break;
case VME_A32:
temp_ctl |= 0x20000;
break;
case VME_CRCSR:
temp_ctl |= 0x50000;
break;
case VME_USER1:
temp_ctl |= 0x60000;
break;
case VME_USER2:
temp_ctl |= 0x70000;
break;
}
// Disable while we are mucking around
writel(0x00000000, vmechip_baseaddr + outCTL[vmeOut->windowNbr]);
writel(vmeOut->pciBusAddrL,
vmechip_baseaddr + outBS[vmeOut->windowNbr]);
writel(vmeOut->pciBusAddrL + vmeOut->windowSizeL,
vmechip_baseaddr + outBD[vmeOut->windowNbr]);
writel(vmeOut->xlatedAddrL - vmeOut->pciBusAddrL,
vmechip_baseaddr + outTO[vmeOut->windowNbr]);
// Sanity check.
if (vmeOut->pciBusAddrL !=
readl(vmechip_baseaddr + outBS[vmeOut->windowNbr])) {
printk(KERN_ERR
"ca91c042: out window: %x, failed to configure\n",
vmeOut->windowNbr);
return (-EINVAL);
}
if (vmeOut->pciBusAddrL + vmeOut->windowSizeL !=
readl(vmechip_baseaddr + outBD[vmeOut->windowNbr])) {
printk(KERN_ERR
"ca91c042: out window: %x, failed to configure\n",
vmeOut->windowNbr);
return (-EINVAL);
}
if (vmeOut->xlatedAddrL - vmeOut->pciBusAddrL !=
readl(vmechip_baseaddr + outTO[vmeOut->windowNbr])) {
printk(KERN_ERR
"ca91c042: out window: %x, failed to configure\n",
vmeOut->windowNbr);
return (-EINVAL);
}
// Setup CTL register.
if (vmeOut->wrPostEnable)
temp_ctl |= 0x40000000;
if (vmeOut->userAccessType & VME_SUPER)
temp_ctl |= 0x001000;
if (vmeOut->dataAccessType & VME_PROG)
temp_ctl |= 0x004000;
if (vmeOut->maxDataWidth == VME_D16)
temp_ctl |= 0x00400000;
if (vmeOut->maxDataWidth == VME_D32)
temp_ctl |= 0x00800000;
if (vmeOut->maxDataWidth == VME_D64)
temp_ctl |= 0x00C00000;
if (vmeOut->xferProtocol & (VME_BLT | VME_MBLT))
temp_ctl |= 0x00000100;
// Write ctl reg without enable /* Turn off CR/CSR space */
writel(temp_ctl, vmechip_baseaddr + outCTL[vmeOut->windowNbr]); tmp = ioread32(ca91cx42_bridge->base + VCSR_CTL);
tmp &= ~CA91CX42_VCSR_CTL_EN;
iowrite32(tmp, ca91cx42_bridge->base + VCSR_CTL);
if (vmeOut->windowEnable) /* Free image */
temp_ctl |= 0x80000000; iowrite32(0, ca91cx42_bridge->base + VCSR_TO);
writel(temp_ctl, vmechip_baseaddr + outCTL[vmeOut->windowNbr]); pci_free_consistent(pdev, VME_CRCSR_BUF_SIZE, crcsr_kernel, crcsr_bus);
return (0);
} }
//----------------------------------------------------------------------------- static int ca91cx42_probe(struct pci_dev *pdev, const struct pci_device_id *id)
// Function : uni_get_out_bound
// Description:
//-----------------------------------------------------------------------------
int uni_get_out_bound(vmeOutWindowCfg_t * vmeOut)
{ {
int temp_ctl = 0; int retval, i;
u32 data;
struct list_head *pos = NULL;
struct vme_master_resource *master_image;
struct vme_slave_resource *slave_image;
#if 0
struct vme_dma_resource *dma_ctrlr;
#endif
struct vme_lm_resource *lm;
// Verify input data /* We want to support more than one of each bridge so we need to
if (vmeOut->windowNbr > 7) { * dynamically allocate the bridge structure
return (-EINVAL); */
ca91cx42_bridge = kmalloc(sizeof(struct vme_bridge), GFP_KERNEL);
if (ca91cx42_bridge == NULL) {
dev_err(&pdev->dev, "Failed to allocate memory for device "
"structure\n");
retval = -ENOMEM;
goto err_struct;
}
memset(ca91cx42_bridge, 0, sizeof(struct vme_bridge));
/* Enable the device */
retval = pci_enable_device(pdev);
if (retval) {
dev_err(&pdev->dev, "Unable to enable device\n");
goto err_enable;
}
/* Map Registers */
retval = pci_request_regions(pdev, driver_name);
if (retval) {
dev_err(&pdev->dev, "Unable to reserve resources\n");
goto err_resource;
}
/* map registers in BAR 0 */
ca91cx42_bridge->base = ioremap_nocache(pci_resource_start(pdev, 0),
4096);
if (!ca91cx42_bridge->base) {
dev_err(&pdev->dev, "Unable to remap CRG region\n");
retval = -EIO;
goto err_remap;
}
/* Check to see if the mapping worked out */
data = ioread32(ca91cx42_bridge->base + CA91CX42_PCI_ID) & 0x0000FFFF;
if (data != PCI_VENDOR_ID_TUNDRA) {
dev_err(&pdev->dev, "PCI_ID check failed\n");
retval = -EIO;
goto err_test;
}
/* Initialize wait queues & mutual exclusion flags */
/* XXX These need to be moved to the vme_bridge structure */
init_waitqueue_head(&dma_queue);
init_waitqueue_head(&iack_queue);
mutex_init(&(vme_int));
mutex_init(&(vme_irq));
mutex_init(&(vme_rmw));
ca91cx42_bridge->parent = &(pdev->dev);
strcpy(ca91cx42_bridge->name, driver_name);
/* Setup IRQ */
retval = ca91cx42_irq_init(ca91cx42_bridge);
if (retval != 0) {
dev_err(&pdev->dev, "Chip Initialization failed.\n");
goto err_irq;
}
/* Add master windows to list */
INIT_LIST_HEAD(&(ca91cx42_bridge->master_resources));
for (i = 0; i < CA91C142_MAX_MASTER; i++) {
master_image = kmalloc(sizeof(struct vme_master_resource),
GFP_KERNEL);
if (master_image == NULL) {
dev_err(&pdev->dev, "Failed to allocate memory for "
"master resource structure\n");
retval = -ENOMEM;
goto err_master;
}
master_image->parent = ca91cx42_bridge;
spin_lock_init(&(master_image->lock));
master_image->locked = 0;
master_image->number = i;
master_image->address_attr = VME_A16 | VME_A24 | VME_A32 |
VME_CRCSR | VME_USER1 | VME_USER2;
master_image->cycle_attr = VME_SCT | VME_BLT | VME_MBLT |
VME_SUPER | VME_USER | VME_PROG | VME_DATA;
master_image->width_attr = VME_D8 | VME_D16 | VME_D32 | VME_D64;
memset(&(master_image->pci_resource), 0,
sizeof(struct resource));
master_image->kern_base = NULL;
list_add_tail(&(master_image->list),
&(ca91cx42_bridge->master_resources));
}
/* Add slave windows to list */
INIT_LIST_HEAD(&(ca91cx42_bridge->slave_resources));
for (i = 0; i < CA91C142_MAX_SLAVE; i++) {
slave_image = kmalloc(sizeof(struct vme_slave_resource),
GFP_KERNEL);
if (slave_image == NULL) {
dev_err(&pdev->dev, "Failed to allocate memory for "
"slave resource structure\n");
retval = -ENOMEM;
goto err_slave;
}
slave_image->parent = ca91cx42_bridge;
mutex_init(&(slave_image->mtx));
slave_image->locked = 0;
slave_image->number = i;
slave_image->address_attr = VME_A24 | VME_A32 | VME_USER1 |
VME_USER2;
/* Only windows 0 and 4 support A16 */
if (i == 0 || i == 4)
slave_image->address_attr |= VME_A16;
slave_image->cycle_attr = VME_SCT | VME_BLT | VME_MBLT |
VME_SUPER | VME_USER | VME_PROG | VME_DATA;
list_add_tail(&(slave_image->list),
&(ca91cx42_bridge->slave_resources));
}
#if 0
/* Add dma engines to list */
INIT_LIST_HEAD(&(ca91cx42_bridge->dma_resources));
for (i = 0; i < CA91C142_MAX_DMA; i++) {
dma_ctrlr = kmalloc(sizeof(struct vme_dma_resource),
GFP_KERNEL);
if (dma_ctrlr == NULL) {
dev_err(&pdev->dev, "Failed to allocate memory for "
"dma resource structure\n");
retval = -ENOMEM;
goto err_dma;
}
dma_ctrlr->parent = ca91cx42_bridge;
mutex_init(&(dma_ctrlr->mtx));
dma_ctrlr->locked = 0;
dma_ctrlr->number = i;
INIT_LIST_HEAD(&(dma_ctrlr->pending));
INIT_LIST_HEAD(&(dma_ctrlr->running));
list_add_tail(&(dma_ctrlr->list),
&(ca91cx42_bridge->dma_resources));
} }
// Get Window mappings. #endif
vmeOut->pciBusAddrL = /* Add location monitor to list */
readl(vmechip_baseaddr + outBS[vmeOut->windowNbr]); INIT_LIST_HEAD(&(ca91cx42_bridge->lm_resources));
vmeOut->xlatedAddrL = lm = kmalloc(sizeof(struct vme_lm_resource), GFP_KERNEL);
vmeOut->pciBusAddrL + readl(vmechip_baseaddr + if (lm == NULL) {
outTO[vmeOut->windowNbr]); dev_err(&pdev->dev, "Failed to allocate memory for "
vmeOut->windowSizeL = "location monitor resource structure\n");
readl(vmechip_baseaddr + outBD[vmeOut->windowNbr]) - retval = -ENOMEM;
vmeOut->pciBusAddrL; goto err_lm;
}
lm->parent = ca91cx42_bridge;
mutex_init(&(lm->mtx));
lm->locked = 0;
lm->number = 1;
lm->monitors = 4;
list_add_tail(&(lm->list), &(ca91cx42_bridge->lm_resources));
ca91cx42_bridge->slave_get = ca91cx42_slave_get;
ca91cx42_bridge->slave_set = ca91cx42_slave_set;
ca91cx42_bridge->master_get = ca91cx42_master_get;
ca91cx42_bridge->master_set = ca91cx42_master_set;
ca91cx42_bridge->master_read = ca91cx42_master_read;
ca91cx42_bridge->master_write = ca91cx42_master_write;
#if 0
ca91cx42_bridge->master_rmw = ca91cx42_master_rmw;
ca91cx42_bridge->dma_list_add = ca91cx42_dma_list_add;
ca91cx42_bridge->dma_list_exec = ca91cx42_dma_list_exec;
ca91cx42_bridge->dma_list_empty = ca91cx42_dma_list_empty;
#endif
ca91cx42_bridge->request_irq = ca91cx42_request_irq;
ca91cx42_bridge->free_irq = ca91cx42_free_irq;
ca91cx42_bridge->generate_irq = ca91cx42_generate_irq;
#if 0
ca91cx42_bridge->lm_set = ca91cx42_lm_set;
ca91cx42_bridge->lm_get = ca91cx42_lm_get;
ca91cx42_bridge->lm_attach = ca91cx42_lm_attach;
ca91cx42_bridge->lm_detach = ca91cx42_lm_detach;
#endif
ca91cx42_bridge->slot_get = ca91cx42_slot_get;
temp_ctl = readl(vmechip_baseaddr + outCTL[vmeOut->windowNbr]); data = ioread32(ca91cx42_bridge->base + MISC_CTL);
dev_info(&pdev->dev, "Board is%s the VME system controller\n",
(data & CA91CX42_MISC_CTL_SYSCON) ? "" : " not");
dev_info(&pdev->dev, "Slot ID is %d\n", ca91cx42_slot_get());
// Get Control & BUS attributes if (ca91cx42_crcsr_init(pdev)) {
if (temp_ctl & 0x40000000) dev_err(&pdev->dev, "CR/CSR configuration failed.\n");
vmeOut->wrPostEnable = 1; retval = -EINVAL;
if (temp_ctl & 0x001000) #if 0
vmeOut->userAccessType = VME_SUPER; goto err_crcsr;
else #endif
vmeOut->userAccessType = VME_USER; }
if (temp_ctl & 0x004000)
vmeOut->dataAccessType = VME_PROG;
else
vmeOut->dataAccessType = VME_DATA;
if (temp_ctl & 0x80000000)
vmeOut->windowEnable = 1;
switch ((temp_ctl & 0x00C00000) >> 22) { /* Need to save ca91cx42_bridge pointer locally in link list for use in
case 0: * ca91cx42_remove()
vmeOut->maxDataWidth = VME_D8; */
break; retval = vme_register_bridge(ca91cx42_bridge);
case 1: if (retval != 0) {
vmeOut->maxDataWidth = VME_D16; dev_err(&pdev->dev, "Chip Registration failed.\n");
break; goto err_reg;
case 2:
vmeOut->maxDataWidth = VME_D32;
break;
case 3:
vmeOut->maxDataWidth = VME_D64;
break;
} }
if (temp_ctl & 0x00000100)
vmeOut->xferProtocol = VME_BLT;
else
vmeOut->xferProtocol = VME_SCT;
switch ((temp_ctl & 0x70000) >> 16) { return 0;
case 0x0:
vmeOut->addrSpace = VME_A16; vme_unregister_bridge(ca91cx42_bridge);
break; err_reg:
case 0x1: ca91cx42_crcsr_exit(pdev);
vmeOut->addrSpace = VME_A24; err_crcsr:
break; err_lm:
case 0x2: /* resources are stored in link list */
vmeOut->addrSpace = VME_A32; list_for_each(pos, &(ca91cx42_bridge->lm_resources)) {
break; lm = list_entry(pos, struct vme_lm_resource, list);
case 0x5: list_del(pos);
vmeOut->addrSpace = VME_CRCSR; kfree(lm);
break; }
case 0x6: #if 0
vmeOut->addrSpace = VME_USER1; err_dma:
break; /* resources are stored in link list */
case 0x7: list_for_each(pos, &(ca91cx42_bridge->dma_resources)) {
vmeOut->addrSpace = VME_USER2; dma_ctrlr = list_entry(pos, struct vme_dma_resource, list);
break; list_del(pos);
kfree(dma_ctrlr);
} }
#endif
err_slave:
/* resources are stored in link list */
list_for_each(pos, &(ca91cx42_bridge->slave_resources)) {
slave_image = list_entry(pos, struct vme_slave_resource, list);
list_del(pos);
kfree(slave_image);
}
err_master:
/* resources are stored in link list */
list_for_each(pos, &(ca91cx42_bridge->master_resources)) {
master_image = list_entry(pos, struct vme_master_resource,
list);
list_del(pos);
kfree(master_image);
}
ca91cx42_irq_exit(pdev);
err_irq:
err_test:
iounmap(ca91cx42_bridge->base);
err_remap:
pci_release_regions(pdev);
err_resource:
pci_disable_device(pdev);
err_enable:
kfree(ca91cx42_bridge);
err_struct:
return retval;
return (0);
} }
//----------------------------------------------------------------------------- void ca91cx42_remove(struct pci_dev *pdev)
// Function : uni_setup_lm
// Description:
//-----------------------------------------------------------------------------
int uni_setup_lm(vmeLmCfg_t * vmeLm)
{ {
int temp_ctl = 0; struct list_head *pos = NULL;
struct vme_master_resource *master_image;
struct vme_slave_resource *slave_image;
struct vme_dma_resource *dma_ctrlr;
struct vme_lm_resource *lm;
int i;
if (vmeLm->addrU) { /* Turn off Ints */
return (-EINVAL); iowrite32(0, ca91cx42_bridge->base + LINT_EN);
/* Turn off the windows */
iowrite32(0x00800000, ca91cx42_bridge->base + LSI0_CTL);
iowrite32(0x00800000, ca91cx42_bridge->base + LSI1_CTL);
iowrite32(0x00800000, ca91cx42_bridge->base + LSI2_CTL);
iowrite32(0x00800000, ca91cx42_bridge->base + LSI3_CTL);
iowrite32(0x00800000, ca91cx42_bridge->base + LSI4_CTL);
iowrite32(0x00800000, ca91cx42_bridge->base + LSI5_CTL);
iowrite32(0x00800000, ca91cx42_bridge->base + LSI6_CTL);
iowrite32(0x00800000, ca91cx42_bridge->base + LSI7_CTL);
iowrite32(0x00F00000, ca91cx42_bridge->base + VSI0_CTL);
iowrite32(0x00F00000, ca91cx42_bridge->base + VSI1_CTL);
iowrite32(0x00F00000, ca91cx42_bridge->base + VSI2_CTL);
iowrite32(0x00F00000, ca91cx42_bridge->base + VSI3_CTL);
iowrite32(0x00F00000, ca91cx42_bridge->base + VSI4_CTL);
iowrite32(0x00F00000, ca91cx42_bridge->base + VSI5_CTL);
iowrite32(0x00F00000, ca91cx42_bridge->base + VSI6_CTL);
iowrite32(0x00F00000, ca91cx42_bridge->base + VSI7_CTL);
vme_unregister_bridge(ca91cx42_bridge);
#if 0
ca91cx42_crcsr_exit(pdev);
#endif
/* resources are stored in link list */
list_for_each(pos, &(ca91cx42_bridge->lm_resources)) {
lm = list_entry(pos, struct vme_lm_resource, list);
list_del(pos);
kfree(lm);
} }
switch (vmeLm->addrSpace) {
case VME_A64: /* resources are stored in link list */
case VME_USER3: list_for_each(pos, &(ca91cx42_bridge->dma_resources)) {
case VME_USER4: dma_ctrlr = list_entry(pos, struct vme_dma_resource, list);
return (-EINVAL); list_del(pos);
case VME_A16: kfree(dma_ctrlr);
temp_ctl |= 0x00000;
break;
case VME_A24:
temp_ctl |= 0x10000;
break;
case VME_A32:
temp_ctl |= 0x20000;
break;
case VME_CRCSR:
temp_ctl |= 0x50000;
break;
case VME_USER1:
temp_ctl |= 0x60000;
break;
case VME_USER2:
temp_ctl |= 0x70000;
break;
} }
// Disable while we are mucking around /* resources are stored in link list */
writel(0x00000000, vmechip_baseaddr + LM_CTL); list_for_each(pos, &(ca91cx42_bridge->slave_resources)) {
slave_image = list_entry(pos, struct vme_slave_resource, list);
list_del(pos);
kfree(slave_image);
}
writel(vmeLm->addr, vmechip_baseaddr + LM_BS); /* resources are stored in link list */
list_for_each(pos, &(ca91cx42_bridge->master_resources)) {
master_image = list_entry(pos, struct vme_master_resource,
list);
list_del(pos);
kfree(master_image);
}
// Setup CTL register. ca91cx42_irq_exit(pdev);
if (vmeLm->userAccessType & VME_SUPER)
temp_ctl |= 0x00200000; iounmap(ca91cx42_bridge->base);
if (vmeLm->userAccessType & VME_USER)
temp_ctl |= 0x00100000;
if (vmeLm->dataAccessType & VME_PROG)
temp_ctl |= 0x00800000;
if (vmeLm->dataAccessType & VME_DATA)
temp_ctl |= 0x00400000;
uni_lm_event = 0; pci_release_regions(pdev);
// Write ctl reg and enable pci_disable_device(pdev);
writel(0x80000000 | temp_ctl, vmechip_baseaddr + LM_CTL);
temp_ctl = readl(vmechip_baseaddr + LM_CTL);
return (0); kfree(ca91cx42_bridge);
} }
//----------------------------------------------------------------------------- static void __exit ca91cx42_exit(void)
// Function : uni_wait_lm
// Description:
//-----------------------------------------------------------------------------
int uni_wait_lm(vmeLmCfg_t * vmeLm)
{ {
unsigned long flags; pci_unregister_driver(&ca91cx42_driver);
unsigned int tmp; }
spin_lock_irqsave(&lm_lock, flags); MODULE_DESCRIPTION("VME driver for the Tundra Universe II VME bridge");
tmp = uni_lm_event; MODULE_LICENSE("GPL");
spin_unlock_irqrestore(&lm_lock, flags);
if (tmp == 0) {
if (vmeLm->lmWait < 10)
vmeLm->lmWait = 10;
interruptible_sleep_on_timeout(&lm_queue, vmeLm->lmWait);
}
writel(0x00000000, vmechip_baseaddr + LM_CTL);
vmeLm->lmEvents = uni_lm_event;
return (0); module_init(ca91cx42_init);
} module_exit(ca91cx42_exit);
/*----------------------------------------------------------------------------
* STAGING
*--------------------------------------------------------------------------*/
#if 0
#define SWIZZLE(X) ( ((X & 0xFF000000) >> 24) | ((X & 0x00FF0000) >> 8) | ((X & 0x0000FF00) << 8) | ((X & 0x000000FF) << 24)) #define SWIZZLE(X) ( ((X & 0xFF000000) >> 24) | ((X & 0x00FF0000) >> 8) | ((X & 0x0000FF00) << 8) | ((X & 0x000000FF) << 24))
//----------------------------------------------------------------------------- int ca91cx42_master_rmw(vmeRmwCfg_t *vmeRmw)
// Function : uni_do_rmw
// Description:
//-----------------------------------------------------------------------------
int uni_do_rmw(vmeRmwCfg_t * vmeRmw)
{ {
int temp_ctl = 0; int temp_ctl = 0;
int tempBS = 0; int tempBS = 0;
...@@ -1252,26 +1372,27 @@ int uni_do_rmw(vmeRmwCfg_t * vmeRmw) ...@@ -1252,26 +1372,27 @@ int uni_do_rmw(vmeRmwCfg_t * vmeRmw)
int i; int i;
vmeOutWindowCfg_t vmeOut; vmeOutWindowCfg_t vmeOut;
if (vmeRmw->maxAttempts < 1) { if (vmeRmw->maxAttempts < 1) {
return (-EINVAL); return -EINVAL;
} }
if (vmeRmw->targetAddrU) { if (vmeRmw->targetAddrU) {
return (-EINVAL); return -EINVAL;
} }
// Find the PCI address that maps to the desired VME address /* Find the PCI address that maps to the desired VME address */
for (i = 0; i < 8; i++) { for (i = 0; i < 8; i++) {
temp_ctl = readl(vmechip_baseaddr + outCTL[i]); temp_ctl = ioread32(ca91cx42_bridge->base +
CA91CX42_LSI_CTL[i]);
if ((temp_ctl & 0x80000000) == 0) { if ((temp_ctl & 0x80000000) == 0) {
continue; continue;
} }
memset(&vmeOut, 0, sizeof(vmeOut)); memset(&vmeOut, 0, sizeof(vmeOut));
vmeOut.windowNbr = i; vmeOut.windowNbr = i;
uni_get_out_bound(&vmeOut); ca91cx42_get_out_bound(&vmeOut);
if (vmeOut.addrSpace != vmeRmw->addrSpace) { if (vmeOut.addrSpace != vmeRmw->addrSpace) {
continue; continue;
} }
tempBS = readl(vmechip_baseaddr + outBS[i]); tempBS = ioread32(ca91cx42_bridge->base + CA91CX42_LSI_BS[i]);
tempBD = readl(vmechip_baseaddr + outBD[i]); tempBD = ioread32(ca91cx42_bridge->base + CA91CX42_LSI_BD[i]);
tempTO = readl(vmechip_baseaddr + outTO[i]); tempTO = ioread32(ca91cx42_bridge->base + CA91CX42_LSI_TO[i]);
vmeBS = tempBS + tempTO; vmeBS = tempBS + tempTO;
vmeBD = tempBD + tempTO; vmeBD = tempBD + tempTO;
if ((vmeRmw->targetAddr >= vmeBS) && if ((vmeRmw->targetAddr >= vmeBS) &&
...@@ -1285,44 +1406,41 @@ int uni_do_rmw(vmeRmwCfg_t * vmeRmw) ...@@ -1285,44 +1406,41 @@ int uni_do_rmw(vmeRmwCfg_t * vmeRmw)
} }
} }
// If no window - fail. /* If no window - fail. */
if (rmw_pci_data_ptr == NULL) { if (rmw_pci_data_ptr == NULL) {
return (-EINVAL); return -EINVAL;
} }
// Setup the RMW registers. /* Setup the RMW registers. */
writel(0, vmechip_baseaddr + SCYC_CTL); iowrite32(0, ca91cx42_bridge->base + SCYC_CTL);
writel(SWIZZLE(vmeRmw->enableMask), vmechip_baseaddr + SCYC_EN); iowrite32(SWIZZLE(vmeRmw->enableMask), ca91cx42_bridge->base + SCYC_EN);
writel(SWIZZLE(vmeRmw->compareData), vmechip_baseaddr + SCYC_CMP); iowrite32(SWIZZLE(vmeRmw->compareData), ca91cx42_bridge->base +
writel(SWIZZLE(vmeRmw->swapData), vmechip_baseaddr + SCYC_SWP); SCYC_CMP);
writel((int)rmw_pci_data_ptr, vmechip_baseaddr + SCYC_ADDR); iowrite32(SWIZZLE(vmeRmw->swapData), ca91cx42_bridge->base + SCYC_SWP);
writel(1, vmechip_baseaddr + SCYC_CTL); iowrite32((int)rmw_pci_data_ptr, ca91cx42_bridge->base + SCYC_ADDR);
iowrite32(1, ca91cx42_bridge->base + SCYC_CTL);
// Run the RMW cycle until either success or max attempts.
/* Run the RMW cycle until either success or max attempts. */
vmeRmw->numAttempts = 1; vmeRmw->numAttempts = 1;
while (vmeRmw->numAttempts <= vmeRmw->maxAttempts) { while (vmeRmw->numAttempts <= vmeRmw->maxAttempts) {
if ((readl(vaDataPtr) & vmeRmw->enableMask) == if ((ioread32(vaDataPtr) & vmeRmw->enableMask) ==
(vmeRmw->swapData & vmeRmw->enableMask)) { (vmeRmw->swapData & vmeRmw->enableMask)) {
writel(0, vmechip_baseaddr + SCYC_CTL); iowrite32(0, ca91cx42_bridge->base + SCYC_CTL);
break; break;
} }
vmeRmw->numAttempts++; vmeRmw->numAttempts++;
} }
// If no success, set num Attempts to be greater than max attempts /* If no success, set num Attempts to be greater than max attempts */
if (vmeRmw->numAttempts > vmeRmw->maxAttempts) { if (vmeRmw->numAttempts > vmeRmw->maxAttempts) {
vmeRmw->numAttempts = vmeRmw->maxAttempts + 1; vmeRmw->numAttempts = vmeRmw->maxAttempts + 1;
} }
return (0); return 0;
} }
//-----------------------------------------------------------------------------
// Function : uniSetupDctlReg
// Description:
//-----------------------------------------------------------------------------
int uniSetupDctlReg(vmeDmaPacket_t * vmeDma, int *dctlregreturn) int uniSetupDctlReg(vmeDmaPacket_t * vmeDma, int *dctlregreturn)
{ {
unsigned int dctlreg = 0x80; unsigned int dctlreg = 0x80;
...@@ -1366,11 +1484,11 @@ int uniSetupDctlReg(vmeDmaPacket_t * vmeDma, int *dctlregreturn) ...@@ -1366,11 +1484,11 @@ int uniSetupDctlReg(vmeDmaPacket_t * vmeDma, int *dctlregreturn)
dctlreg |= 0x00070000; dctlreg |= 0x00070000;
break; break;
case VME_A64: // not supported in Universe DMA case VME_A64: /* not supported in Universe DMA */
case VME_CRCSR: case VME_CRCSR:
case VME_USER3: case VME_USER3:
case VME_USER4: case VME_USER4:
return (-EINVAL); return -EINVAL;
break; break;
} }
if (vmeAttr->userAccessType == VME_PROG) { if (vmeAttr->userAccessType == VME_PROG) {
...@@ -1383,50 +1501,46 @@ int uniSetupDctlReg(vmeDmaPacket_t * vmeDma, int *dctlregreturn) ...@@ -1383,50 +1501,46 @@ int uniSetupDctlReg(vmeDmaPacket_t * vmeDma, int *dctlregreturn)
dctlreg |= 0x00000100; dctlreg |= 0x00000100;
} }
*dctlregreturn = dctlreg; *dctlregreturn = dctlreg;
return (0); return 0;
} }
//-----------------------------------------------------------------------------
// Function : uni_start_dma
// Description:
//-----------------------------------------------------------------------------
unsigned int unsigned int
uni_start_dma(int channel, unsigned int dgcsreg, TDMA_Cmd_Packet * vmeLL) ca91cx42_start_dma(int channel, unsigned int dgcsreg, TDMA_Cmd_Packet *vmeLL)
{ {
unsigned int val; unsigned int val;
// Setup registers as needed for direct or chained. /* Setup registers as needed for direct or chained. */
if (dgcsreg & 0x8000000) { if (dgcsreg & 0x8000000) {
writel(0, vmechip_baseaddr + DTBC); iowrite32(0, ca91cx42_bridge->base + DTBC);
writel((unsigned int)vmeLL, vmechip_baseaddr + DCPP); iowrite32((unsigned int)vmeLL, ca91cx42_bridge->base + DCPP);
} else { } else {
#if 0 #if 0
printk("Starting: DGCS = %08x\n", dgcsreg); printk(KERN_ERR "Starting: DGCS = %08x\n", dgcsreg);
printk("Starting: DVA = %08x\n", readl(&vmeLL->dva)); printk(KERN_ERR "Starting: DVA = %08x\n",
printk("Starting: DLV = %08x\n", readl(&vmeLL->dlv)); ioread32(&vmeLL->dva));
printk("Starting: DTBC = %08x\n", readl(&vmeLL->dtbc)); printk(KERN_ERR "Starting: DLV = %08x\n",
printk("Starting: DCTL = %08x\n", readl(&vmeLL->dctl)); ioread32(&vmeLL->dlv));
printk(KERN_ERR "Starting: DTBC = %08x\n",
ioread32(&vmeLL->dtbc));
printk(KERN_ERR "Starting: DCTL = %08x\n",
ioread32(&vmeLL->dctl));
#endif #endif
// Write registers /* Write registers */
writel(readl(&vmeLL->dva), vmechip_baseaddr + DVA); iowrite32(ioread32(&vmeLL->dva), ca91cx42_bridge->base + DVA);
writel(readl(&vmeLL->dlv), vmechip_baseaddr + DLA); iowrite32(ioread32(&vmeLL->dlv), ca91cx42_bridge->base + DLA);
writel(readl(&vmeLL->dtbc), vmechip_baseaddr + DTBC); iowrite32(ioread32(&vmeLL->dtbc), ca91cx42_bridge->base + DTBC);
writel(readl(&vmeLL->dctl), vmechip_baseaddr + DCTL); iowrite32(ioread32(&vmeLL->dctl), ca91cx42_bridge->base + DCTL);
writel(0, vmechip_baseaddr + DCPP); iowrite32(0, ca91cx42_bridge->base + DCPP);
} }
// Start the operation /* Start the operation */
writel(dgcsreg, vmechip_baseaddr + DGCS); iowrite32(dgcsreg, ca91cx42_bridge->base + DGCS);
val = get_tbl(); val = get_tbl();
writel(dgcsreg | 0x8000000F, vmechip_baseaddr + DGCS); iowrite32(dgcsreg | 0x8000000F, ca91cx42_bridge->base + DGCS);
return (val); return val;
} }
//----------------------------------------------------------------------------- TDMA_Cmd_Packet *ca91cx42_setup_dma(vmeDmaPacket_t * vmeDma)
// Function : uni_setup_dma
// Description:
//-----------------------------------------------------------------------------
TDMA_Cmd_Packet *uni_setup_dma(vmeDmaPacket_t * vmeDma)
{ {
vmeDmaPacket_t *vmeCur; vmeDmaPacket_t *vmeCur;
int maxPerPage; int maxPerPage;
...@@ -1439,9 +1553,9 @@ TDMA_Cmd_Packet *uni_setup_dma(vmeDmaPacket_t * vmeDma) ...@@ -1439,9 +1553,9 @@ TDMA_Cmd_Packet *uni_setup_dma(vmeDmaPacket_t * vmeDma)
maxPerPage = PAGESIZE / sizeof(TDMA_Cmd_Packet) - 1; maxPerPage = PAGESIZE / sizeof(TDMA_Cmd_Packet) - 1;
startLL = (TDMA_Cmd_Packet *) __get_free_pages(GFP_KERNEL, 0); startLL = (TDMA_Cmd_Packet *) __get_free_pages(GFP_KERNEL, 0);
if (startLL == 0) { if (startLL == 0) {
return (startLL); return startLL;
} }
// First allocate pages for descriptors and create linked list /* First allocate pages for descriptors and create linked list */
vmeCur = vmeDma; vmeCur = vmeDma;
currentLL = startLL; currentLL = startLL;
currentLLcount = 0; currentLLcount = 0;
...@@ -1461,57 +1575,53 @@ TDMA_Cmd_Packet *uni_setup_dma(vmeDmaPacket_t * vmeDma) ...@@ -1461,57 +1575,53 @@ TDMA_Cmd_Packet *uni_setup_dma(vmeDmaPacket_t * vmeDma)
vmeCur = vmeCur->pNextPacket; vmeCur = vmeCur->pNextPacket;
} }
// Next fill in information for each descriptor /* Next fill in information for each descriptor */
vmeCur = vmeDma; vmeCur = vmeDma;
currentLL = startLL; currentLL = startLL;
while (vmeCur != 0) { while (vmeCur != 0) {
if (vmeCur->srcBus == VME_DMA_VME) { if (vmeCur->srcBus == VME_DMA_VME) {
writel(vmeCur->srcAddr, &currentLL->dva); iowrite32(vmeCur->srcAddr, &currentLL->dva);
writel(vmeCur->dstAddr, &currentLL->dlv); iowrite32(vmeCur->dstAddr, &currentLL->dlv);
} else { } else {
writel(vmeCur->srcAddr, &currentLL->dlv); iowrite32(vmeCur->srcAddr, &currentLL->dlv);
writel(vmeCur->dstAddr, &currentLL->dva); iowrite32(vmeCur->dstAddr, &currentLL->dva);
} }
uniSetupDctlReg(vmeCur, &dctlreg); uniSetupDctlReg(vmeCur, &dctlreg);
writel(dctlreg, &currentLL->dctl); iowrite32(dctlreg, &currentLL->dctl);
writel(vmeCur->byteCount, &currentLL->dtbc); iowrite32(vmeCur->byteCount, &currentLL->dtbc);
currentLL = (TDMA_Cmd_Packet *) currentLL->dcpp; currentLL = (TDMA_Cmd_Packet *) currentLL->dcpp;
vmeCur = vmeCur->pNextPacket; vmeCur = vmeCur->pNextPacket;
} }
// Convert Links to PCI addresses. /* Convert Links to PCI addresses. */
currentLL = startLL; currentLL = startLL;
while (currentLL != 0) { while (currentLL != 0) {
nextLL = (TDMA_Cmd_Packet *) currentLL->dcpp; nextLL = (TDMA_Cmd_Packet *) currentLL->dcpp;
if (nextLL == 0) { if (nextLL == 0) {
writel(1, &currentLL->dcpp); iowrite32(1, &currentLL->dcpp);
} else { } else {
writel((unsigned int)virt_to_bus(nextLL), iowrite32((unsigned int)virt_to_bus(nextLL),
&currentLL->dcpp); &currentLL->dcpp);
} }
currentLL = nextLL; currentLL = nextLL;
} }
// Return pointer to descriptors list /* Return pointer to descriptors list */
return (startLL); return startLL;
} }
//----------------------------------------------------------------------------- int ca91cx42_free_dma(TDMA_Cmd_Packet *startLL)
// Function : uni_free_dma
// Description:
//-----------------------------------------------------------------------------
int uni_free_dma(TDMA_Cmd_Packet * startLL)
{ {
TDMA_Cmd_Packet *currentLL; TDMA_Cmd_Packet *currentLL;
TDMA_Cmd_Packet *prevLL; TDMA_Cmd_Packet *prevLL;
TDMA_Cmd_Packet *nextLL; TDMA_Cmd_Packet *nextLL;
unsigned int dcppreg; unsigned int dcppreg;
// Convert Links to virtual addresses. /* Convert Links to virtual addresses. */
currentLL = startLL; currentLL = startLL;
while (currentLL != 0) { while (currentLL != 0) {
dcppreg = readl(&currentLL->dcpp); dcppreg = ioread32(&currentLL->dcpp);
dcppreg &= ~6; dcppreg &= ~6;
if (dcppreg & 1) { if (dcppreg & 1) {
currentLL->dcpp = 0; currentLL->dcpp = 0;
...@@ -1521,7 +1631,7 @@ int uni_free_dma(TDMA_Cmd_Packet * startLL) ...@@ -1521,7 +1631,7 @@ int uni_free_dma(TDMA_Cmd_Packet * startLL)
currentLL = (TDMA_Cmd_Packet *) currentLL->dcpp; currentLL = (TDMA_Cmd_Packet *) currentLL->dcpp;
} }
// Free all pages associated with the descriptors. /* Free all pages associated with the descriptors. */
currentLL = startLL; currentLL = startLL;
prevLL = currentLL; prevLL = currentLL;
while (currentLL != 0) { while (currentLL != 0) {
...@@ -1533,15 +1643,11 @@ int uni_free_dma(TDMA_Cmd_Packet * startLL) ...@@ -1533,15 +1643,11 @@ int uni_free_dma(TDMA_Cmd_Packet * startLL)
currentLL = nextLL; currentLL = nextLL;
} }
// Return pointer to descriptors list /* Return pointer to descriptors list */
return (0); return 0;
} }
//----------------------------------------------------------------------------- int ca91cx42_do_dma(vmeDmaPacket_t *vmeDma)
// Function : uni_do_dma
// Description:
//-----------------------------------------------------------------------------
int uni_do_dma(vmeDmaPacket_t * vmeDma)
{ {
unsigned int dgcsreg = 0; unsigned int dgcsreg = 0;
unsigned int dctlreg = 0; unsigned int dctlreg = 0;
...@@ -1550,55 +1656,55 @@ int uni_do_dma(vmeDmaPacket_t * vmeDma) ...@@ -1550,55 +1656,55 @@ int uni_do_dma(vmeDmaPacket_t * vmeDma)
vmeDmaPacket_t *curDma; vmeDmaPacket_t *curDma;
TDMA_Cmd_Packet *dmaLL; TDMA_Cmd_Packet *dmaLL;
// Sanity check the VME chain. /* Sanity check the VME chain. */
channel = vmeDma->channel_number; channel = vmeDma->channel_number;
if (channel > 0) { if (channel > 0) {
return (-EINVAL); return -EINVAL;
} }
curDma = vmeDma; curDma = vmeDma;
while (curDma != 0) { while (curDma != 0) {
if (curDma->byteCount == 0) { if (curDma->byteCount == 0) {
return (-EINVAL); return -EINVAL;
} }
if (curDma->byteCount >= 0x1000000) { if (curDma->byteCount >= 0x1000000) {
return (-EINVAL); return -EINVAL;
} }
if ((curDma->srcAddr & 7) != (curDma->dstAddr & 7)) { if ((curDma->srcAddr & 7) != (curDma->dstAddr & 7)) {
return (-EINVAL); return -EINVAL;
} }
switch (curDma->srcBus) { switch (curDma->srcBus) {
case VME_DMA_PCI: case VME_DMA_PCI:
if (curDma->dstBus != VME_DMA_VME) { if (curDma->dstBus != VME_DMA_VME) {
return (-EINVAL); return -EINVAL;
} }
break; break;
case VME_DMA_VME: case VME_DMA_VME:
if (curDma->dstBus != VME_DMA_PCI) { if (curDma->dstBus != VME_DMA_PCI) {
return (-EINVAL); return -EINVAL;
} }
break; break;
default: default:
return (-EINVAL); return -EINVAL;
break; break;
} }
if (uniSetupDctlReg(curDma, &dctlreg) < 0) { if (uniSetupDctlReg(curDma, &dctlreg) < 0) {
return (-EINVAL); return -EINVAL;
} }
curDma = curDma->pNextPacket; curDma = curDma->pNextPacket;
if (curDma == vmeDma) { // Endless Loop! if (curDma == vmeDma) { /* Endless Loop! */
return (-EINVAL); return -EINVAL;
} }
} }
// calculate control register /* calculate control register */
if (vmeDma->pNextPacket != 0) { if (vmeDma->pNextPacket != 0) {
dgcsreg = 0x8000000; dgcsreg = 0x8000000;
} else { } else {
dgcsreg = 0; dgcsreg = 0;
} }
for (x = 0; x < 8; x++) { // vme block size for (x = 0; x < 8; x++) { /* vme block size */
if ((256 << x) >= vmeDma->maxVmeBlockSize) { if ((256 << x) >= vmeDma->maxVmeBlockSize) {
break; break;
} }
...@@ -1608,7 +1714,7 @@ int uni_do_dma(vmeDmaPacket_t * vmeDma) ...@@ -1608,7 +1714,7 @@ int uni_do_dma(vmeDmaPacket_t * vmeDma)
dgcsreg |= (x << 20); dgcsreg |= (x << 20);
if (vmeDma->vmeBackOffTimer) { if (vmeDma->vmeBackOffTimer) {
for (x = 1; x < 8; x++) { // vme timer for (x = 1; x < 8; x++) { /* vme timer */
if ((16 << (x - 1)) >= vmeDma->vmeBackOffTimer) { if ((16 << (x - 1)) >= vmeDma->vmeBackOffTimer) {
break; break;
} }
...@@ -1617,195 +1723,211 @@ int uni_do_dma(vmeDmaPacket_t * vmeDma) ...@@ -1617,195 +1723,211 @@ int uni_do_dma(vmeDmaPacket_t * vmeDma)
x = 7; x = 7;
dgcsreg |= (x << 16); dgcsreg |= (x << 16);
} }
// Setup the dma chain /*` Setup the dma chain */
dmaLL = uni_setup_dma(vmeDma); dmaLL = ca91cx42_setup_dma(vmeDma);
// Start the DMA /* Start the DMA */
if (dgcsreg & 0x8000000) { if (dgcsreg & 0x8000000) {
vmeDma->vmeDmaStartTick = vmeDma->vmeDmaStartTick =
uni_start_dma(channel, dgcsreg, ca91cx42_start_dma(channel, dgcsreg,
(TDMA_Cmd_Packet *) virt_to_phys(dmaLL)); (TDMA_Cmd_Packet *) virt_to_phys(dmaLL));
} else { } else {
vmeDma->vmeDmaStartTick = vmeDma->vmeDmaStartTick =
uni_start_dma(channel, dgcsreg, dmaLL); ca91cx42_start_dma(channel, dgcsreg, dmaLL);
} }
wait_event_interruptible(dma_queue[0], wait_event_interruptible(dma_queue,
readl(vmechip_baseaddr + DGCS) & 0x800); ioread32(ca91cx42_bridge->base + DGCS) & 0x800);
val = readl(vmechip_baseaddr + DGCS); val = ioread32(ca91cx42_bridge->base + DGCS);
writel(val | 0xF00, vmechip_baseaddr + DGCS); iowrite32(val | 0xF00, ca91cx42_bridge->base + DGCS);
vmeDma->vmeDmaStatus = 0; vmeDma->vmeDmaStatus = 0;
vmeDma->vmeDmaStopTick = uni_dma_irq_time;
if (vmeDma->vmeDmaStopTick < vmeDma->vmeDmaStartTick) {
vmeDma->vmeDmaElapsedTime =
(0xFFFFFFFF - vmeDma->vmeDmaStartTick) +
vmeDma->vmeDmaStopTick;
} else {
vmeDma->vmeDmaElapsedTime =
vmeDma->vmeDmaStopTick - vmeDma->vmeDmaStartTick;
}
vmeDma->vmeDmaElapsedTime -= vmechip_irq_overhead_ticks;
vmeDma->vmeDmaElapsedTime /= (tb_speed / 1000000);
if (!(val & 0x00000800)) { if (!(val & 0x00000800)) {
vmeDma->vmeDmaStatus = val & 0x700; vmeDma->vmeDmaStatus = val & 0x700;
printk(KERN_ERR printk(KERN_ERR "ca91c042: DMA Error in ca91cx42_DMA_irqhandler"
"ca91c042: DMA Error in DMA_uni_irqhandler DGCS=%08X\n", " DGCS=%08X\n", val);
val); val = ioread32(ca91cx42_bridge->base + DCPP);
val = readl(vmechip_baseaddr + DCPP);
printk(KERN_ERR "ca91c042: DCPP=%08X\n", val); printk(KERN_ERR "ca91c042: DCPP=%08X\n", val);
val = readl(vmechip_baseaddr + DCTL); val = ioread32(ca91cx42_bridge->base + DCTL);
printk(KERN_ERR "ca91c042: DCTL=%08X\n", val); printk(KERN_ERR "ca91c042: DCTL=%08X\n", val);
val = readl(vmechip_baseaddr + DTBC); val = ioread32(ca91cx42_bridge->base + DTBC);
printk(KERN_ERR "ca91c042: DTBC=%08X\n", val); printk(KERN_ERR "ca91c042: DTBC=%08X\n", val);
val = readl(vmechip_baseaddr + DLA); val = ioread32(ca91cx42_bridge->base + DLA);
printk(KERN_ERR "ca91c042: DLA=%08X\n", val); printk(KERN_ERR "ca91c042: DLA=%08X\n", val);
val = readl(vmechip_baseaddr + DVA); val = ioread32(ca91cx42_bridge->base + DVA);
printk(KERN_ERR "ca91c042: DVA=%08X\n", val); printk(KERN_ERR "ca91c042: DVA=%08X\n", val);
} }
// Free the dma chain /* Free the dma chain */
uni_free_dma(dmaLL); ca91cx42_free_dma(dmaLL);
return (0); return 0;
} }
//----------------------------------------------------------------------------- int ca91cx42_lm_set(vmeLmCfg_t *vmeLm)
// Function : uni_shutdown
// Description: Put VME bridge in quiescent state.
//-----------------------------------------------------------------------------
void uni_shutdown(void)
{ {
writel(0, vmechip_baseaddr + LINT_EN); // Turn off Ints int temp_ctl = 0;
// Turn off the windows if (vmeLm->addrU)
writel(0x00800000, vmechip_baseaddr + LSI0_CTL); return -EINVAL;
writel(0x00800000, vmechip_baseaddr + LSI1_CTL);
writel(0x00800000, vmechip_baseaddr + LSI2_CTL); switch (vmeLm->addrSpace) {
writel(0x00800000, vmechip_baseaddr + LSI3_CTL); case VME_A64:
writel(0x00F00000, vmechip_baseaddr + VSI0_CTL); case VME_USER3:
writel(0x00F00000, vmechip_baseaddr + VSI1_CTL); case VME_USER4:
writel(0x00F00000, vmechip_baseaddr + VSI2_CTL); return -EINVAL;
writel(0x00F00000, vmechip_baseaddr + VSI3_CTL); case VME_A16:
if (vmechip_revision >= 2) { temp_ctl |= 0x00000;
writel(0x00800000, vmechip_baseaddr + LSI4_CTL); break;
writel(0x00800000, vmechip_baseaddr + LSI5_CTL); case VME_A24:
writel(0x00800000, vmechip_baseaddr + LSI6_CTL); temp_ctl |= 0x10000;
writel(0x00800000, vmechip_baseaddr + LSI7_CTL); break;
writel(0x00F00000, vmechip_baseaddr + VSI4_CTL); case VME_A32:
writel(0x00F00000, vmechip_baseaddr + VSI5_CTL); temp_ctl |= 0x20000;
writel(0x00F00000, vmechip_baseaddr + VSI6_CTL); break;
writel(0x00F00000, vmechip_baseaddr + VSI7_CTL); case VME_CRCSR:
temp_ctl |= 0x50000;
break;
case VME_USER1:
temp_ctl |= 0x60000;
break;
case VME_USER2:
temp_ctl |= 0x70000;
break;
} }
/* Disable while we are mucking around */
iowrite32(0x00000000, ca91cx42_bridge->base + LM_CTL);
iowrite32(vmeLm->addr, ca91cx42_bridge->base + LM_BS);
/* Setup CTL register. */
if (vmeLm->userAccessType & VME_SUPER)
temp_ctl |= 0x00200000;
if (vmeLm->userAccessType & VME_USER)
temp_ctl |= 0x00100000;
if (vmeLm->dataAccessType & VME_PROG)
temp_ctl |= 0x00800000;
if (vmeLm->dataAccessType & VME_DATA)
temp_ctl |= 0x00400000;
/* Write ctl reg and enable */
iowrite32(0x80000000 | temp_ctl, ca91cx42_bridge->base + LM_CTL);
temp_ctl = ioread32(ca91cx42_bridge->base + LM_CTL);
return 0;
} }
//----------------------------------------------------------------------------- int ca91cx42_wait_lm(vmeLmCfg_t *vmeLm)
// Function : uni_init()
// Description:
//-----------------------------------------------------------------------------
int uni_init(void)
{ {
int result; unsigned long flags;
unsigned int tmp; unsigned int tmp;
unsigned int crcsr_addr;
unsigned int irqOverHeadStart; spin_lock_irqsave(&lm_lock, flags);
int overHeadTicks; spin_unlock_irqrestore(&lm_lock, flags);
if (tmp == 0) {
uni_shutdown(); if (vmeLm->lmWait < 10)
vmeLm->lmWait = 10;
// Write to Misc Register interruptible_sleep_on_timeout(&lm_queue, vmeLm->lmWait);
// Set VME Bus Time-out
// Arbitration Mode
// DTACK Enable
tmp = readl(vmechip_baseaddr + MISC_CTL) & 0x0832BFFF;
tmp |= 0x76040000;
writel(tmp, vmechip_baseaddr + MISC_CTL);
if (tmp & 0x20000) {
vme_syscon = 1;
} else {
vme_syscon = 0;
}
// Clear DMA status log
writel(0x00000F00, vmechip_baseaddr + DGCS);
// Clear and enable error log
writel(0x00800000, vmechip_baseaddr + L_CMDERR);
// Turn off location monitor
writel(0x00000000, vmechip_baseaddr + LM_CTL);
// Initialize crcsr map
if (vme_slotnum != -1) {
writel(vme_slotnum << 27, vmechip_baseaddr + VCSR_BS);
}
crcsr_addr = readl(vmechip_baseaddr + VCSR_BS) >> 8;
writel((unsigned int)vmechip_interboard_datap - crcsr_addr,
vmechip_baseaddr + VCSR_TO);
if (vme_slotnum != -1) {
writel(0x80000000, vmechip_baseaddr + VCSR_CTL);
}
// Turn off interrupts
writel(0x00000000, vmechip_baseaddr + LINT_EN); // Disable interrupts in the Universe first
writel(0x00FFFFFF, vmechip_baseaddr + LINT_STAT); // Clear Any Pending Interrupts
writel(0x00000000, vmechip_baseaddr + VINT_EN); // Disable interrupts in the Universe first
result =
request_irq(vmechip_irq, uni_irqhandler, IRQF_SHARED | IRQF_DISABLED,
"VMEBus (ca91c042)", vmechip_baseaddr);
if (result) {
printk(KERN_ERR
"ca91c042: can't get assigned pci irq vector %02X\n",
vmechip_irq);
return (0);
} else {
writel(0x0000, vmechip_baseaddr + LINT_MAP0); // Map all ints to 0
writel(0x0000, vmechip_baseaddr + LINT_MAP1); // Map all ints to 0
writel(0x0000, vmechip_baseaddr + LINT_MAP2); // Map all ints to 0
} }
iowrite32(0x00000000, ca91cx42_bridge->base + LM_CTL);
// Enable DMA, mailbox, VIRQ & LM Interrupts return 0;
if (vme_syscon) }
tmp = 0x00FF07FE;
else
tmp = 0x00FF0700;
writel(tmp, vmechip_baseaddr + LINT_EN);
// Do a quick sanity test of the bridge
if (readl(vmechip_baseaddr + LINT_EN) != tmp) {
return (0);
}
if (readl(vmechip_baseaddr + PCI_CLASS_REVISION) != 0x06800002) {
return (0);
}
for (tmp = 1; tmp < 0x80000000; tmp = tmp << 1) {
writel(tmp, vmechip_baseaddr + SCYC_EN);
writel(~tmp, vmechip_baseaddr + SCYC_CMP);
if (readl(vmechip_baseaddr + SCYC_EN) != tmp) {
return (0);
}
if (readl(vmechip_baseaddr + SCYC_CMP) != ~tmp) {
return (0);
}
}
// do a mail box interrupt to calibrate the interrupt overhead.
irqOverHeadStart = get_tbl(); int ca91cx42_set_arbiter(vmeArbiterCfg_t *vmeArb)
writel(0, vmechip_baseaddr + MBOX1); {
for (tmp = 0; tmp < 10; tmp++) { int temp_ctl = 0;
} int vbto = 0;
irqOverHeadStart = get_tbl(); temp_ctl = ioread32(ca91cx42_bridge->base + MISC_CTL);
writel(0, vmechip_baseaddr + MBOX1); temp_ctl &= 0x00FFFFFF;
for (tmp = 0; tmp < 10; tmp++) {
}
overHeadTicks = uni_irq_time - irqOverHeadStart; if (vmeArb->globalTimeoutTimer == 0xFFFFFFFF) {
if (overHeadTicks > 0) { vbto = 7;
vmechip_irq_overhead_ticks = overHeadTicks; } else if (vmeArb->globalTimeoutTimer > 1024) {
return -EINVAL;
} else if (vmeArb->globalTimeoutTimer == 0) {
vbto = 0;
} else { } else {
vmechip_irq_overhead_ticks = 1; vbto = 1;
while ((16 * (1 << (vbto - 1))) < vmeArb->globalTimeoutTimer)
vbto += 1;
} }
return (1); temp_ctl |= (vbto << 28);
if (vmeArb->arbiterMode == VME_PRIORITY_MODE)
temp_ctl |= 1 << 26;
if (vmeArb->arbiterTimeoutFlag)
temp_ctl |= 2 << 24;
iowrite32(temp_ctl, ca91cx42_bridge->base + MISC_CTL);
return 0;
}
int ca91cx42_get_arbiter(vmeArbiterCfg_t *vmeArb)
{
int temp_ctl = 0;
int vbto = 0;
temp_ctl = ioread32(ca91cx42_bridge->base + MISC_CTL);
vbto = (temp_ctl >> 28) & 0xF;
if (vbto != 0)
vmeArb->globalTimeoutTimer = (16 * (1 << (vbto - 1)));
if (temp_ctl & (1 << 26))
vmeArb->arbiterMode = VME_PRIORITY_MODE;
else
vmeArb->arbiterMode = VME_R_ROBIN_MODE;
if (temp_ctl & (3 << 24))
vmeArb->arbiterTimeoutFlag = 1;
return 0;
}
int ca91cx42_set_requestor(vmeRequesterCfg_t *vmeReq)
{
int temp_ctl = 0;
temp_ctl = ioread32(ca91cx42_bridge->base + MAST_CTL);
temp_ctl &= 0xFF0FFFFF;
if (vmeReq->releaseMode == 1)
temp_ctl |= (1 << 20);
if (vmeReq->fairMode == 1)
temp_ctl |= (1 << 21);
temp_ctl |= (vmeReq->requestLevel << 22);
iowrite32(temp_ctl, ca91cx42_bridge->base + MAST_CTL);
return 0;
}
int ca91cx42_get_requestor(vmeRequesterCfg_t *vmeReq)
{
int temp_ctl = 0;
temp_ctl = ioread32(ca91cx42_bridge->base + MAST_CTL);
if (temp_ctl & (1 << 20))
vmeReq->releaseMode = 1;
if (temp_ctl & (1 << 21))
vmeReq->fairMode = 1;
vmeReq->requestLevel = (temp_ctl & 0xC00000) >> 22;
return 0;
} }
#endif
...@@ -4,9 +4,12 @@ ...@@ -4,9 +4,12 @@
* Support for the Tundra Universe 1 and Universe II VME bridge chips * Support for the Tundra Universe 1 and Universe II VME bridge chips
* *
* Author: Tom Armistead * Author: Tom Armistead
* Updated and maintained by Ajit Prem * Updated by Ajit Prem
* Copyright 2004 Motorola Inc. * Copyright 2004 Motorola Inc.
* *
* Further updated by Martyn Welch <martyn.welch@gefanuc.com>
* Copyright 2009 GE Fanuc Intelligent Platforms Embedded Systems, Inc.
*
* Derived from ca91c042.h by Michael Wyrick * Derived from ca91c042.h by Michael Wyrick
* *
* This program is free software; you can redistribute it and/or modify it * This program is free software; you can redistribute it and/or modify it
...@@ -15,140 +18,50 @@ ...@@ -15,140 +18,50 @@
* option) any later version. * option) any later version.
*/ */
#ifndef _ca91c042_h #ifndef _CA91CX42_H
#define _ca91c042_h #define _CA91CX42_H
#ifndef PCI_VENDOR_ID_TUNDRA #ifndef PCI_VENDOR_ID_TUNDRA
#define PCI_VENDOR_ID_TUNDRA 0x10e3 #define PCI_VENDOR_ID_TUNDRA 0x10e3
#endif #endif
#ifndef PCI_DEVICE_ID_TUNDRA_CA91C042 #ifndef PCI_DEVICE_ID_TUNDRA_CA91C142
#define PCI_DEVICE_ID_TUNDRA_CA91C042 0x0000 #define PCI_DEVICE_ID_TUNDRA_CA91C142 0x0000
#endif #endif
//-----------------------------------------------------------------------------
// Public Functions
//-----------------------------------------------------------------------------
// This is the typedef for a VmeIrqHandler
typedef void (*TirqHandler) (int vmeirq, int vector, void *dev_id,
struct pt_regs * regs);
// This is the typedef for a DMA Transfer Callback function
typedef void (*TDMAcallback) (int status);
// Returns the PCI baseaddress of the Universe chip
char *Universe_BaseAddr(void);
// Returns the PCI IRQ That the universe is using
int Universe_IRQ(void);
char *mapvme(unsigned int pci, unsigned int vme, unsigned int size,
int image, int ctl);
void unmapvme(char *ptr, int image);
// Interrupt Stuff
void enable_vmeirq(unsigned int irq);
void disable_vmeirq(unsigned int irq);
int request_vmeirq(unsigned int irq, TirqHandler);
void free_vmeirq(unsigned int irq);
// DMA Stuff
int VME_Bus_Error(void);
int uni_procinfo(char *);
#define IRQ_VOWN 0x0001
#define IRQ_VIRQ1 0x0002
#define IRQ_VIRQ2 0x0004
#define IRQ_VIRQ3 0x0008
#define IRQ_VIRQ4 0x0010
#define IRQ_VIRQ5 0x0020
#define IRQ_VIRQ6 0x0040
#define IRQ_VIRQ7 0x0080
#define IRQ_DMA 0x0100
#define IRQ_LERR 0x0200
#define IRQ_VERR 0x0400
#define IRQ_res 0x0800
#define IRQ_IACK 0x1000
#define IRQ_SWINT 0x2000
#define IRQ_SYSFAIL 0x4000
#define IRQ_ACFAIL 0x8000
// See Page 2-77 in the Universe User Manual
typedef struct {
unsigned int dctl; // DMA Control
unsigned int dtbc; // Transfer Byte Count
unsigned int dlv; // PCI Address
unsigned int res1; // Reserved
unsigned int dva; // Vme Address
unsigned int res2; // Reserved
unsigned int dcpp; // Pointer to Numed Cmd Packet with rPN
unsigned int res3; // Reserved
} TDMA_Cmd_Packet;
/* /*
* Below here is normaly not used by a user module * Define the number of each that the CA91C142 supports.
*/ */
#define DMATIMEOUT 2*HZ; #define CA91C142_MAX_MASTER 8 /* Max Master Windows */
#define CA91C142_MAX_SLAVE 8 /* Max Slave Windows */
// Define for the Universe #define CA91C142_MAX_DMA 1 /* Max DMA Controllers */
#define SEEK_SET 0 #define CA91C142_MAX_MAILBOX 4 /* Max Mail Box registers */
#define SEEK_CUR 1
/* See Page 2-77 in the Universe User Manual */
#define CONFIG_REG_SPACE 0xA0000000 struct ca91cx42_dma_descriptor {
unsigned int dctl; /* DMA Control */
unsigned int dtbc; /* Transfer Byte Count */
unsigned int dlv; /* PCI Address */
unsigned int res1; /* Reserved */
unsigned int dva; /* Vme Address */
unsigned int res2; /* Reserved */
unsigned int dcpp; /* Pointer to Numed Cmd Packet with rPN */
unsigned int res3; /* Reserved */
};
struct ca91cx42_dma_entry {
struct ca91cx42_dma_descriptor descriptor;
struct list_head list;
};
/* Universe Register Offsets */ /* Universe Register Offsets */
/* general PCI configuration registers */ /* general PCI configuration registers */
#define UNIV_PCI_ID 0x000 #define CA91CX42_PCI_ID 0x000
#define UNIV_PCI_CSR 0x004 #define CA91CX42_PCI_CSR 0x004
#define UNIV_PCI_CLASS 0x008 #define CA91CX42_PCI_CLASS 0x008
#define UNIV_BM_PCI_CLASS_BASE 0xFF000000 #define CA91CX42_PCI_MISC0 0x00C
#define UNIV_OF_PCI_CLASS_BASE 24 #define CA91CX42_PCI_BS 0x010
#define UNIV_BM_PCI_CLASS_SUB 0x00FF0000 #define CA91CX42_PCI_MISC1 0x03C
#define UNIV_OF_PCI_CLASS_SUB 16
#define UNIV_BM_PCI_CLASS_PROG 0x0000FF00
#define UNIV_OF_PCI_CLASS_PROG 8
#define UNIV_BM_PCI_CLASS_RID 0x000000FF
#define UNIV_OF_PCI_CLASS_RID 0
#define UNIV_OF_PCI_CLASS_RID_UNIVERSE_I 0
#define UNIV_OF_PCI_CLASS_RID_UNIVERSE_II 1
#define UNIV_PCI_MISC0 0x00C
#define UNIV_BM_PCI_MISC0_BISTC 0x80000000
#define UNIV_BM_PCI_MISC0_SBIST 0x60000000
#define UNIV_BM_PCI_MISC0_CCODE 0x0F000000
#define UNIV_BM_PCI_MISC0_MFUNCT 0x00800000
#define UNIV_BM_PCI_MISC0_LAYOUT 0x007F0000
#define UNIV_BM_PCI_MISC0_LTIMER 0x0000FF00
#define UNIV_OF_PCI_MISC0_LTIMER 8
#define UNIV_PCI_BS 0x010
#define UNIV_PCI_MISC1 0x03C
#define UNIV_BM_LSI_CTL_EN 0x80000000
#define UNIV_BM_LSI_CTL_PWEN 0x40000000
#define UNIV_BM_LSI_CTL_VDW 0x00C00000
#define UNIV_OF_LSI_CTL_VDW 22
#define UNIV_BM_LSI_CTL_VAS 0x00070000
#define UNIV_OF_LSI_CTL_VAS 16
#define UNIV_BM_LSI_CTL_PGM 0x0000C000
#define UNIV_OF_LSI_CTL_PGM 14
#define UNIV_BM_LSI_CTL_SUPER 0x00003000
#define UNIV_OF_LSI_CTL_SUPER 12
#define UNIV_BM_LSI_CTL_VCT 0x00000100
#define UNIV_BM_LSI_CTL_LAS 0x00000003
#define UNIV_OF_LSI_CTL_LAS 0
#define UNIV_BM_LSI_CTL_RESERVED (~ (UNIV_BM_LSI_CTL_EN | UNIV_BM_LSI_CTL_PWEN | UNIV_BM_LSI_CTL_VDW | UNIV_BM_LSI_CTL_VAS | UNIV_BM_LSI_CTL_PGM | UNIV_BM_LSI_CTL_SUPER | UNIV_BM_LSI_CTL_VCT | UNIV_BM_LSI_CTL_LAS))
#define PCI_SIZE_8 0x0001
#define PCI_SIZE_16 0x0002
#define PCI_SIZE_32 0x0003
#define IOCTL_SET_CTL 0xF001
#define IOCTL_SET_BS 0xF002
#define IOCTL_SET_BD 0xF003
#define IOCTL_SET_TO 0xF004
#define IOCTL_PCI_SIZE 0xF005
#define IOCTL_SET_MODE 0xF006
#define IOCTL_SET_WINT 0xF007 // Wait for interrupt before read
#define LSI0_CTL 0x0100 #define LSI0_CTL 0x0100
#define LSI0_BS 0x0104 #define LSI0_BS 0x0104
...@@ -190,30 +103,25 @@ typedef struct { ...@@ -190,30 +103,25 @@ typedef struct {
#define LSI7_BD 0x01E4 #define LSI7_BD 0x01E4
#define LSI7_TO 0x01E8 #define LSI7_TO 0x01E8
static const int CA91CX42_LSI_CTL[] = { LSI0_CTL, LSI1_CTL, LSI2_CTL, LSI3_CTL,
LSI4_CTL, LSI5_CTL, LSI6_CTL, LSI7_CTL };
static const int CA91CX42_LSI_BS[] = { LSI0_BS, LSI1_BS, LSI2_BS, LSI3_BS,
LSI4_BS, LSI5_BS, LSI6_BS, LSI7_BS };
static const int CA91CX42_LSI_BD[] = { LSI0_BD, LSI1_BD, LSI2_BD, LSI3_BD,
LSI4_BD, LSI5_BD, LSI6_BD, LSI7_BD };
static const int CA91CX42_LSI_TO[] = { LSI0_TO, LSI1_TO, LSI2_TO, LSI3_TO,
LSI4_TO, LSI5_TO, LSI6_TO, LSI7_TO };
#define SCYC_CTL 0x0170 #define SCYC_CTL 0x0170
#define SCYC_ADDR 0x0174 #define SCYC_ADDR 0x0174
#define SCYC_EN 0x0178 #define SCYC_EN 0x0178
#define SCYC_CMP 0x017C #define SCYC_CMP 0x017C
#define SCYC_SWP 0x0180 #define SCYC_SWP 0x0180
#define LMISC 0x0184 #define LMISC 0x0184
#define UNIV_BM_LMISC_CRT 0xF0000000
#define UNIV_OF_LMISC_CRT 28
#define UNIV_BM_LMISC_CWT 0x0F000000
#define UNIV_OF_LMISC_CWT 24
#define SLSI 0x0188 #define SLSI 0x0188
#define UNIV_BM_SLSI_EN 0x80000000
#define UNIV_BM_SLSI_PWEN 0x40000000
#define UNIV_BM_SLSI_VDW 0x00F00000
#define UNIV_OF_SLSI_VDW 20
#define UNIV_BM_SLSI_PGM 0x0000F000
#define UNIV_OF_SLSI_PGM 12
#define UNIV_BM_SLSI_SUPER 0x00000F00
#define UNIV_OF_SLSI_SUPER 8
#define UNIV_BM_SLSI_BS 0x000000F6
#define UNIV_OF_SLSI_BS 2
#define UNIV_BM_SLSI_LAS 0x00000003
#define UNIV_OF_SLSI_LAS 0
#define UNIV_BM_SLSI_RESERVED 0x3F0F0000
#define L_CMDERR 0x018C #define L_CMDERR 0x018C
#define LAERR 0x0190 #define LAERR 0x0190
...@@ -226,24 +134,6 @@ typedef struct { ...@@ -226,24 +134,6 @@ typedef struct {
#define D_LLUE 0x0224 #define D_LLUE 0x0224
#define LINT_EN 0x0300 #define LINT_EN 0x0300
#define UNIV_BM_LINT_ACFAIL 0x00008000
#define UNIV_BM_LINT_SYSFAIL 0x00004000
#define UNIV_BM_LINT_SW_INT 0x00002000
#define UNIV_BM_LINT_SW_IACK 0x00001000
#define UNIV_BM_LINT_VERR 0x00000400
#define UNIV_BM_LINT_LERR 0x00000200
#define UNIV_BM_LINT_DMA 0x00000100
#define UNIV_BM_LINT_LM 0x00F00000
#define UNIV_BM_LINT_MBOX 0x000F0000
#define UNIV_BM_LINT_VIRQ 0x000000FE
#define UNIV_BM_LINT_VIRQ7 0x00000080
#define UNIV_BM_LINT_VIRQ6 0x00000040
#define UNIV_BM_LINT_VIRQ5 0x00000020
#define UNIV_BM_LINT_VIRQ4 0x00000010
#define UNIV_BM_LINT_VIRQ3 0x00000008
#define UNIV_BM_LINT_VIRQ2 0x00000004
#define UNIV_BM_LINT_VIRQ1 0x00000002
#define UNIV_BM_LINT_VOWN 0x00000001
#define LINT_STAT 0x0304 #define LINT_STAT 0x0304
#define LINT_MAP0 0x0308 #define LINT_MAP0 0x0308
#define LINT_MAP1 0x030C #define LINT_MAP1 0x030C
...@@ -252,6 +142,7 @@ typedef struct { ...@@ -252,6 +142,7 @@ typedef struct {
#define VINT_MAP0 0x0318 #define VINT_MAP0 0x0318
#define VINT_MAP1 0x031C #define VINT_MAP1 0x031C
#define STATID 0x0320 #define STATID 0x0320
#define V1_STATID 0x0324 #define V1_STATID 0x0324
#define V2_STATID 0x0328 #define V2_STATID 0x0328
#define V3_STATID 0x032C #define V3_STATID 0x032C
...@@ -259,6 +150,11 @@ typedef struct { ...@@ -259,6 +150,11 @@ typedef struct {
#define V5_STATID 0x0334 #define V5_STATID 0x0334
#define V6_STATID 0x0338 #define V6_STATID 0x0338
#define V7_STATID 0x033C #define V7_STATID 0x033C
static const int CA91CX42_V_STATID[8] = { 0, V1_STATID, V2_STATID, V3_STATID,
V4_STATID, V5_STATID, V6_STATID,
V7_STATID };
#define LINT_MAP2 0x0340 #define LINT_MAP2 0x0340
#define VINT_MAP2 0x0344 #define VINT_MAP2 0x0344
...@@ -270,46 +166,8 @@ typedef struct { ...@@ -270,46 +166,8 @@ typedef struct {
#define SEMA1 0x035C #define SEMA1 0x035C
#define MAST_CTL 0x0400 #define MAST_CTL 0x0400
#define UNIV_BM_MAST_CTL_MAXRTRY 0xF0000000
#define UNIV_OF_MAST_CTL_MAXRTRY 28
#define UNIV_BM_MAST_CTL_PWON 0x0F000000
#define UNIV_OF_MAST_CTL_PWON 24
#define UNIV_BM_MAST_CTL_VRL 0x00C00000
#define UNIV_OF_MAST_CTL_VRL 22
#define UNIV_BM_MAST_CTL_VRM 0x00200000
#define UNIV_BM_MAST_CTL_VREL 0x00100000
#define UNIV_BM_MAST_CTL_VOWN 0x00080000
#define UNIV_BM_MAST_CTL_VOWN_ACK 0x00040000
#define UNIV_BM_MAST_CTL_PABS 0x00001000
#define UNIV_BM_MAST_CTL_BUS_NO 0x0000000F
#define UNIV_OF_MAST_CTL_BUS_NO 0
#define MISC_CTL 0x0404 #define MISC_CTL 0x0404
#define UNIV_BM_MISC_CTL_VBTO 0xF0000000
#define UNIV_OF_MISC_CTL_VBTO 28
#define UNIV_BM_MISC_CTL_VARB 0x04000000
#define UNIV_BM_MISC_CTL_VARBTO 0x03000000
#define UNIV_OF_MISC_CTL_VARBTO 24
#define UNIV_BM_MISC_CTL_SW_LRST 0x00800000
#define UNIV_BM_MISC_CTL_SW_SRST 0x00400000
#define UNIV_BM_MISC_CTL_BI 0x00100000
#define UNIV_BM_MISC_CTL_ENGBI 0x00080000
#define UNIV_BM_MISC_CTL_RESCIND 0x00040000
#define UNIV_BM_MISC_CTL_SYSCON 0x00020000
#define UNIV_BM_MISC_CTL_V64AUTO 0x00010000
#define UNIV_BM_MISC_CTL_RESERVED 0x0820FFFF
#define MISC_STAT 0x0408 #define MISC_STAT 0x0408
#define UNIV_BM_MISC_STAT_ENDIAN 0x80000000
#define UNIV_BM_MISC_STAT_LCLSIZE 0x40000000
#define UNIV_BM_MISC_STAT_DY4AUTO 0x08000000
#define UNIV_BM_MISC_STAT_MYBBSY 0x00200000
#define UNIV_BM_MISC_STAT_DY4DONE 0x00080000
#define UNIV_BM_MISC_STAT_TXFE 0x00040000
#define UNIV_BM_MISC_STAT_RXFE 0x00020000
#define UNIV_BM_MISC_STAT_DY4AUTOID 0x0000FF00
#define UNIV_OF_MISC_STAT_DY4AUTOID 8
#define USER_AM 0x040C #define USER_AM 0x040C
#define VSI0_CTL 0x0F00 #define VSI0_CTL 0x0F00
...@@ -336,13 +194,6 @@ typedef struct { ...@@ -336,13 +194,6 @@ typedef struct {
#define LM_BS 0x0F68 #define LM_BS 0x0F68
#define VRAI_CTL 0x0F70 #define VRAI_CTL 0x0F70
#define UNIV_BM_VRAI_CTL_EN 0x80000000
#define UNIV_BM_VRAI_CTL_PGM 0x00C00000
#define UNIV_OF_VRAI_CTL_PGM 22
#define UNIV_BM_VRAI_CTL_SUPER 0x00300000
#define UNIV_OF_VRAI_CTL_SUPER 20
#define UNIV_BM_VRAI_CTL_VAS 0x00030000
#define UNIV_OF_VRAI_CTL_VAS 16
#define VRAI_BS 0x0F74 #define VRAI_BS 0x0F74
#define VCSR_CTL 0x0F80 #define VCSR_CTL 0x0F80
...@@ -370,6 +221,18 @@ typedef struct { ...@@ -370,6 +221,18 @@ typedef struct {
#define VSI7_BD 0x0FD4 #define VSI7_BD 0x0FD4
#define VSI7_TO 0x0FD8 #define VSI7_TO 0x0FD8
static const int CA91CX42_VSI_CTL[] = { VSI0_CTL, VSI1_CTL, VSI2_CTL, VSI3_CTL,
VSI4_CTL, VSI5_CTL, VSI6_CTL, VSI7_CTL };
static const int CA91CX42_VSI_BS[] = { VSI0_BS, VSI1_BS, VSI2_BS, VSI3_BS,
VSI4_BS, VSI5_BS, VSI6_BS, VSI7_BS };
static const int CA91CX42_VSI_BD[] = { VSI0_BD, VSI1_BD, VSI2_BD, VSI3_BD,
VSI4_BD, VSI5_BD, VSI6_BD, VSI7_BD };
static const int CA91CX42_VSI_TO[] = { VSI0_TO, VSI1_TO, VSI2_TO, VSI3_TO,
VSI4_TO, VSI5_TO, VSI6_TO, VSI7_TO };
#define VCSR_CLR 0x0FF4 #define VCSR_CLR 0x0FF4
#define VCSR_SET 0x0FF8 #define VCSR_SET 0x0FF8
#define VCSR_BS 0x0FFC #define VCSR_BS 0x0FFC
...@@ -400,4 +263,243 @@ typedef struct { ...@@ -400,4 +263,243 @@ typedef struct {
// 0110 = 512 // 0110 = 512
// 0111 = 1024 // 0111 = 1024
#endif /* _ca91c042_h */ /*
* PCI Class Register
* offset 008
*/
#define CA91CX42_BM_PCI_CLASS_BASE 0xFF000000
#define CA91CX42_OF_PCI_CLASS_BASE 24
#define CA91CX42_BM_PCI_CLASS_SUB 0x00FF0000
#define CA91CX42_OF_PCI_CLASS_SUB 16
#define CA91CX42_BM_PCI_CLASS_PROG 0x0000FF00
#define CA91CX42_OF_PCI_CLASS_PROG 8
#define CA91CX42_BM_PCI_CLASS_RID 0x000000FF
#define CA91CX42_OF_PCI_CLASS_RID 0
#define CA91CX42_OF_PCI_CLASS_RID_UNIVERSE_I 0
#define CA91CX42_OF_PCI_CLASS_RID_UNIVERSE_II 1
/*
* PCI Misc Register
* offset 00C
*/
#define CA91CX42_BM_PCI_MISC0_BISTC 0x80000000
#define CA91CX42_BM_PCI_MISC0_SBIST 0x60000000
#define CA91CX42_BM_PCI_MISC0_CCODE 0x0F000000
#define CA91CX42_BM_PCI_MISC0_MFUNCT 0x00800000
#define CA91CX42_BM_PCI_MISC0_LAYOUT 0x007F0000
#define CA91CX42_BM_PCI_MISC0_LTIMER 0x0000FF00
#define CA91CX42_OF_PCI_MISC0_LTIMER 8
/*
* LSI Control Register
* offset 100
*/
#define CA91CX42_LSI_CTL_EN (1<<31)
#define CA91CX42_LSI_CTL_PWEN (1<<30)
#define CA91CX42_LSI_CTL_VDW_M (3<<22)
#define CA91CX42_LSI_CTL_VDW_D8 0
#define CA91CX42_LSI_CTL_VDW_D16 (1<<22)
#define CA91CX42_LSI_CTL_VDW_D32 (1<<23)
#define CA91CX42_LSI_CTL_VDW_D64 (3<<22)
#define CA91CX42_LSI_CTL_VAS_M (7<<16)
#define CA91CX42_LSI_CTL_VAS_A16 0
#define CA91CX42_LSI_CTL_VAS_A24 (1<<16)
#define CA91CX42_LSI_CTL_VAS_A32 (1<<17)
#define CA91CX42_LSI_CTL_VAS_CRCSR (5<<16)
#define CA91CX42_LSI_CTL_VAS_USER1 (3<<17)
#define CA91CX42_LSI_CTL_VAS_USER2 (7<<16)
#define CA91CX42_LSI_CTL_PGM_M (1<<14)
#define CA91CX42_LSI_CTL_PGM_DATA 0
#define CA91CX42_LSI_CTL_PGM_PGM (1<<14)
#define CA91CX42_LSI_CTL_SUPER_M (1<<12)
#define CA91CX42_LSI_CTL_SUPER_NPRIV 0
#define CA91CX42_LSI_CTL_SUPER_SUPR (1<<12)
#define CA91CX42_LSI_CTL_VCT_M (1<<8)
#define CA91CX42_LSI_CTL_VCT_BLT (1<<8)
#define CA91CX42_LSI_CTL_VCT_MBLT (1<<8)
#define CA91CX42_LSI_CTL_LAS (1<<0)
/*
* LMISC Register
* offset 184
*/
#define CA91CX42_BM_LMISC_CRT 0xF0000000
#define CA91CX42_OF_LMISC_CRT 28
#define CA91CX42_BM_LMISC_CWT 0x0F000000
#define CA91CX42_OF_LMISC_CWT 24
/*
* SLSI Register
* offset 188
*/
#define CA91CX42_BM_SLSI_EN 0x80000000
#define CA91CX42_BM_SLSI_PWEN 0x40000000
#define CA91CX42_BM_SLSI_VDW 0x00F00000
#define CA91CX42_OF_SLSI_VDW 20
#define CA91CX42_BM_SLSI_PGM 0x0000F000
#define CA91CX42_OF_SLSI_PGM 12
#define CA91CX42_BM_SLSI_SUPER 0x00000F00
#define CA91CX42_OF_SLSI_SUPER 8
#define CA91CX42_BM_SLSI_BS 0x000000F6
#define CA91CX42_OF_SLSI_BS 2
#define CA91CX42_BM_SLSI_LAS 0x00000003
#define CA91CX42_OF_SLSI_LAS 0
#define CA91CX42_BM_SLSI_RESERVED 0x3F0F0000
/*
* PCI Interrupt Enable Register
* offset 300
*/
#define CA91CX42_LINT_LM3 0x00800000
#define CA91CX42_LINT_LM2 0x00400000
#define CA91CX42_LINT_LM1 0x00200000
#define CA91CX42_LINT_LM0 0x00100000
#define CA91CX42_LINT_MBOX3 0x00080000
#define CA91CX42_LINT_MBOX2 0x00040000
#define CA91CX42_LINT_MBOX1 0x00020000
#define CA91CX42_LINT_MBOX0 0x00010000
#define CA91CX42_LINT_ACFAIL 0x00008000
#define CA91CX42_LINT_SYSFAIL 0x00004000
#define CA91CX42_LINT_SW_INT 0x00002000
#define CA91CX42_LINT_SW_IACK 0x00001000
#define CA91CX42_LINT_VERR 0x00000400
#define CA91CX42_LINT_LERR 0x00000200
#define CA91CX42_LINT_DMA 0x00000100
#define CA91CX42_LINT_VIRQ7 0x00000080
#define CA91CX42_LINT_VIRQ6 0x00000040
#define CA91CX42_LINT_VIRQ5 0x00000020
#define CA91CX42_LINT_VIRQ4 0x00000010
#define CA91CX42_LINT_VIRQ3 0x00000008
#define CA91CX42_LINT_VIRQ2 0x00000004
#define CA91CX42_LINT_VIRQ1 0x00000002
#define CA91CX42_LINT_VOWN 0x00000001
static const int CA91CX42_LINT_VIRQ[] = { 0, CA91CX42_LINT_VIRQ1,
CA91CX42_LINT_VIRQ2, CA91CX42_LINT_VIRQ3,
CA91CX42_LINT_VIRQ4, CA91CX42_LINT_VIRQ5,
CA91CX42_LINT_VIRQ6, CA91CX42_LINT_VIRQ7 };
#define CA91CX42_LINT_MBOX 0x000F0000
static const int CA91CX42_LINT_LM[] = { CA91CX42_LINT_LM0, CA91CX42_LINT_LM1,
CA91CX42_LINT_LM2, CA91CX42_LINT_LM3 };
/*
* MAST_CTL Register
* offset 400
*/
#define CA91CX42_BM_MAST_CTL_MAXRTRY 0xF0000000
#define CA91CX42_OF_MAST_CTL_MAXRTRY 28
#define CA91CX42_BM_MAST_CTL_PWON 0x0F000000
#define CA91CX42_OF_MAST_CTL_PWON 24
#define CA91CX42_BM_MAST_CTL_VRL 0x00C00000
#define CA91CX42_OF_MAST_CTL_VRL 22
#define CA91CX42_BM_MAST_CTL_VRM 0x00200000
#define CA91CX42_BM_MAST_CTL_VREL 0x00100000
#define CA91CX42_BM_MAST_CTL_VOWN 0x00080000
#define CA91CX42_BM_MAST_CTL_VOWN_ACK 0x00040000
#define CA91CX42_BM_MAST_CTL_PABS 0x00001000
#define CA91CX42_BM_MAST_CTL_BUS_NO 0x0000000F
#define CA91CX42_OF_MAST_CTL_BUS_NO 0
/*
* MISC_CTL Register
* offset 404
*/
#define CA91CX42_MISC_CTL_VBTO 0xF0000000
#define CA91CX42_MISC_CTL_VARB 0x04000000
#define CA91CX42_MISC_CTL_VARBTO 0x03000000
#define CA91CX42_MISC_CTL_SW_LRST 0x00800000
#define CA91CX42_MISC_CTL_SW_SRST 0x00400000
#define CA91CX42_MISC_CTL_BI 0x00100000
#define CA91CX42_MISC_CTL_ENGBI 0x00080000
#define CA91CX42_MISC_CTL_RESCIND 0x00040000
#define CA91CX42_MISC_CTL_SYSCON 0x00020000
#define CA91CX42_MISC_CTL_V64AUTO 0x00010000
#define CA91CX42_MISC_CTL_RESERVED 0x0820FFFF
#define CA91CX42_OF_MISC_CTL_VARBTO 24
#define CA91CX42_OF_MISC_CTL_VBTO 28
/*
* MISC_STAT Register
* offset 408
*/
#define CA91CX42_BM_MISC_STAT_ENDIAN 0x80000000
#define CA91CX42_BM_MISC_STAT_LCLSIZE 0x40000000
#define CA91CX42_BM_MISC_STAT_DY4AUTO 0x08000000
#define CA91CX42_BM_MISC_STAT_MYBBSY 0x00200000
#define CA91CX42_BM_MISC_STAT_DY4DONE 0x00080000
#define CA91CX42_BM_MISC_STAT_TXFE 0x00040000
#define CA91CX42_BM_MISC_STAT_RXFE 0x00020000
#define CA91CX42_BM_MISC_STAT_DY4AUTOID 0x0000FF00
#define CA91CX42_OF_MISC_STAT_DY4AUTOID 8
/*
* VSI Control Register
* offset F00
*/
#define CA91CX42_VSI_CTL_EN (1<<31)
#define CA91CX42_VSI_CTL_PWEN (1<<30)
#define CA91CX42_VSI_CTL_PREN (1<<29)
#define CA91CX42_VSI_CTL_PGM_M (3<<22)
#define CA91CX42_VSI_CTL_PGM_DATA (1<<22)
#define CA91CX42_VSI_CTL_PGM_PGM (1<<23)
#define CA91CX42_VSI_CTL_SUPER_M (3<<20)
#define CA91CX42_VSI_CTL_SUPER_NPRIV (1<<20)
#define CA91CX42_VSI_CTL_SUPER_SUPR (1<<21)
#define CA91CX42_VSI_CTL_VAS_M (7<<16)
#define CA91CX42_VSI_CTL_VAS_A16 0
#define CA91CX42_VSI_CTL_VAS_A24 (1<<16)
#define CA91CX42_VSI_CTL_VAS_A32 (1<<17)
#define CA91CX42_VSI_CTL_VAS_USER1 (3<<17)
#define CA91CX42_VSI_CTL_VAS_USER2 (7<<16)
#define CA91CX42_VSI_CTL_LD64EN (1<<7)
#define CA91CX42_VSI_CTL_LLRMW (1<<6)
#define CA91CX42_VSI_CTL_LAS_M (3<<0)
#define CA91CX42_VSI_CTL_LAS_PCI_MS 0
#define CA91CX42_VSI_CTL_LAS_PCI_IO (1<<0)
#define CA91CX42_VSI_CTL_LAS_PCI_CONF (1<<1)
/*
* VRAI_CTL Register
* offset F70
*/
#define CA91CX42_BM_VRAI_CTL_EN 0x80000000
#define CA91CX42_BM_VRAI_CTL_PGM 0x00C00000
#define CA91CX42_OF_VRAI_CTL_PGM 22
#define CA91CX42_BM_VRAI_CTL_SUPER 0x00300000
#define CA91CX42_OF_VRAI_CTL_SUPER 20
#define CA91CX42_BM_VRAI_CTL_VAS 0x00030000
#define CA91CX42_OF_VRAI_CTL_VAS 16
/* VCSR_CTL Register
* offset F80
*/
#define CA91CX42_VCSR_CTL_EN (1<<31)
#define CA91CX42_VCSR_CTL_LAS_M (3<<0)
#define CA91CX42_VCSR_CTL_LAS_PCI_MS 0
#define CA91CX42_VCSR_CTL_LAS_PCI_IO (1<<0)
#define CA91CX42_VCSR_CTL_LAS_PCI_CONF (1<<1)
/* VCSR_BS Register
* offset FFC
*/
#define CA91CX42_VCSR_BS_SLOT_M (0x1F<<27)
#endif /* _CA91CX42_H */
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment