NetBSD-5.0.2/sys/dev/ic/rrunnervar.h

/*	$NetBSD: rrunnervar.h,v 1.13 2008/04/28 20:23:51 martin Exp $	*/

/* Copyright (c) 1997, 1998 The NetBSD Foundation, Inc.
 * All rights reserved.
 *
 * This code contributed to The NetBSD Foundation by Kevin M. Lahey
 * of the Numerical Aerospace Simulation Facility, NASA Ames Research
 * Center.
 *
 * Partially based on a HIPPI driver written by Essential Communications
 * Corporation.  Thanks to Jason Thorpe, Matt Jacob, and Fred Templin
 * for invaluable advice and encouragement!
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions and the following disclaimer.
 * 2. Redistributions in binary form must reproduce the above copyright
 *    notice, this list of conditions and the following disclaimer in the
 *    documentation and/or other materials provided with the distribution.
 *
 * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
 * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
 * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
 * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 * POSSIBILITY OF SUCH DAMAGE.
 */

/* RoadRunner software status per interface */

struct rr_tuning {

	/* Performance tuning registers: */

	u_int32_t		rt_mode_and_status;
	u_int32_t		rt_conn_retry_count;
	u_int32_t		rt_conn_retry_timer;
	u_int32_t		rt_conn_timeout;
	u_int32_t		rt_stats_timer;
	u_int32_t		rt_interrupt_timer;
	u_int32_t		rt_tx_timeout;
	u_int32_t		rt_rx_timeout;

	/* DMA performance tuning registers: */

	u_int32_t		rt_pci_state;
	u_int32_t		rt_dma_write_state;
	u_int32_t		rt_dma_read_state;
	u_int32_t		rt_driver_param;
};



struct rr_eeprom {
	u_int32_t	ifr_offset;	/* initial offset in bytes */
	u_int32_t	ifr_length;	/* length in bytes to write */
	u_int32_t	*ifr_buffer;	/* data to be written */
};

#define EIOCGTUNE   1	/* retrieve tuning */
#define EIOCSTUNE   2	/* set tuning */
#define EIOCGEEPROM 3	/* get eeprom */
#define EIOCSEEPROM 4	/* set eeprom */
#define EIOCGSTATS  5	/* get statistics */
#define EIOCRESET   6   /* reset the card */

#ifdef _KERNEL

/* Per-ring information for the SNAP (network) receive ring */

struct esh_snap_ring_ctl {
	bus_dmamap_t ec_dma[RR_MAX_DESCR];
	struct mbuf *ec_m[RR_MAX_DESCR];
	struct mbuf *ec_cur_pkt;	/* current packet being processed */
	struct mbuf *ec_cur_mbuf;	/* current mbuf being processed */
	int ec_error;			/* encountered error? */
	u_int16_t ec_producer;		/* latest buffer driver produced */
	u_int16_t ec_consumer;		/* latest buffer runcode consumed */
	struct rr_descr *ec_descr;	/* array of descriptors for ring */
};

TAILQ_HEAD(esh_dmainfo_list, esh_dmainfo);

struct esh_dmainfo {
	u_int32_t ed_flags;
#define ESH_DI_BUSY	0x1
#define ESH_DI_READING	0x2
	bus_dmamap_t ed_dma;
	struct buf *ed_buf;
	int ed_read_len;
	int ed_error;
	TAILQ_ENTRY(esh_dmainfo) ed_list;
};

struct esh_send_ring_ctl {
	bus_dmamap_t ec_dma;		/* dmamap for data to transmit */
	int ec_offset;			/* offset in dmamap to send next */
	size_t ec_len;			/* total length of current buf */
	struct mbuf *ec_cur_mbuf;	/* current mbuf being processed */
	struct buf *ec_cur_buf;		/* current buf being processed */
	struct esh_dmainfo *ec_cur_dmainfo;
					/* current dmainfo being processed */
	struct bufq_state *ec_buf_queue;/* queue of bufs to send */
	int ec_error;			/* encountered error? */
	u_int16_t ec_producer;		/* latest buffer driver produced */
	u_int16_t ec_consumer;		/* latest buffer runcode consumed */
	struct rr_descr *ec_descr;	/* array of descriptors for ring */
	struct esh_dmainfo_list ec_di_queue;
};

struct esh_fp_ring_ctl {
	struct esh_dmainfo *ec_dmainfo[RR_MAX_DESCR];
	struct esh_dmainfo *ec_cur_dmainfo;
	int ec_offset;			/* offset of current buf */
	int ec_error;			/* encountered error? */
	int ec_seen_end;		/* seen the end of the buffer? */
	int ec_dmainfo_count;		/* dmainfo buffers in use count */
	u_int16_t ec_producer;		/* latest buffer driver produced */
	u_int16_t ec_consumer;		/* latest buffer runcode consumed */
	u_int32_t ec_read_len;		/* length of packet being read in */
	struct rr_descr *ec_descr;	/* array of descriptors for ring */
	struct esh_dmainfo_list ec_queue;
	u_int ec_ulp;			/* ULP for this ring */
	int ec_index;			/* index into list of active rings */
	bus_dmamap_t ec_dma;
	bus_dma_segment_t ec_dmaseg;
};


