/************************************************************************** Copyright (c) 2007-2008, Chelsio Inc. All rights reserved. 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. Neither the name of the Chelsio Corporation nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS 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 COPYRIGHT OWNER 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. $FreeBSD$ ***************************************************************************/ #include #include #include #include #include #include #include #include #ifdef CONFIG_DEFINED #include #include #else #include #include #endif #ifndef _CXGB_OSDEP_H_ #define _CXGB_OSDEP_H_ typedef struct adapter adapter_t; struct sge_rspq; enum { TP_TMR_RES = 200, /* TP timer resolution in usec */ MAX_NPORTS = 4, /* max # of ports */ TP_SRAM_OFFSET = 4096, /* TP SRAM content offset in eeprom */ TP_SRAM_LEN = 2112, /* TP SRAM content offset in eeprom */ }; struct t3_mbuf_hdr { struct mbuf *mh_head; struct mbuf *mh_tail; }; #ifndef PANIC_IF #define PANIC_IF(exp) do { \ if (exp) \ panic("BUG: %s", #exp); \ } while (0) #endif #define m_get_priority(m) ((uintptr_t)(m)->m_pkthdr.rcvif) #define m_set_priority(m, pri) ((m)->m_pkthdr.rcvif = (struct ifnet *)((uintptr_t)pri)) #define m_set_sgl(m, sgl) ((m)->m_pkthdr.header = (sgl)) #define m_get_sgl(m) ((bus_dma_segment_t *)(m)->m_pkthdr.header) #define m_set_sgllen(m, len) ((m)->m_pkthdr.ether_vtag = len) #define m_get_sgllen(m) ((m)->m_pkthdr.ether_vtag) /* * XXX FIXME */ #define m_set_toep(m, a) ((m)->m_pkthdr.header = (a)) #define m_get_toep(m) ((m)->m_pkthdr.header) #define m_set_handler(m, handler) ((m)->m_pkthdr.header = (handler)) #define m_set_socket(m, a) ((m)->m_pkthdr.header = (a)) #define m_get_socket(m) ((m)->m_pkthdr.header) #define KTR_CXGB KTR_SPARE2 void cxgb_log_tcb(struct adapter *sc, unsigned int tid); #define MT_DONTFREE 128 #if __FreeBSD_version > 700030 #define INTR_FILTERS #define FIRMWARE_LATEST #endif #if ((__FreeBSD_version > 602103) && (__FreeBSD_version < 700000)) #define FIRMWARE_LATEST #endif #if __FreeBSD_version > 700000 #define MSI_SUPPORTED #define TSO_SUPPORTED #define VLAN_SUPPORTED #define TASKQUEUE_CURRENT #else #define if_name(ifp) (ifp)->if_xname #define M_SANITY(m, n) #endif #define __read_mostly __attribute__((__section__(".data.read_mostly"))) /* * Workaround for weird Chelsio issue */ #if __FreeBSD_version > 700029 #define PRIV_SUPPORTED #endif #define CXGB_TX_CLEANUP_THRESHOLD 32 #ifdef DEBUG_PRINT #define DPRINTF printf #else #define DPRINTF(...) #endif #define TX_MAX_SIZE (1 << 16) /* 64KB */ #define TX_MAX_SEGS 36 /* maximum supported by card */ #define TX_MAX_DESC 4 /* max descriptors per packet */ #define TX_START_MIN_DESC (TX_MAX_DESC << 2) #define TX_START_MAX_DESC (TX_MAX_DESC << 3) /* maximum number of descriptors * call to start used per */ #define TX_CLEAN_MAX_DESC (TX_MAX_DESC << 4) /* maximum tx descriptors * to clean per iteration */ #define TX_WR_SIZE_MAX 11*1024 /* the maximum total size of packets aggregated into a single * TX WR */ #define TX_WR_COUNT_MAX 7 /* the maximum total number of packets that can be * aggregated into a single TX WR */ #if defined(__i386__) || defined(__amd64__) #define mb() __asm volatile("mfence":::"memory") #define rmb() __asm volatile("lfence":::"memory") #define wmb() __asm volatile("sfence" ::: "memory") #define smp_mb() mb() #define L1_CACHE_BYTES 128 static __inline void prefetch(void *x) { __asm volatile("prefetcht0 %0" :: "m" (*(unsigned long *)x)); } extern void kdb_backtrace(void); #define WARN_ON(condition) do { \ if (__predict_false((condition)!=0)) { \ log(LOG_WARNING, "BUG: warning at %s:%d/%s()\n", __FILE__, __LINE__, __FUNCTION__); \ kdb_backtrace(); \ } \ } while (0) #else /* !i386 && !amd64 */ #define mb() #define rmb() #define wmb() #define smp_mb() #define prefetch(x) #define L1_CACHE_BYTES 32 #endif struct buf_ring { caddr_t *br_ring; volatile uint32_t br_cons; volatile uint32_t br_prod; int br_size; struct mtx br_lock; }; struct buf_ring *buf_ring_alloc(int count, int flags); void buf_ring_free(struct buf_ring *); static __inline int buf_ring_count(struct buf_ring *mr) { int size = mr->br_size; uint32_t mask = size - 1; return ((size + mr->br_prod - mr->br_cons) & mask); } static __inline int buf_ring_empty(struct buf_ring *mr) { return (mr->br_cons == mr->br_prod); } static __inline int buf_ring_full(struct buf_ring *mr) { uint32_t mask; mask = mr->br_size - 1; return (mr->br_cons == ((mr->br_prod + 1) & mask)); } /* * The producer and consumer are independently locked * this relies on the consumer providing his own serialization * */ static __inline void * buf_ring_dequeue(struct buf_ring *mr) { uint32_t prod, cons, mask; caddr_t *ring, m; ring = (caddr_t *)mr->br_ring; mask = mr->br_size - 1; cons = mr->br_cons; mb(); prod = mr->br_prod; m = NULL; if (cons != prod) { m = ring[cons]; ring[cons] = NULL; mr->br_cons = (cons + 1) & mask; mb(); } return (m); } #ifdef DEBUG_BUFRING static __inline void __buf_ring_scan(struct buf_ring *mr, void *m, char *file, int line) { int i; for (i = 0; i < mr->br_size; i++) if (m == mr->br_ring[i]) panic("%s:%d m=%p present prod=%d cons=%d idx=%d", file, line, m, mr->br_prod, mr->br_cons, i); } static __inline void buf_ring_scan(struct buf_ring *mr, void *m, char *file, int line) { mtx_lock(&mr->br_lock); __buf_ring_scan(mr, m, file, line); mtx_unlock(&mr->br_lock); } #else static __inline void __buf_ring_scan(struct buf_ring *mr, void *m, char *file, int line) { } static __inline void buf_ring_scan(struct buf_ring *mr, void *m, char *file, int line) { } #endif static __inline int __buf_ring_enqueue(struct buf_ring *mr, void *m, char *file, int line) { uint32_t prod, cons, mask; int err; mask = mr->br_size - 1; prod = mr->br_prod; mb(); cons = mr->br_cons; __buf_ring_scan(mr, m, file, line); if (((prod + 1) & mask) != cons) { KASSERT(mr->br_ring[prod] == NULL, ("overwriting entry")); mr->br_ring[prod] = m; mb(); mr->br_prod = (prod + 1) & mask; err = 0; } else err = ENOBUFS; return (err); } static __inline int buf_ring_enqueue_(struct buf_ring *mr, void *m, char *file, int line) { int err; mtx_lock(&mr->br_lock); err = __buf_ring_enqueue(mr, m, file, line); mtx_unlock(&mr->br_lock); return (err); } #define buf_ring_enqueue(mr, m) buf_ring_enqueue_((mr), (m), __FILE__, __LINE__) static __inline void * buf_ring_peek(struct buf_ring *mr) { int prod, cons, mask; caddr_t *ring, m; ring = (caddr_t *)mr->br_ring; mask = mr->br_size - 1; cons = mr->br_cons; prod = mr->br_prod; m = NULL; if (cons != prod) m = ring[cons]; return (m); } #define DBG_RX (1 << 0) static const int debug_flags = DBG_RX; #ifdef DEBUG_PRINT #define DBG(flag, msg) do { \ if ((flag & debug_flags)) \ printf msg; \ } while (0) #else #define DBG(...) #endif #include #define promisc_rx_mode(rm) ((rm)->port->ifp->if_flags & IFF_PROMISC) #define allmulti_rx_mode(rm) ((rm)->port->ifp->if_flags & IFF_ALLMULTI) #define CH_ERR(adap, fmt, ...) log(LOG_ERR, fmt, ##__VA_ARGS__) #define CH_WARN(adap, fmt, ...) log(LOG_WARNING, fmt, ##__VA_ARGS__) #define CH_ALERT(adap, fmt, ...) log(LOG_ALERT, fmt, ##__VA_ARGS__) #define t3_os_sleep(x) DELAY((x) * 1000) #define test_and_clear_bit(bit, p) atomic_cmpset_int((p), ((*(p)) | (1<