code: purgatorio

ref: 09d11c64e5014a64dbc7b807899b3dd2581fe26f
dir: /os/cerf405/io.h/

View raw version
typedef struct BD BD;
typedef struct Ring Ring;
typedef struct MALdev MALdev;
typedef struct I2Cdev I2Cdev;

enum
{
	/* 405EP UIC interrupt vectors (IBM bit numbering) */
	VectorUIC= 0,
		VectorUART0=VectorUIC,
		VectorUART1,
		VectorIIC,
		VectorPCIECW,
		VectorRsvd1,
		VectorDMA0,
		VectorDMA1,
		VectorDMA2,
		VectorDMA3,
		VectorEtherwake,
		VectorMALSERR,
		VectorMALTXEOB,
		VectorMALRXEOB,
		VectorMALTXDE,
		VectorMALRXDE,
		VectorEMAC0,
		VectorPCISERR,
		VectorEMAC1,
		VectorPCIPM,
		VectorGPT0,
		VectorGPT1,
		VectorGPT2,
		VectorGPT3,
		VectorGPT4,
		/* 1 reserved */
	VectorIRQ=	VectorUIC+25,	/* IRQ0 to IRQ6 */
	MaxVector=	VectorIRQ+7,

	/* some flags to change polarity and sensitivity */
	IRQmask=		0xFF,	/* actual vector address */
	IRQactivelow=	1<<8,
	IRQedge=		1<<9,
	IRQcritical=	1<<10,
};

/*
 * these are defined to keep the interface compatible with other
 * architectures, but only BUSUNKNOWN is currently used
 */
#define MKBUS(t,b,d,f)	(((t)<<24)|(((b)&0xFF)<<16)|(((d)&0x1F)<<11)|(((f)&0x07)<<8))
#define BUSFNO(tbdf)	(((tbdf)>>8)&0x07)
#define BUSDNO(tbdf)	(((tbdf)>>11)&0x1F)
#define BUSBNO(tbdf)	(((tbdf)>>16)&0xFF)
#define BUSTYPE(tbdf)	((tbdf)>>24)
#define BUSBDF(tbdf)	((tbdf)&0x00FFFF00)
#define BUSUNKNOWN	(-1)

enum {
	BusOPB,
	BusPLB,
	BusPCI,
	MaxBus
};

/*
 * MAL Buffer Descriptors and IO Rings
 */

struct BD {
	ushort	status;
	ushort	length;
	ulong	addr;
};
#define	MAXIORING	256	/* hardware limit to ring size */
#define	BDBUFLIM	(4096-16)	/* no MAL buffer larger than this */

BD*	bdalloc(ulong);
void	bdfree(BD*, int);
void	dumpbd(char*, BD*, int);

enum {
	/* Rx BDs, bits common to all protocols */
	BDEmpty=	1<<15,
	BDWrap=		1<<14,	/* end of ring */
	BDContin=	1<<13,	/* continuous mode */
	BDLast=		1<<12,	/* last buffer in current packet */
	BDFirst=		1<<11,	/* first buffer in current packet (set by MAL) */
	BDInt=		1<<10,	/* interrupt when done */

	/* Tx BDs */
	BDReady=		1<<15,	/* ready to transmit; set by driver, cleared by MAL */
	/* BDWrap, BDInt, BDLast as above */
};

struct Ring {
	BD*	rdr;				/* receive descriptor ring */
	Block**	rxb;			/* receive ring buffers */
	int	rdrx;				/* index into rdr */
	int	nrdre;			/* length of rdr */

	BD*	tdr;				/* transmit descriptor ring */
	Block**	txb;			/* transmit ring buffers */
	int	tdrh;				/* host index into tdr */
	int	tdri;				/* interface index into tdr */
	int	ntdre;			/* length of tdr */
	int	ntq;				/* pending transmit requests */
};

#define NEXT(x, l)	(((x)+1)%(l))
#define PREV(x, l)	(((x) == 0) ? (l)-1: (x)-1)
#define	HOWMANY(x, y)	(((x)+((y)-1))/(y))
#define ROUNDUP(x, y)	(HOWMANY((x), (y))*(y))

/*
 * one per mal channel
 */
typedef struct Mal Mal;
struct Mal {
	int	n;
	int	len;
	int	tx;
	ulong	mask;

	void*	arg;
	void	(*interrupt)(Ureg*, void*);
};

Mal*	malchannel(int, int, void (*)(Ureg*, void*), void*);
void	maltxreset(Mal*);
void	maltxinit(Mal*, Ring*);
void	maltxenable(Mal*);
void	malrxreset(Mal*);
void	malrxinit(Mal*, Ring*, ulong);
void	malrxenable(Mal*);
void	ioringreserve(int, ulong, int, ulong);
int	ioringinit(Ring*, int, int);

typedef struct Gpioregs Gpioregs;
struct Gpioregs {
	ulong	or;	/* output register */
	ulong	tcr;	/* tristate control */
	ulong	osrh;	/* output select high (0-15) */
	ulong	osrl;	/* output select low (16-31) */
	ulong	tsrh;	/* tristate select high (0-15) */
	ulong	tsrl;	/* tristate select low (16-31) */
	ulong	odr;	/* open drain */
	ulong	ir;	/* input */
	ulong	rr1;	/* receive register */
	ulong	pad[3];
	ulong	isr1h;	/* input select 1 high (0-15) */
	ulong	isr1l;	/* input select 1 low (16-31) */
};