struct esh_softc {
	struct device		sc_dev;
	struct ifnet		sc_if;
	struct ifmedia		sc_media;

	volatile int		sc_flags;
#define ESH_FL_INITIALIZING	0x001
#define ESH_FL_INITIALIZED	0x002
#define ESH_FL_RUNCODE_UP	0x004
#define ESH_FL_LINK_UP		0x008
#define ESH_FL_SNAP_RING_UP	0x010
#define ESH_FL_FP_RING_UP	0x020
#define ESH_FL_EEPROM_BUSY	0x040
#define ESH_FL_FP_OPEN		0x080
#define ESH_FL_CRASHED		0x100
#define ESH_FL_CLOSING_SNAP	0x200

	void			*sc_ih;

	bus_space_tag_t		sc_iot;	     /* bus cookie      */
	bus_space_handle_t	sc_ioh;      /* bus i/o handle  */

	bus_dma_tag_t		sc_dmat;     /* dma tag */

	bus_dma_segment_t	sc_dmaseg;   /* segment holding the various
					        data structures in host memory
					        that are DMA'ed to the NIC */
	bus_dmamap_t		sc_dma;	     /* dma map for the segment */
	char 		        *sc_dma_addr; /* address in kernel of DMA mem */
	bus_size_t		sc_dma_size; /* size of dma-able region */

	u_int8_t	(*sc_bist_read)(struct esh_softc *);
	void		(*sc_bist_write)(struct esh_softc *, u_int8_t);

	/*
	 * Definitions for the various driver structures that sit in host
	 * memory and are read by the NIC via DMA:
	 */

	struct rr_gen_info	*sc_gen_info;	/* gen info block pointer */
	bus_addr_t		sc_gen_info_dma;

	struct rr_ring_ctl	*sc_recv_ring_table;
	bus_addr_t		sc_recv_ring_table_dma;

	struct rr_event		*sc_event_ring;
	bus_addr_t		sc_event_ring_dma;

	struct rr_descr		*sc_send_ring;
	struct rr2_descr	*sc2_send_ring;
	bus_addr_t		sc_send_ring_dma;

	struct rr_descr		*sc_snap_recv_ring;
	struct rr2_descr	*sc2_snap_recv_ring;
	bus_addr_t		sc_snap_recv_ring_dma;

	/*
	 * Control structures for the various rings that we definitely
	 * know we want to keep track of.
	 */

	struct esh_send_ring_ctl
				sc_send;
	struct esh_snap_ring_ctl
				sc_snap_recv;
	struct esh_fp_ring_ctl	*sc_fp_recv[RR_ULP_COUNT];
	struct esh_fp_ring_ctl	*sc_fp_recv_index[RR_MAX_RECV_RING];
	int			sc_event_consumer;
	int			sc_event_producer;
	int			sc_cmd_consumer;
	int			sc_cmd_producer;

	/*
	 * Various maintainance values we need
	 */

	int			sc_watchdog;

	/*
	 * Various hardware parameters we need to keep track of.
	 */

	u_int32_t		sc_sram_size;
	u_int32_t		sc_runcode_start;
	u_int32_t		sc_runcode_version;
	u_int32_t		sc_version; /* interface of runcode (1 or 2) */
	u_int16_t		sc_options; /* options in current RunCode */
	u_int			sc_max_rings;

	u_int32_t		sc_pci_latency;
	u_int32_t		sc_pci_lat_gnt;
	u_int32_t		sc_pci_cache_line;

	/* ULA assigned to hardware */

	u_int8_t		sc_ula[6];

	/* Tuning parameters */

	struct rr_tuning	sc_tune;

	/* Measure of how ugly this is. */

	u_int32_t		sc_misaligned_bufs;
	u_int32_t		sc_bad_lens;

	struct esh_dmainfo_list sc_dmainfo_freelist;
	u_int			sc_dmainfo_freelist_count;
	u_int			sc_fp_rings;
};

void	eshconfig(struct esh_softc *);
int	eshintr(void *);
#endif /* _KERNEL */

/* Define a few constants for future use */

#define ESH_MAX_NSEGS			512     /* room for 2MB of data */
#define ESH_STATS_TIMER_DEFAULT		1030900
	/* 1000000 usecs / 0.97 usecs/tick */

#define NEXT_EVENT(i)  (((i) + 1) & (RR_EVENT_RING_SIZE - 1))
#define NEXT_SEND(i)  (((i) + 1) & (RR_SEND_RING_SIZE - 1))
#define NEXT_RECV(i)  (((i) + 1) & (RR_SNAP_RECV_RING_SIZE - 1))

#define PREV_SEND(i)  (((i) + RR_SEND_RING_SIZE - 1) & (RR_SEND_RING_SIZE - 1))
#define PREV_RECV(i)  \
	(((i) + RR_SNAP_RECV_RING_SIZE - 1) & (RR_SNAP_RECV_RING_SIZE - 1))