#ifndef _SYS_GLDPRIV_H
#define _SYS_GLDPRIV_H
#ifdef __cplusplus
extern "C" {
#endif
#ifdef DEBUG
#define GLD_DEBUG 1
#endif
#define GLD_VERSION_200 0x200
#define GLD_VERSION GLD_VERSION_200
#define GLD_VERSION_STRING "v2"
#define GLD_OPT_NO_IPQ 0x00000001
#define GLD_OPT_NO_FASTPATH 0x00000002
#define GLD_OPT_NO_ETHRXSNAP 0x00000008
#define GLDOPT_FAST_RECV 0x40
#define GLDOPT_CANONICAL_ADDR 0x08
#define GLD_VLAN_SCALE 1000
#define GLD_MAX_PPA (GLD_VLAN_SCALE-1)
#define GLD_USE_STYLE2 0
#define GLD_MIN_STYLE1_MINOR 1
#define GLD_MAX_STYLE1_MINOR (GLD_MAX_PPA+1)
#define GLD_STYLE1_MINOR_TO_PPA(minor) (minor - 1)
#define GLD_STYLE1_PPA_TO_MINOR(ppa) (ppa + 1)
#define GLD_MIN_CLONE_MINOR (GLD_MAX_STYLE1_MINOR+1)
#define GLD_MAX_CLONE_MINOR 0x3ffff
#define GLD_MAC_READY 0x0001
#define GLD_INTR_READY 0x0001
#define GLD_INTR_WAIT 0x0002
#define GLD_LOCK_INITED 0x0004
#define GLD_UNREGISTERED 0x0008
#define GLD_MAX_ADDRLEN 32
#define GLD_MAX_MULTICAST 64
typedef struct gld_multicast_addr {
int gldm_refcnt;
unsigned char gldm_addr[GLD_MAX_ADDRLEN];
} gld_mcast_t;
#define GLD_RAW 0x0001
#define GLD_FAST 0x0002
#define GLD_PROM_PHYS 0x0004
#define GLD_PROM_SAP 0x0008
#define GLD_PROM_MULT 0x0010
#define GLD_STR_CLOSING 0x0020
typedef struct gld {
struct gld *gld_next, *gld_prev;
caddr_t gld_dummy1;
int32_t gld_state;
int32_t gld_style;
int32_t gld_minor;
int32_t gld_type;
int32_t gld_sap;
int32_t gld_flags;
int32_t gld_multicnt;
gld_mcast_t **gld_mcast;
queue_t *gld_qptr;
caddr_t gld_dummy2;
caddr_t gld_dummy3;
struct gld_mac_info *gld_mac_info;
caddr_t gld_dummy4;
struct glddevice *gld_device;
volatile boolean_t gld_xwait;
volatile boolean_t gld_sched_ran;
volatile boolean_t gld_in_unbind;
volatile uint32_t gld_wput_count;
volatile boolean_t gld_in_wsrv;
boolean_t gld_ethertype;
uint32_t gld_notifications;
uint32_t gld_upri;
void *gld_vlan;
int (*gld_send)();
} gld_t;
typedef struct glddevice {
struct glddevice *gld_next, *gld_prev;
int gld_ndevice;
gld_mac_info_t *gld_mac_next, *gld_mac_prev;
gld_t *gld_str_next, *gld_str_prev;
char gld_name[16];
kmutex_t gld_devlock;
int gld_nextminor;
int gld_major;
int gld_multisize;
int gld_type;
int gld_minsdu;
int gld_maxsdu;
int gld_addrlen;
int gld_saplen;
unsigned char *gld_broadcast;
int gld_styles;
} glddev_t;
typedef struct pktinfo {
uint_t isBroadcast:1;
uint_t isMulticast:1;
uint_t isLooped:1;
uint_t isForMe:1;
uint_t isLLC:1;
uint_t user_pri:3;
uint_t cfi:1;
uint_t vid:12;
uint_t wasAccepted:1;
uint_t nosource:1;
uint_t isTagged:1;
uint_t macLen;
uint_t hdrLen;
uint_t pktLen;
uchar_t dhost[GLD_MAX_ADDRLEN];
uchar_t shost[GLD_MAX_ADDRLEN];
uint_t ethertype;
} pktinfo_t;
typedef enum packet_flag {
GLD_RXQUICK,
GLD_RXLOOP,
GLD_RX,
GLD_TX
} packet_flag_t;
typedef struct {
uint_t mac_type;
uint_t mtu_size;
int hdr_size;
int (*interpreter)(gld_mac_info_t *, mblk_t *, pktinfo_t *,
packet_flag_t);
mblk_t *(*mkfastpath)(gld_t *, mblk_t *);
mblk_t *(*mkunitdata)(gld_t *, mblk_t *);
void (*init)(gld_mac_info_t *);
void (*uninit)(gld_mac_info_t *);
char *mac_string;
} gld_interface_t;
typedef union media_kstats {
struct dot3kstat {
kstat_named_t first_coll;
kstat_named_t multi_coll;
kstat_named_t sqe_error;
kstat_named_t mac_xmt_error;
kstat_named_t frame_too_long;
kstat_named_t mac_rcv_error;
} dot3;
struct dot5kstat {
kstat_named_t ace_error;
kstat_named_t internal_error;
kstat_named_t lost_frame_error;
kstat_named_t frame_copied_error;
kstat_named_t token_error;
kstat_named_t freq_error;
} dot5;
struct fddikstat {
kstat_named_t mac_error;
kstat_named_t mac_lost;
kstat_named_t mac_token;
kstat_named_t mac_tvx_expired;
kstat_named_t mac_late;
kstat_named_t mac_ring_op;
} fddi;
} media_kstats_t;
struct gldkstats {
kstat_named_t glds_pktxmt;
kstat_named_t glds_pktrcv;
kstat_named_t glds_errxmt;
kstat_named_t glds_errrcv;
kstat_named_t glds_collisions;
kstat_named_t glds_bytexmt;
kstat_named_t glds_bytercv;
kstat_named_t glds_multixmt;
kstat_named_t glds_multircv;
kstat_named_t glds_brdcstxmt;
kstat_named_t glds_brdcstrcv;
kstat_named_t glds_unknowns;
kstat_named_t glds_blocked;
kstat_named_t glds_excoll;
kstat_named_t glds_defer;
kstat_named_t glds_frame;
kstat_named_t glds_crc;
kstat_named_t glds_overflow;
kstat_named_t glds_underflow;
kstat_named_t glds_short;
kstat_named_t glds_missed;
kstat_named_t glds_xmtlatecoll;
kstat_named_t glds_nocarrier;
kstat_named_t glds_noxmtbuf;
kstat_named_t glds_norcvbuf;
kstat_named_t glds_xmtbadinterp;
kstat_named_t glds_rcvbadinterp;
kstat_named_t glds_intr;
kstat_named_t glds_xmtretry;
kstat_named_t glds_pktxmt64;
kstat_named_t glds_pktrcv64;
kstat_named_t glds_bytexmt64;
kstat_named_t glds_bytercv64;
kstat_named_t glds_speed;
kstat_named_t glds_duplex;
kstat_named_t glds_media;
kstat_named_t glds_prom;
media_kstats_t glds_media_specific;
};
typedef struct gld_mac_pvt gld_mac_pvt_t;
typedef struct gld_vlan {
struct gld_vlan *gldv_next, *gldv_prev;
uint32_t gldv_id;
uint32_t gldv_ptag;
int gldv_nstreams;
gld_mac_info_t *gldv_mac;
queue_t *gldv_ipq;
queue_t *gldv_ipv6q;
struct gld *gldv_str_next;
struct gld *gldv_str_prev;
kstat_t *gldv_kstatp;
struct gld_stats *gldv_stats;
uint_t gldv_nprom;
uint_t gldv_nvlan_sap;
} gld_vlan_t;
#define VLAN_HASHSZ 23
struct gld_mac_pvt {
gld_interface_t *interfacep;
kmutex_t datalock;
caddr_t data;
gld_vlan_t *vlan_hash[VLAN_HASHSZ];
struct gld *last_sched;
struct glddevice *major_dev;
int nvlan;
int nprom;
int nprom_multi;
gld_mcast_t *mcast_table;
unsigned char *curr_macaddr;
kstat_t *kstatp;
struct gld_stats *statistics;
int rde_enabled;
int rde_str_indicator_ste;
int rde_timeout;
uint32_t notifications;
boolean_t started;
};
#define GLDE_OK (-1)
#define GLDE_RETRY 0x1002
#define GLD_WPUT 0
#define GLD_WSRV 1
#define GLD_MAX_802_SAP 0xff
#define GLDTRACE 0x0001
#define GLDERRS 0x0002
#define GLDRECV 0x0004
#define GLDSEND 0x0008
#define GLDPROT 0x0010
#define GLDNOBR 0x0020
#define GLDETRACE 0x0040
#define GLDRDE 0x0080
#define GLDM_LOCK_INIT(macinfo) \
rw_init(&(macinfo)->gldm_lock.gldl_rw_lock, NULL, \
RW_DRIVER, (macinfo)->gldm_cookie); \
(macinfo)->gldm_GLD_flags |= GLD_LOCK_INITED
#define GLDM_LOCK_INITED(macinfo) \
((macinfo)->gldm_GLD_flags & GLD_LOCK_INITED)
#define GLDM_LOCK_DESTROY(macinfo) \
if ((macinfo)->gldm_GLD_flags & GLD_LOCK_INITED) { \
rw_destroy(&(macinfo)->gldm_lock.gldl_rw_lock); \
(macinfo)->gldm_GLD_flags &= ~GLD_LOCK_INITED; \
}
#define GLDM_LOCK(macinfo, rw) \
rw_enter(&(macinfo)->gldm_lock.gldl_rw_lock, (rw))
#define GLDM_UNLOCK(macinfo) \
rw_exit(&(macinfo)->gldm_lock.gldl_rw_lock)
#define GLDM_TRYLOCK(macinfo, rw) \
rw_tryenter(&(macinfo)->gldm_lock.gldl_rw_lock, (rw))
#define GLDM_LOCK_HELD(macinfo) \
rw_lock_held(&(macinfo)->gldm_lock.gldl_rw_lock)
#define GLDM_LOCK_HELD_WRITE(macinfo) \
rw_write_held(&(macinfo)->gldm_lock.gldl_rw_lock)
#define mac_eq(a, b, l) (bcmp((caddr_t)(a), (caddr_t)(b), (l)) == 0)
#define mac_copy(a, b, l) (bcopy((caddr_t)(a), (caddr_t)(b), (l)))
#define cmac_copy(a, b, l, macinfo) { \
if ((macinfo)->gldm_options & GLDOPT_CANONICAL_ADDR) \
gld_bitrevcopy((caddr_t)(a), (caddr_t)(b), (l)); \
else \
mac_copy((a), (b), (l)); \
}
#if (_ALIGNMENT_REQUIRED == 0)
#define REF_HOST_USHORT(lvalue) (lvalue)
#define REF_NET_USHORT(lvalue) (ntohs(lvalue))
#define SET_NET_USHORT(lvalue, val) ((lvalue) = htons(val))
#else
#define REF_NET_USHORT(lvalue) \
((ushort_t)((((uchar_t *)(&(lvalue)))[0]<<8) | \
((uchar_t *)(&(lvalue)))[1]))
#define SET_NET_USHORT(lvalue, val) { \
((uchar_t *)(&(lvalue)))[0] = (uchar_t)((val)>>8); \
((uchar_t *)(&(lvalue)))[1] = (uchar_t)(val); \
}
#if defined(_LITTLE_ENDIAN)
#define REF_HOST_USHORT(lvalue) \
((ushort_t)((((uchar_t *)(&(lvalue)))[1]<<8) | \
((uchar_t *)(&(lvalue)))[0]))
#elif defined(_BIG_ENDIAN)
#define REF_HOST_USHORT(lvalue) \
((ushort_t)((((uchar_t *)(&(lvalue)))[0]<<8) | \
((uchar_t *)(&(lvalue)))[1]))
#else
#error "what endian is this machine?"
#endif
#endif
struct rde_pdu {
uchar_t rde_ver;
uchar_t rde_ptype;
uchar_t rde_target_mac[6];
uchar_t rde_orig_mac[6];
uchar_t rde_target_sap;
uchar_t rde_orig_sap;
};
#define LSAP_RDE 0xa6
#define RDE_RQC 0x01
#define RDE_RQR 0x02
#define RDE_RS 0x03
#define MAX_RDFLDS 14
struct gld_ri {
#if defined(_BIT_FIELDS_LTOH)
uchar_t len:5;
uchar_t rt:3;
uchar_t res:4;
uchar_t mtu:3;
uchar_t dir:1;
struct tr_rd {
ushort_t bridge:4;
ushort_t ring:12;
} rd[MAX_RDFLDS];
#elif defined(_BIT_FIELDS_HTOL)
uchar_t rt:3;
uchar_t len:5;
uchar_t dir:1;
uchar_t mtu:3;
uchar_t res:4;
struct tr_rd {
ushort_t ring:12;
ushort_t bridge:4;
} rd[MAX_RDFLDS];
#else
#error "which way do bit fields get allocated?"
#endif
};
#define RT_SRF 0x0
#define RT_ARE 0x4
#define RT_STE 0x6
#define RT_MTU_MAX 0x7
struct srtab {
struct srtab *sr_next;
uchar_t sr_mac[6];
struct gld_ri sr_ri;
clock_t sr_timer;
};
#define SR_HASH_SIZE 256
struct gld_dlsap {
unsigned char glda_addr[ETHERADDRL];
unsigned short glda_sap;
};
#define DLSAP(p, offset) ((struct gld_dlsap *)((caddr_t)(p)+offset))
typedef uchar_t mac_addr_t[ETHERADDRL];
struct llc_snap_hdr {
uchar_t d_lsap;
uchar_t s_lsap;
uchar_t control;
uchar_t org[3];
ushort_t type;
};
#define LLC_HDR1_LEN 3
#define LLC_SNAP_HDR_LEN 8
#define LSAP_SNAP 0xaa
#define CNTL_LLC_UI 0x03
struct fddi_mac_frm {
uchar_t fddi_fc;
mac_addr_t fddi_dhost;
mac_addr_t fddi_shost;
};
struct tr_mac_frm_nori {
uchar_t tr_ac;
uchar_t tr_fc;
mac_addr_t tr_dhost;
mac_addr_t tr_shost;
};
struct tr_mac_frm {
uchar_t tr_ac;
uchar_t tr_fc;
mac_addr_t tr_dhost;
mac_addr_t tr_shost;
struct gld_ri tr_ri;
};
#define GLD_SAVE_MBLK_VTAG(mp, vtag) (DB_TCI(mp) = GLD_VTAG_TCI(vtag))
#define GLD_CLEAR_MBLK_VTAG(mp) GLD_SAVE_MBLK_VTAG(mp, 0)
#define GLD_GET_MBLK_VTAG(mp) GLD_TCI2VTAG(DB_TCI(mp))
int gld_interpret_ether(gld_mac_info_t *, mblk_t *, pktinfo_t *, packet_flag_t);
int gld_interpret_fddi(gld_mac_info_t *, mblk_t *, pktinfo_t *, packet_flag_t);
int gld_interpret_tr(gld_mac_info_t *, mblk_t *, pktinfo_t *, packet_flag_t);
int gld_interpret_ib(gld_mac_info_t *, mblk_t *, pktinfo_t *, packet_flag_t);
mblk_t *gld_fastpath_ether(gld_t *, mblk_t *);
mblk_t *gld_fastpath_fddi(gld_t *, mblk_t *);
mblk_t *gld_fastpath_tr(gld_t *, mblk_t *);
mblk_t *gld_fastpath_ib(gld_t *, mblk_t *);
mblk_t *gld_insert_vtag_ether(mblk_t *, uint32_t);
mblk_t *gld_unitdata_ether(gld_t *, mblk_t *);
mblk_t *gld_unitdata_fddi(gld_t *, mblk_t *);
mblk_t *gld_unitdata_tr(gld_t *, mblk_t *);
mblk_t *gld_unitdata_ib(gld_t *, mblk_t *);
void gld_init_ether(gld_mac_info_t *);
void gld_init_fddi(gld_mac_info_t *);
void gld_init_tr(gld_mac_info_t *);
void gld_init_ib(gld_mac_info_t *);
void gld_uninit_ether(gld_mac_info_t *);
void gld_uninit_fddi(gld_mac_info_t *);
void gld_uninit_tr(gld_mac_info_t *);
void gld_uninit_ib(gld_mac_info_t *);
#ifdef __cplusplus
}
#endif
#endif