|
|
Plan 9 NeXT
/*
* All the goo for PC ethernet cards.
*/
typedef struct Card Card;
typedef struct RingBuf RingBuf;
typedef struct Type Type;
typedef struct Ctlr Ctlr;
/*
* Hardware interface.
*/
struct Card {
ISAConf;
int (*reset)(Ctlr*);
void (*init)(Ctlr*);
void (*attach)(Ctlr*);
void (*mode)(Ctlr*, int);
void *(*read)(Ctlr*, void*, ulong, ulong);
void *(*write)(Ctlr*, ulong, void*, ulong);
void (*receive)(Ctlr*);
void (*transmit)(Ctlr*);
void (*intr)(Ctlr*);
void (*watch)(Ctlr*);
void (*overflow)(Ctlr*);
uchar bit16; /* true if a 16 bit interface */
uchar ram; /* true if card has shared memory */
ulong dp8390; /* I/O address of 8390 (if any) */
ulong data; /* I/O data port if no shared memory */
uchar nxtpkt; /* software bndry */
uchar tstart; /* 8390 ring addresses */
uchar pstart;
uchar pstop;
};
/*
* Software ring buffer.
*/
struct RingBuf {
uchar owner;
uchar busy; /* unused */
ushort len;
uchar pkt[sizeof(Etherpkt)];
};
enum {
Host = 0, /* buffer owned by host */
Interface = 1, /* buffer owned by card */
Nrb = 16, /* default number of receive buffers */
Ntb = 4, /* default number of transmit buffers */
};
/*
* One per ethernet packet type.
*/
struct Type {
QLock;
Netprot; /* stat info */
int type; /* ethernet type */
int prom; /* promiscuous mode */
int filter; /* address filter */
uchar ea[6]; /* ethernet address */
Queue *q;
int inuse;
Ctlr *ctlr;
Rendez cr; /* rendezvous for close */
Type *clist; /* close list */
};
enum {
NType = 9, /* types/card */
Eaddrlen = 6, /* Ethernet address length */
};
/*
* Software controller.
*/
struct Ctlr {
QLock;
int ctlrno;
Ctlr *next;
Card card; /* hardware info */
int present;
int debug;
ushort nrb; /* number of software receive buffers */
ushort ntb; /* number of software transmit buffers */
RingBuf *rb; /* software receive buffers */
RingBuf *tb; /* software transmit buffers */
uchar ea[6]; /* ethernet address */
uchar ba[6]; /* broadcast address */
Rendez rr; /* rendezvous for a receive buffer */
ushort rh; /* first receive buffer belonging to host */
ushort ri; /* first receive buffer belonging to card */
Rendez tr; /* rendezvous for a transmit buffer */
QLock tlock; /* semaphore on th */
ushort th; /* first transmit buffer belonging to host */
ushort ti; /* first transmit buffer belonging to card */
int tbusy; /* transmitter is busy */
Type type[NType];
int all; /* number of channels listening to all packets */
Type *clist; /* channels waiting to close */
Lock clock; /* lock for clist */
int prom; /* number of promiscuous channels */
int filter; /* number of address filters */
int early; /* true if ra set */
uchar ra[8]; /* received address if filtering */
int kproc; /* true if kproc started */
char name[NAMELEN]; /* name of kproc */
Network net;
Queue lbq; /* software loopback packet queue */
int inpackets;
int outpackets;
int crcs; /* input crc errors */
int oerrs; /* output errors */
int frames; /* framing errors */
int overflows; /* packet overflows */
int buffs; /* buffering errors */
};
#define NEXT(x, l) (((x)+1)%(l))
#define HOWMANY(x, y) (((x)+((y)-1))/(y))
#define ROUNDUP(x, y) (HOWMANY((x), (y))*(y))
/*
* The Western Digital 8003 and 8013 series, the NE2000
* and the 3Com 503 all use the DP8390 Network Interface
* Chip.
*/
extern void dp8390reset(Ctlr*);
extern void dp8390attach(Ctlr*);
extern void dp8390mode(Ctlr*, int);
extern void dp8390getea(Ctlr*);
extern void dp8390setea(Ctlr*);
extern void *dp8390read(Ctlr*, void*, ulong, ulong);
extern void *dp8390write(Ctlr*, ulong, void*, ulong);
extern void dp8390receive(Ctlr*);
extern void dp8390transmit(Ctlr*);
extern void dp8390intr(Ctlr*);
extern void dp8390watch(Ctlr*);
extern void dp8390overflow(Ctlr*);
extern void dp8390debug(Ctlr*);
/*
* The DP8390 needs some time between successive
* chip selects, so we need special I/O routines.
* See l.s.
* These routines only need to be used for accessing
* the chip registers. Data I/O, either through
* remote DMA or shared memory, can use the normal
* routines.
*/
extern int dp8390inb(ulong);
extern void dp8390outb(ulong, uchar);
enum {
Dp8390BufSz = 256, /* hardware ring buffer size */
};
extern void addethercard(char*, int (*)(Ctlr*));
extern int eaddrmatch(Ctlr*, uchar*);
extern int parseether(void*, void*);
This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.