127 128#if defined(SCTP_LOCAL_TRACE_BUF) 129 130#define SCTP_GET_CYCLECOUNT get_cyclecount() 131#define SCTP_CTR6 sctp_log_trace 132 133#else 134#define SCTP_CTR6 CTR6 135#endif 136 137/* 138 * Macros to expand out globals defined by various modules 139 * to either a real global or a virtualized instance of one, 140 * depending on whether VIMAGE is defined. 141 */ 142/* then define the macro(s) that hook into the vimage macros */ 143#define MODULE_GLOBAL(__SYMBOL) V_##__SYMBOL 144 145#define V_system_base_info VNET(system_base_info) 146#define SCTP_BASE_INFO(__m) V_system_base_info.sctppcbinfo.__m 147#define SCTP_BASE_STATS V_system_base_info.sctpstat 148#define SCTP_BASE_STATS_SYSCTL VNET_NAME(system_base_info.sctpstat) 149#define SCTP_BASE_STAT(__m) V_system_base_info.sctpstat.__m 150#define SCTP_BASE_SYSCTL(__m) VNET_NAME(system_base_info.sctpsysctl.__m) 151#define SCTP_BASE_VAR(__m) V_system_base_info.__m 152 153/* 154 * 155 */ 156#define USER_ADDR_NULL (NULL) /* FIX ME: temp */ 157 158#if defined(SCTP_DEBUG) 159#define SCTPDBG(level, params...) \ 160{ \ 161 do { \ 162 if (SCTP_BASE_SYSCTL(sctp_debug_on) & level ) { \ 163 printf(params); \ 164 } \ 165 } while (0); \ 166} 167#define SCTPDBG_ADDR(level, addr) \ 168{ \ 169 do { \ 170 if (SCTP_BASE_SYSCTL(sctp_debug_on) & level ) { \ 171 sctp_print_address(addr); \ 172 } \ 173 } while (0); \ 174} 175#define SCTPDBG_PKT(level, iph, sh) \ 176{ \ 177 do { \ 178 if (SCTP_BASE_SYSCTL(sctp_debug_on) & level) { \ 179 sctp_print_address_pkt(iph, sh); \ 180 } \ 181 } while (0); \ 182} 183#else 184#define SCTPDBG(level, params...) 185#define SCTPDBG_ADDR(level, addr) 186#define SCTPDBG_PKT(level, iph, sh) 187#endif 188#define SCTP_PRINTF(params...) printf(params) 189 190#ifdef SCTP_LTRACE_CHUNKS 191#define SCTP_LTRACE_CHK(a, b, c, d) if(SCTP_BASE_SYSCTL(sctp_logging_level) & SCTP_LTRACE_CHUNK_ENABLE) SCTP_CTR6(KTR_SUBSYS, "SCTP:%d[%d]:%x-%x-%x-%x", SCTP_LOG_CHUNK_PROC, 0, a, b, c, d) 192#else 193#define SCTP_LTRACE_CHK(a, b, c, d) 194#endif 195 196#ifdef SCTP_LTRACE_ERRORS 197#define SCTP_LTRACE_ERR_RET_PKT(m, inp, stcb, net, file, err) if(SCTP_BASE_SYSCTL(sctp_logging_level) & SCTP_LTRACE_ERROR_ENABLE) \ 198 printf("mbuf:%p inp:%p stcb:%p net:%p file:%x line:%d error:%d\n", \ 199 m, inp, stcb, net, file, __LINE__, err); 200#define SCTP_LTRACE_ERR_RET(inp, stcb, net, file, err) if(SCTP_BASE_SYSCTL(sctp_logging_level) & SCTP_LTRACE_ERROR_ENABLE) \ 201 printf("inp:%p stcb:%p net:%p file:%x line:%d error:%d\n", \ 202 inp, stcb, net, file, __LINE__, err); 203#else 204#define SCTP_LTRACE_ERR_RET_PKT(m, inp, stcb, net, file, err) 205#define SCTP_LTRACE_ERR_RET(inp, stcb, net, file, err) 206#endif 207 208 209/* 210 * Local address and interface list handling 211 */ 212#define SCTP_MAX_VRF_ID 0 213#define SCTP_SIZE_OF_VRF_HASH 3 214#define SCTP_IFNAMSIZ IFNAMSIZ 215#define SCTP_DEFAULT_VRFID 0 216#define SCTP_VRF_ADDR_HASH_SIZE 16 217#define SCTP_VRF_IFN_HASH_SIZE 3 218#define SCTP_INIT_VRF_TABLEID(vrf) 219 220#define SCTP_IFN_IS_IFT_LOOP(ifn) ((ifn)->ifn_type == IFT_LOOP) 221#define SCTP_ROUTE_IS_REAL_LOOP(ro) ((ro)->ro_rt && (ro)->ro_rt->rt_ifa && (ro)->ro_rt->rt_ifa->ifa_ifp && (ro)->ro_rt->rt_ifa->ifa_ifp->if_type == IFT_LOOP) 222 223/* 224 * Access to IFN's to help with src-addr-selection 225 */ 226/* This could return VOID if the index works but for BSD we provide both. */ 227#define SCTP_GET_IFN_VOID_FROM_ROUTE(ro) (void *)ro->ro_rt->rt_ifp 228#define SCTP_GET_IF_INDEX_FROM_ROUTE(ro) (ro)->ro_rt->rt_ifp->if_index 229#define SCTP_ROUTE_HAS_VALID_IFN(ro) ((ro)->ro_rt && (ro)->ro_rt->rt_ifp) 230 231/* 232 * general memory allocation 233 */ 234#define SCTP_MALLOC(var, type, size, name) \ 235 do { \ 236 var = (type)malloc(size, name, M_NOWAIT); \ 237 } while (0) 238 239#define SCTP_FREE(var, type) free(var, type) 240 241#define SCTP_MALLOC_SONAME(var, type, size) \ 242 do { \ 243 var = (type)malloc(size, M_SONAME, M_WAITOK | M_ZERO); \ 244 } while (0) 245 246#define SCTP_FREE_SONAME(var) free(var, M_SONAME) 247 248#define SCTP_PROCESS_STRUCT struct proc * 249 250/* 251 * zone allocation functions 252 */ 253#include <vm/uma.h> 254 255/* SCTP_ZONE_INIT: initialize the zone */ 256typedef struct uma_zone *sctp_zone_t; 257 258#define SCTP_ZONE_INIT(zone, name, size, number) { \ 259 zone = uma_zcreate(name, size, NULL, NULL, NULL, NULL, UMA_ALIGN_PTR,\ 260 0); \ 261 uma_zone_set_max(zone, number); \ 262} 263 264#define SCTP_ZONE_DESTROY(zone) uma_zdestroy(zone) 265 266/* SCTP_ZONE_GET: allocate element from the zone */ 267#define SCTP_ZONE_GET(zone, type) \ 268 (type *)uma_zalloc(zone, M_NOWAIT); 269 270/* SCTP_ZONE_FREE: free element from the zone */ 271#define SCTP_ZONE_FREE(zone, element) \ 272 uma_zfree(zone, element); 273 274#define SCTP_HASH_INIT(size, hashmark) hashinit_flags(size, M_PCB, hashmark, HASH_NOWAIT) 275#define SCTP_HASH_FREE(table, hashmark) hashdestroy(table, M_PCB, hashmark) 276 277#define SCTP_M_COPYM m_copym 278 279/* 280 * timers 281 */ 282#include <sys/callout.h> 283typedef struct callout sctp_os_timer_t; 284 285 286#define SCTP_OS_TIMER_INIT(tmr) callout_init(tmr, 1) 287#define SCTP_OS_TIMER_START callout_reset 288#define SCTP_OS_TIMER_STOP callout_stop 289#define SCTP_OS_TIMER_STOP_DRAIN callout_drain 290#define SCTP_OS_TIMER_PENDING callout_pending 291#define SCTP_OS_TIMER_ACTIVE callout_active 292#define SCTP_OS_TIMER_DEACTIVATE callout_deactivate 293 294#define sctp_get_tick_count() (ticks) 295 296#define SCTP_UNUSED __attribute__((unused)) 297 298/* 299 * Functions 300 */ 301/* Mbuf manipulation and access macros */ 302#define SCTP_BUF_LEN(m) (m->m_len) 303#define SCTP_BUF_NEXT(m) (m->m_next) 304#define SCTP_BUF_NEXT_PKT(m) (m->m_nextpkt) 305#define SCTP_BUF_RESV_UF(m, size) m->m_data += size 306#define SCTP_BUF_AT(m, size) m->m_data + size 307#define SCTP_BUF_IS_EXTENDED(m) (m->m_flags & M_EXT) 308#define SCTP_BUF_EXTEND_SIZE(m) (m->m_ext.ext_size) 309#define SCTP_BUF_TYPE(m) (m->m_type) 310#define SCTP_BUF_RECVIF(m) (m->m_pkthdr.rcvif) 311#define SCTP_BUF_PREPEND M_PREPEND 312 313#define SCTP_ALIGN_TO_END(m, len) if(m->m_flags & M_PKTHDR) { \ 314 MH_ALIGN(m, len); \ 315 } else if ((m->m_flags & M_EXT) == 0) { \ 316 M_ALIGN(m, len); \ 317 } 318 319/* We make it so if you have up to 4 threads 320 * writing based on the default size of 321 * the packet log 65 k, that would be 322 * 4 16k packets before we would hit 323 * a problem. 324 */ 325#define SCTP_PKTLOG_WRITERS_NEED_LOCK 3 326 327/*************************/ 328/* MTU */ 329/*************************/ 330#define SCTP_GATHER_MTU_FROM_IFN_INFO(ifn, ifn_index, af) ((struct ifnet *)ifn)->if_mtu 331#define SCTP_GATHER_MTU_FROM_ROUTE(sctp_ifa, sa, rt) ((rt != NULL) ? rt->rt_rmx.rmx_mtu : 0) 332#define SCTP_GATHER_MTU_FROM_INTFC(sctp_ifn) ((sctp_ifn->ifn_p != NULL) ? ((struct ifnet *)(sctp_ifn->ifn_p))->if_mtu : 0) 333#define SCTP_SET_MTU_OF_ROUTE(sa, rt, mtu) do { \ 334 if (rt != NULL) \ 335 rt->rt_rmx.rmx_mtu = mtu; \ 336 } while(0) 337 338/* (de-)register interface event notifications */ 339#define SCTP_REGISTER_INTERFACE(ifhandle, af) 340#define SCTP_DEREGISTER_INTERFACE(ifhandle, af) 341 342 343/*************************/ 344/* These are for logging */ 345/*************************/ 346/* return the base ext data pointer */ 347#define SCTP_BUF_EXTEND_BASE(m) (m->m_ext.ext_buf) 348 /* return the refcnt of the data pointer */ 349#define SCTP_BUF_EXTEND_REFCNT(m) (*m->m_ext.ref_cnt) 350/* return any buffer related flags, this is 351 * used beyond logging for apple only. 352 */ 353#define SCTP_BUF_GET_FLAGS(m) (m->m_flags) 354 355/* For BSD this just accesses the M_PKTHDR length 356 * so it operates on an mbuf with hdr flag. Other 357 * O/S's may have separate packet header and mbuf 358 * chain pointers.. thus the macro. 359 */ 360#define SCTP_HEADER_TO_CHAIN(m) (m) 361#define SCTP_DETACH_HEADER_FROM_CHAIN(m) 362#define SCTP_HEADER_LEN(m) (m->m_pkthdr.len) 363#define SCTP_GET_HEADER_FOR_OUTPUT(o_pak) 0 364#define SCTP_RELEASE_HEADER(m) 365#define SCTP_RELEASE_PKT(m) sctp_m_freem(m) 366#define SCTP_ENABLE_UDP_CSUM(m) do { \ 367 m->m_pkthdr.csum_flags = CSUM_UDP; \ 368 m->m_pkthdr.csum_data = offsetof(struct udphdr, uh_sum); \ 369 } while (0) 370 371#define SCTP_GET_PKT_VRFID(m, vrf_id) ((vrf_id = SCTP_DEFAULT_VRFID) != SCTP_DEFAULT_VRFID) 372 373 374 375/* Attach the chain of data into the sendable packet. */ 376#define SCTP_ATTACH_CHAIN(pak, m, packet_length) do { \ 377 pak = m; \ 378 pak->m_pkthdr.len = packet_length; \ 379 } while(0) 380 381/* Other m_pkthdr type things */ 382#define SCTP_IS_IT_BROADCAST(dst, m) ((m->m_flags & M_PKTHDR) ? in_broadcast(dst, m->m_pkthdr.rcvif) : 0) 383#define SCTP_IS_IT_LOOPBACK(m) ((m->m_flags & M_PKTHDR) && ((m->m_pkthdr.rcvif == NULL) || (m->m_pkthdr.rcvif->if_type == IFT_LOOP))) 384 385 386/* This converts any input packet header 387 * into the chain of data holders, for BSD 388 * its a NOP. 389 */ 390 391/* Macro's for getting length from V6/V4 header */ 392#define SCTP_GET_IPV4_LENGTH(iph) (iph->ip_len) 393#define SCTP_GET_IPV6_LENGTH(ip6) (ntohs(ip6->ip6_plen)) 394 395/* get the v6 hop limit */ 396#define SCTP_GET_HLIM(inp, ro) in6_selecthlim((struct in6pcb *)&inp->ip_inp.inp, (ro ? (ro->ro_rt ? (ro->ro_rt->rt_ifp) : (NULL)) : (NULL))); 397 398/* is the endpoint v6only? */ 399#define SCTP_IPV6_V6ONLY(inp) (((struct inpcb *)inp)->inp_flags & IN6P_IPV6_V6ONLY) 400/* is the socket non-blocking? */ 401#define SCTP_SO_IS_NBIO(so) ((so)->so_state & SS_NBIO) 402#define SCTP_SET_SO_NBIO(so) ((so)->so_state |= SS_NBIO) 403#define SCTP_CLEAR_SO_NBIO(so) ((so)->so_state &= ~SS_NBIO) 404/* get the socket type */ 405#define SCTP_SO_TYPE(so) ((so)->so_type) 406/* reserve sb space for a socket */ 407#define SCTP_SORESERVE(so, send, recv) soreserve(so, send, recv) 408/* wakeup a socket */ 409#define SCTP_SOWAKEUP(so) wakeup(&(so)->so_timeo) 410/* clear the socket buffer state */ 411#define SCTP_SB_CLEAR(sb) \ 412 (sb).sb_cc = 0; \ 413 (sb).sb_mb = NULL; \ 414 (sb).sb_mbcnt = 0; 415 416#define SCTP_SB_LIMIT_RCV(so) so->so_rcv.sb_hiwat 417#define SCTP_SB_LIMIT_SND(so) so->so_snd.sb_hiwat 418 419/* 420 * routes, output, etc. 421 */ 422typedef struct route sctp_route_t; 423typedef struct rtentry sctp_rtentry_t; 424 425#define SCTP_RTALLOC(ro, vrf_id) rtalloc_ign((struct route *)ro, 0UL) 426 427/* Future zero copy wakeup/send function */ 428#define SCTP_ZERO_COPY_EVENT(inp, so) 429/* This is re-pulse ourselves for sendbuf */ 430#define SCTP_ZERO_COPY_SENDQ_EVENT(inp, so) 431 432/* 433 * IP output routines 434 */ 435#define SCTP_IP_OUTPUT(result, o_pak, ro, stcb, vrf_id) \ 436{ \ 437 int o_flgs = IP_RAWOUTPUT; \ 438 struct sctp_tcb *local_stcb = stcb; \ 439 if (local_stcb && \ 440 local_stcb->sctp_ep && \ 441 local_stcb->sctp_ep->sctp_socket) \ 442 o_flgs |= local_stcb->sctp_ep->sctp_socket->so_options & SO_DONTROUTE; \ 443 result = ip_output(o_pak, NULL, ro, o_flgs, 0, NULL); \ 444} 445 446#define SCTP_IP6_OUTPUT(result, o_pak, ro, ifp, stcb, vrf_id) \ 447{ \ 448 struct sctp_tcb *local_stcb = stcb; \ 449 if (local_stcb && local_stcb->sctp_ep) \ 450 result = ip6_output(o_pak, \ 451 ((struct in6pcb *)(local_stcb->sctp_ep))->in6p_outputopts, \ 452 (ro), 0, 0, ifp, NULL); \ 453 else \ 454 result = ip6_output(o_pak, NULL, (ro), 0, 0, ifp, NULL); \ 455} 456 457struct mbuf * 458sctp_get_mbuf_for_msg(unsigned int space_needed, 459 int want_header, int how, int allonebuf, int type); 460 461 462/* 463 * SCTP AUTH 464 */ 465#define HAVE_SHA2 466 467#define SCTP_READ_RANDOM(buf, len) read_random(buf, len) 468 469#ifdef USE_SCTP_SHA1 470#include <netinet/sctp_sha1.h> 471#else 472#include <crypto/sha1.h> 473/* map standard crypto API names */ 474#define SHA1_Init SHA1Init 475#define SHA1_Update SHA1Update 476#define SHA1_Final(x,y) SHA1Final((caddr_t)x, y) 477#endif 478 479#if defined(HAVE_SHA2) 480#include <crypto/sha2/sha2.h> 481#endif 482 483#endif 484 485#define SCTP_DECREMENT_AND_CHECK_REFCOUNT(addr) (atomic_fetchadd_int(addr, -1) == 1) 486#if defined(INVARIANTS) 487#define SCTP_SAVE_ATOMIC_DECREMENT(addr, val) \ 488{ \ 489 int32_t oldval; \ 490 oldval = atomic_fetchadd_int(addr, -val); \ 491 if (oldval < val) { \ 492 panic("Counter goes negative"); \ 493 } \ 494} 495#else 496#define SCTP_SAVE_ATOMIC_DECREMENT(addr, val) \ 497{ \ 498 int32_t oldval; \ 499 oldval = atomic_fetchadd_int(addr, -val); \ 500 if (oldval < val) { \ 501 *addr = 0; \ 502 } \ 503} 504#endif
|