enum {
	/* software configuration bits for gpioconfig */
	Gpio_Alt1=	1<<0,	/* implies specific settings of all the others, but include in or out */
	Gpio_OD=	1<<1,
	Gpio_Tri=		1<<2,
	Gpio_in=		1<<4,
	Gpio_out=	1<<5,
};

void	gpioreserve(ulong);
void	gpioconfig(ulong, ulong);
ulong	gpioget(ulong);
void	gpioset(ulong, ulong);
void	gpiorelease(ulong);

/*
 * used by ../port/devi2c.c and iic.c
 */
struct I2Cdev {
	int	addr;
	int	salen;	/* length in bytes of subaddress, if used; 0 otherwise */
	int	tenbit;	/* 10-bit addresses */
};

long	i2crecv(I2Cdev*, void*, long, ulong);
long	i2csend(I2Cdev*, void*, long, ulong);
void	i2csetup(int);

/*
 * PCI support code.
 */
enum {					/* type 0 and type 1 pre-defined header */
	PciVID		= 0x00,		/* vendor ID */
	PciDID		= 0x02,		/* device ID */
	PciPCR		= 0x04,		/* command */
	PciPSR		= 0x06,		/* status */
	PciRID		= 0x08,		/* revision ID */
	PciCCRp		= 0x09,		/* programming interface class code */
	PciCCRu		= 0x0A,		/* sub-class code */
	PciCCRb		= 0x0B,		/* base class code */
	PciCLS		= 0x0C,		/* cache line size */
	PciLTR		= 0x0D,		/* latency timer */
	PciHDT		= 0x0E,		/* header type */
	PciBST		= 0x0F,		/* BIST */

	PciBAR0		= 0x10,		/* base address */
	PciBAR1		= 0x14,

	PciINTL		= 0x3C,		/* interrupt line */
	PciINTP		= 0x3D,		/* interrupt pin */
};

enum {					/* type 0 pre-defined header */
	PciBAR2		= 0x18,
	PciBAR3		= 0x1C,
	PciBAR4		= 0x20,
	PciBAR5		= 0x24,
	PciCIS		= 0x28,		/* cardbus CIS pointer */
	PciSVID		= 0x2C,		/* subsystem vendor ID */
	PciSID		= 0x2E,		/* cardbus CIS pointer */
	PciEBAR0	= 0x30,		/* expansion ROM base address */
	PciMGNT		= 0x3E,		/* burst period length */
	PciMLT		= 0x3F,		/* maximum latency between bursts */
};

enum {					/* type 1 pre-defined header */
	PciPBN		= 0x18,		/* primary bus number */
	PciSBN		= 0x19,		/* secondary bus number */
	PciUBN		= 0x1A,		/* subordinate bus number */
	PciSLTR		= 0x1B,		/* secondary latency timer */
	PciIBR		= 0x1C,		/* I/O base */
	PciILR		= 0x1D,		/* I/O limit */
	PciSPSR		= 0x1E,		/* secondary status */
	PciMBR		= 0x20,		/* memory base */
	PciMLR		= 0x22,		/* memory limit */
	PciPMBR		= 0x24,		/* prefetchable memory base */
	PciPMLR		= 0x26,		/* prefetchable memory limit */
	PciPUBR		= 0x28,		/* prefetchable base upper 32 bits */
	PciPULR		= 0x2C,		/* prefetchable limit upper 32 bits */
	PciIUBR		= 0x30,		/* I/O base upper 16 bits */
	PciIULR		= 0x32,		/* I/O limit upper 16 bits */
	PciEBAR1	= 0x28,		/* expansion ROM base address */
	PciBCR		= 0x3E,		/* bridge control register */
};

enum {					/* type 2 pre-defined header */
	PciCBExCA	= 0x10,
	PciCBSPSR	= 0x16,
	PciCBPBN	= 0x18,		/* primary bus number */
	PciCBSBN	= 0x19,		/* secondary bus number */
	PciCBUBN	= 0x1A,		/* subordinate bus number */
	PciCBSLTR	= 0x1B,		/* secondary latency timer */
	PciCBMBR0	= 0x1C,
	PciCBMLR0	= 0x20,
	PciCBMBR1	= 0x24,
	PciCBMLR1	= 0x28,
	PciCBIBR0	= 0x2C,		/* I/O base */
	PciCBILR0	= 0x30,		/* I/O limit */
	PciCBIBR1	= 0x34,		/* I/O base */
	PciCBILR1	= 0x38,		/* I/O limit */
	PciCBSVID	= 0x40,		/* subsystem vendor ID */
	PciCBSID	= 0x42,		/* subsystem ID */
	PciCBLMBAR	= 0x44,		/* legacy mode base address */
};

typedef struct Pcisiz Pcisiz;
struct Pcisiz
{
	Pcidev*	dev;
	int	siz;
	int	bar;
};

typedef struct Pcidev Pcidev;
struct Pcidev
{
	int	tbdf;			/* type+bus+device+function */
	ushort	vid;			/* vendor ID */
	ushort	did;			/* device ID */

	uchar	rid;
	uchar	ccrp;
	uchar	ccru;
	uchar	ccrb;

	struct {
		ulong	bar;		/* base address */
		int	size;
	} mem[6];

	struct {
		ulong	bar;	
		int	size;
	} rom;
	uchar	intl;			/* interrupt line */

	Pcidev*	list;
	Pcidev*	link;			/* next device on this bno */

	Pcidev*	bridge;			/* down a bus */
	struct {
		ulong	bar;
		int	size;
	} ioa, mema;
	ulong	pcr;
};

#define PCIWINDOW	0x80000000
#define PCIWADDR(va)	(PADDR(va)+PCIWINDOW)