1 /*- 2 * Copyright (c) 2010-2011, by Michael Tuexen. All rights reserved. 3 * Copyright (c) 2010-2011, by Randall Stewart. All rights reserved. 4 * Copyright (c) 2010-2011, by Robin Seggelmann. All rights reserved. 5 * 6 * Redistribution and use in source and binary forms, with or without 7 * modification, are permitted provided that the following conditions are met: 8 * 9 * a) Redistributions of source code must retain the above copyright notice, 10 * this list of conditions and the following disclaimer. 11 * 12 * b) Redistributions in binary form must reproduce the above copyright 13 * notice, this list of conditions and the following disclaimer in 14 * the documentation and/or other materials provided with the distribution. 15 * 16 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 17 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, 18 * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 19 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE 20 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 21 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 22 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 23 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 24 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 25 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF 26 * THE POSSIBILITY OF SUCH DAMAGE. 27 */ 28 29 #include <sys/cdefs.h> 30 __FBSDID("$FreeBSD$"); 31 32 #include <netinet/sctp_pcb.h> 33 34 /* 35 * Default simple round-robin algorithm. 36 * Just interates the streams in the order they appear. 37 */ 38 39 static void 40 sctp_ss_default_add(struct sctp_tcb *, struct sctp_association *, 41 struct sctp_stream_out *, 42 struct sctp_stream_queue_pending *, int); 43 44 static void 45 sctp_ss_default_remove(struct sctp_tcb *, struct sctp_association *, 46 struct sctp_stream_out *, 47 struct sctp_stream_queue_pending *, int); 48 49 static void 50 sctp_ss_default_init(struct sctp_tcb *stcb, struct sctp_association *asoc, 51 int holds_lock) 52 { 53 uint16_t i; 54 55 TAILQ_INIT(&asoc->ss_data.out_wheel); 56 /* 57 * If there is data in the stream queues already, the scheduler of 58 * an existing association has been changed. We need to add all 59 * stream queues to the wheel. 60 */ 61 for (i = 0; i < stcb->asoc.streamoutcnt; i++) { 62 stcb->asoc.ss_functions.sctp_ss_add_to_stream(stcb, &stcb->asoc, 63 &stcb->asoc.strmout[i], 64 NULL, holds_lock); 65 } 66 return; 67 } 68 69 static void 70 sctp_ss_default_clear(struct sctp_tcb *stcb, struct sctp_association *asoc, 71 int clear_values, int holds_lock) 72 { 73 if (holds_lock == 0) { 74 SCTP_TCB_SEND_LOCK(stcb); 75 } 76 while (!TAILQ_EMPTY(&asoc->ss_data.out_wheel)) { 77 struct sctp_stream_out *strq = TAILQ_FIRST(&asoc->ss_data.out_wheel); 78 79 TAILQ_REMOVE(&asoc->ss_data.out_wheel, TAILQ_FIRST(&asoc->ss_data.out_wheel), ss_params.rr.next_spoke); 80 strq->ss_params.rr.next_spoke.tqe_next = NULL; 81 strq->ss_params.rr.next_spoke.tqe_prev = NULL; 82 } 83 asoc->last_out_stream = NULL; 84 if (holds_lock == 0) { 85 SCTP_TCB_SEND_UNLOCK(stcb); 86 } 87 return; 88 } 89 90 static void 91 sctp_ss_default_init_stream(struct sctp_stream_out *strq, struct sctp_stream_out *with_strq) 92 { 93 strq->ss_params.rr.next_spoke.tqe_next = NULL; 94 strq->ss_params.rr.next_spoke.tqe_prev = NULL; 95 return; 96 } 97 98 static void 99 sctp_ss_default_add(struct sctp_tcb *stcb, struct sctp_association *asoc, 100 struct sctp_stream_out *strq, 101 struct sctp_stream_queue_pending *sp, int holds_lock) 102 { 103 if (holds_lock == 0) { 104 SCTP_TCB_SEND_LOCK(stcb); 105 } 106 /* Add to wheel if not already on it and stream queue not empty */ 107 if (!TAILQ_EMPTY(&strq->outqueue) && 108 (strq->ss_params.rr.next_spoke.tqe_next == NULL) && 109 (strq->ss_params.rr.next_spoke.tqe_prev == NULL)) { 110 TAILQ_INSERT_TAIL(&asoc->ss_data.out_wheel, 111 strq, ss_params.rr.next_spoke); 112 } 113 if (holds_lock == 0) { 114 SCTP_TCB_SEND_UNLOCK(stcb); 115 } 116 return; 117 } 118 119 static int 120 sctp_ss_default_is_empty(struct sctp_tcb *stcb, struct sctp_association *asoc) 121 { 122 if (TAILQ_EMPTY(&asoc->ss_data.out_wheel)) { 123 return (1); 124 } else { 125 return (0); 126 } 127 } 128 129 static void 130 sctp_ss_default_remove(struct sctp_tcb *stcb, struct sctp_association *asoc, 131 struct sctp_stream_out *strq, 132 struct sctp_stream_queue_pending *sp, int holds_lock) 133 { 134 if (holds_lock == 0) { 135 SCTP_TCB_SEND_LOCK(stcb); 136 } 137 /* 138 * Remove from wheel if stream queue is empty and actually is on the 139 * wheel 140 */ 141 if (TAILQ_EMPTY(&strq->outqueue) && 142 (strq->ss_params.rr.next_spoke.tqe_next != NULL || 143 strq->ss_params.rr.next_spoke.tqe_prev != NULL)) { 144 if (asoc->last_out_stream == strq) { 145 asoc->last_out_stream = TAILQ_PREV(asoc->last_out_stream, 146 sctpwheel_listhead, 147 ss_params.rr.next_spoke); 148 if (asoc->last_out_stream == NULL) { 149 asoc->last_out_stream = TAILQ_LAST(&asoc->ss_data.out_wheel, 150 sctpwheel_listhead); 151 } 152 if (asoc->last_out_stream == strq) { 153 asoc->last_out_stream = NULL; 154 } 155 } 156 TAILQ_REMOVE(&asoc->ss_data.out_wheel, strq, ss_params.rr.next_spoke); 157 strq->ss_params.rr.next_spoke.tqe_next = NULL; 158 strq->ss_params.rr.next_spoke.tqe_prev = NULL; 159 } 160 if (holds_lock == 0) { 161 SCTP_TCB_SEND_UNLOCK(stcb); 162 } 163 return; 164 } 165 166 167 static struct sctp_stream_out * 168 sctp_ss_default_select(struct sctp_tcb *stcb, struct sctp_nets *net, 169 struct sctp_association *asoc) 170 { 171 struct sctp_stream_out *strq, *strqt; 172 173 strqt = asoc->last_out_stream; 174 default_again: 175 /* Find the next stream to use */ 176 if (strqt == NULL) { 177 strq = TAILQ_FIRST(&asoc->ss_data.out_wheel); 178 } else { 179 strq = TAILQ_NEXT(strqt, ss_params.rr.next_spoke); 180 if (strq == NULL) { 181 strq = TAILQ_FIRST(&asoc->ss_data.out_wheel); 182 } 183 } 184 185 /* 186 * If CMT is off, we must validate that the stream in question has 187 * the first item pointed towards are network destination requested 188 * by the caller. Note that if we turn out to be locked to a stream 189 * (assigning TSN's then we must stop, since we cannot look for 190 * another stream with data to send to that destination). In CMT's 191 * case, by skipping this check, we will send one data packet 192 * towards the requested net. 193 */ 194 if (net != NULL && strq != NULL && 195 SCTP_BASE_SYSCTL(sctp_cmt_on_off) == 0) { 196 if (TAILQ_FIRST(&strq->outqueue) && 197 TAILQ_FIRST(&strq->outqueue)->net != NULL && 198 TAILQ_FIRST(&strq->outqueue)->net != net) { 199 if (strq == asoc->last_out_stream) { 200 return (NULL); 201 } else { 202 strqt = strq; 203 goto default_again; 204 } 205 } 206 } 207 return (strq); 208 } 209 210 static void 211 sctp_ss_default_scheduled(struct sctp_tcb *stcb, struct sctp_nets *net, 212 struct sctp_association *asoc, 213 struct sctp_stream_out *strq, int moved_how_much) 214 { 215 asoc->last_out_stream = strq; 216 return; 217 } 218 219 static void 220 sctp_ss_default_packet_done(struct sctp_tcb *stcb, struct sctp_nets *net, 221 struct sctp_association *asoc) 222 { 223 /* Nothing to be done here */ 224 return; 225 } 226 227 static int 228 sctp_ss_default_get_value(struct sctp_tcb *stcb, struct sctp_association *asoc, 229 struct sctp_stream_out *strq, uint16_t * value) 230 { 231 /* Nothing to be done here */ 232 return (-1); 233 } 234 235 static int 236 sctp_ss_default_set_value(struct sctp_tcb *stcb, struct sctp_association *asoc, 237 struct sctp_stream_out *strq, uint16_t value) 238 { 239 /* Nothing to be done here */ 240 return (-1); 241 } 242 243 /* 244 * Real round-robin algorithm. 245 * Always interates the streams in ascending order. 246 */ 247 static void 248 sctp_ss_rr_add(struct sctp_tcb *stcb, struct sctp_association *asoc, 249 struct sctp_stream_out *strq, 250 struct sctp_stream_queue_pending *sp, int holds_lock) 251 { 252 struct sctp_stream_out *strqt; 253 254 if (holds_lock == 0) { 255 SCTP_TCB_SEND_LOCK(stcb); 256 } 257 if (!TAILQ_EMPTY(&strq->outqueue) && 258 (strq->ss_params.rr.next_spoke.tqe_next == NULL) && 259 (strq->ss_params.rr.next_spoke.tqe_prev == NULL)) { 260 if (TAILQ_EMPTY(&asoc->ss_data.out_wheel)) { 261 TAILQ_INSERT_HEAD(&asoc->ss_data.out_wheel, strq, ss_params.rr.next_spoke); 262 } else { 263 strqt = TAILQ_FIRST(&asoc->ss_data.out_wheel); 264 while (strqt != NULL && (strqt->stream_no < strq->stream_no)) { 265 strqt = TAILQ_NEXT(strqt, ss_params.rr.next_spoke); 266 } 267 if (strqt != NULL) { 268 TAILQ_INSERT_BEFORE(strqt, strq, ss_params.rr.next_spoke); 269 } else { 270 TAILQ_INSERT_TAIL(&asoc->ss_data.out_wheel, strq, ss_params.rr.next_spoke); 271 } 272 } 273 } 274 if (holds_lock == 0) { 275 SCTP_TCB_SEND_UNLOCK(stcb); 276 } 277 return; 278 } 279 280 /* 281 * Real round-robin per packet algorithm. 282 * Always interates the streams in ascending order and 283 * only fills messages of the same stream in a packet. 284 */ 285 static struct sctp_stream_out * 286 sctp_ss_rrp_select(struct sctp_tcb *stcb, struct sctp_nets *net, 287 struct sctp_association *asoc) 288 { 289 return asoc->last_out_stream; 290 } 291 292 static void 293 sctp_ss_rrp_packet_done(struct sctp_tcb *stcb, struct sctp_nets *net, 294 struct sctp_association *asoc) 295 { 296 struct sctp_stream_out *strq, *strqt; 297 298 strqt = asoc->last_out_stream; 299 rrp_again: 300 /* Find the next stream to use */ 301 if (strqt == NULL) { 302 strq = TAILQ_FIRST(&asoc->ss_data.out_wheel); 303 } else { 304 strq = TAILQ_NEXT(strqt, ss_params.rr.next_spoke); 305 if (strq == NULL) { 306 strq = TAILQ_FIRST(&asoc->ss_data.out_wheel); 307 } 308 } 309 310 /* 311 * If CMT is off, we must validate that the stream in question has 312 * the first item pointed towards are network destination requested 313 * by the caller. Note that if we turn out to be locked to a stream 314 * (assigning TSN's then we must stop, since we cannot look for 315 * another stream with data to send to that destination). In CMT's 316 * case, by skipping this check, we will send one data packet 317 * towards the requested net. 318 */ 319 if (net != NULL && strq != NULL && 320 SCTP_BASE_SYSCTL(sctp_cmt_on_off) == 0) { 321 if (TAILQ_FIRST(&strq->outqueue) && 322 TAILQ_FIRST(&strq->outqueue)->net != NULL && 323 TAILQ_FIRST(&strq->outqueue)->net != net) { 324 if (strq == asoc->last_out_stream) { 325 strq = NULL; 326 } else { 327 strqt = strq; 328 goto rrp_again; 329 } 330 } 331 } 332 asoc->last_out_stream = strq; 333 return; 334 } 335 336 337 /* 338 * Priority algorithm. 339 * Always prefers streams based on their priority id. 340 */ 341 static void 342 sctp_ss_prio_clear(struct sctp_tcb *stcb, struct sctp_association *asoc, 343 int clear_values, int holds_lock) 344 { 345 if (holds_lock == 0) { 346 SCTP_TCB_SEND_LOCK(stcb); 347 } 348 while (!TAILQ_EMPTY(&asoc->ss_data.out_wheel)) { 349 struct sctp_stream_out *strq = TAILQ_FIRST(&asoc->ss_data.out_wheel); 350 351 if (clear_values) { 352 strq->ss_params.prio.priority = 0; 353 } 354 TAILQ_REMOVE(&asoc->ss_data.out_wheel, TAILQ_FIRST(&asoc->ss_data.out_wheel), ss_params.prio.next_spoke); 355 strq->ss_params.prio.next_spoke.tqe_next = NULL; 356 strq->ss_params.prio.next_spoke.tqe_prev = NULL; 357 358 } 359 asoc->last_out_stream = NULL; 360 if (holds_lock == 0) { 361 SCTP_TCB_SEND_UNLOCK(stcb); 362 } 363 return; 364 } 365 366 static void 367 sctp_ss_prio_init_stream(struct sctp_stream_out *strq, struct sctp_stream_out *with_strq) 368 { 369 strq->ss_params.prio.next_spoke.tqe_next = NULL; 370 strq->ss_params.prio.next_spoke.tqe_prev = NULL; 371 if (with_strq != NULL) { 372 strq->ss_params.prio.priority = with_strq->ss_params.prio.priority; 373 } else { 374 strq->ss_params.prio.priority = 0; 375 } 376 return; 377 } 378 379 static void 380 sctp_ss_prio_add(struct sctp_tcb *stcb, struct sctp_association *asoc, 381 struct sctp_stream_out *strq, struct sctp_stream_queue_pending *sp, 382 int holds_lock) 383 { 384 struct sctp_stream_out *strqt; 385 386 if (holds_lock == 0) { 387 SCTP_TCB_SEND_LOCK(stcb); 388 } 389 /* Add to wheel if not already on it and stream queue not empty */ 390 if (!TAILQ_EMPTY(&strq->outqueue) && 391 (strq->ss_params.prio.next_spoke.tqe_next == NULL) && 392 (strq->ss_params.prio.next_spoke.tqe_prev == NULL)) { 393 if (TAILQ_EMPTY(&asoc->ss_data.out_wheel)) { 394 TAILQ_INSERT_HEAD(&asoc->ss_data.out_wheel, strq, ss_params.prio.next_spoke); 395 } else { 396 strqt = TAILQ_FIRST(&asoc->ss_data.out_wheel); 397 while (strqt != NULL && strqt->ss_params.prio.priority < strq->ss_params.prio.priority) { 398 strqt = TAILQ_NEXT(strqt, ss_params.prio.next_spoke); 399 } 400 if (strqt != NULL) { 401 TAILQ_INSERT_BEFORE(strqt, strq, ss_params.prio.next_spoke); 402 } else { 403 TAILQ_INSERT_TAIL(&asoc->ss_data.out_wheel, strq, ss_params.prio.next_spoke); 404 } 405 } 406 } 407 if (holds_lock == 0) { 408 SCTP_TCB_SEND_UNLOCK(stcb); 409 } 410 return; 411 } 412 413 static void 414 sctp_ss_prio_remove(struct sctp_tcb *stcb, struct sctp_association *asoc, 415 struct sctp_stream_out *strq, struct sctp_stream_queue_pending *sp, 416 int holds_lock) 417 { 418 if (holds_lock == 0) { 419 SCTP_TCB_SEND_LOCK(stcb); 420 } 421 /* 422 * Remove from wheel if stream queue is empty and actually is on the 423 * wheel 424 */ 425 if (TAILQ_EMPTY(&strq->outqueue) && 426 (strq->ss_params.prio.next_spoke.tqe_next != NULL || 427 strq->ss_params.prio.next_spoke.tqe_prev != NULL)) { 428 if (asoc->last_out_stream == strq) { 429 asoc->last_out_stream = TAILQ_PREV(asoc->last_out_stream, sctpwheel_listhead, 430 ss_params.prio.next_spoke); 431 if (asoc->last_out_stream == NULL) { 432 asoc->last_out_stream = TAILQ_LAST(&asoc->ss_data.out_wheel, 433 sctpwheel_listhead); 434 } 435 if (asoc->last_out_stream == strq) { 436 asoc->last_out_stream = NULL; 437 } 438 } 439 TAILQ_REMOVE(&asoc->ss_data.out_wheel, strq, ss_params.prio.next_spoke); 440 strq->ss_params.prio.next_spoke.tqe_next = NULL; 441 strq->ss_params.prio.next_spoke.tqe_prev = NULL; 442 } 443 if (holds_lock == 0) { 444 SCTP_TCB_SEND_UNLOCK(stcb); 445 } 446 return; 447 } 448 449 static struct sctp_stream_out * 450 sctp_ss_prio_select(struct sctp_tcb *stcb, struct sctp_nets *net, 451 struct sctp_association *asoc) 452 { 453 struct sctp_stream_out *strq, *strqt, *strqn; 454 455 strqt = asoc->last_out_stream; 456 prio_again: 457 /* Find the next stream to use */ 458 if (strqt == NULL) { 459 strq = TAILQ_FIRST(&asoc->ss_data.out_wheel); 460 } else { 461 strqn = TAILQ_NEXT(strqt, ss_params.prio.next_spoke); 462 if (strqn != NULL && 463 strqn->ss_params.prio.priority == strqt->ss_params.prio.priority) { 464 strq = strqn; 465 } else { 466 strq = TAILQ_FIRST(&asoc->ss_data.out_wheel); 467 } 468 } 469 470 /* 471 * If CMT is off, we must validate that the stream in question has 472 * the first item pointed towards are network destination requested 473 * by the caller. Note that if we turn out to be locked to a stream 474 * (assigning TSN's then we must stop, since we cannot look for 475 * another stream with data to send to that destination). In CMT's 476 * case, by skipping this check, we will send one data packet 477 * towards the requested net. 478 */ 479 if (net != NULL && strq != NULL && 480 SCTP_BASE_SYSCTL(sctp_cmt_on_off) == 0) { 481 if (TAILQ_FIRST(&strq->outqueue) && 482 TAILQ_FIRST(&strq->outqueue)->net != NULL && 483 TAILQ_FIRST(&strq->outqueue)->net != net) { 484 if (strq == asoc->last_out_stream) { 485 return (NULL); 486 } else { 487 strqt = strq; 488 goto prio_again; 489 } 490 } 491 } 492 return (strq); 493 } 494 495 static int 496 sctp_ss_prio_get_value(struct sctp_tcb *stcb, struct sctp_association *asoc, 497 struct sctp_stream_out *strq, uint16_t * value) 498 { 499 if (strq == NULL) { 500 return (-1); 501 } 502 *value = strq->ss_params.prio.priority; 503 return (1); 504 } 505 506 static int 507 sctp_ss_prio_set_value(struct sctp_tcb *stcb, struct sctp_association *asoc, 508 struct sctp_stream_out *strq, uint16_t value) 509 { 510 if (strq == NULL) { 511 return (-1); 512 } 513 strq->ss_params.prio.priority = value; 514 sctp_ss_prio_remove(stcb, asoc, strq, NULL, 1); 515 sctp_ss_prio_add(stcb, asoc, strq, NULL, 1); 516 return (1); 517 } 518 519 /* 520 * Fair bandwidth algorithm. 521 * Maintains an equal troughput per stream. 522 */ 523 static void 524 sctp_ss_fb_clear(struct sctp_tcb *stcb, struct sctp_association *asoc, 525 int clear_values, int holds_lock) 526 { 527 if (holds_lock == 0) { 528 SCTP_TCB_SEND_LOCK(stcb); 529 } 530 while (!TAILQ_EMPTY(&asoc->ss_data.out_wheel)) { 531 struct sctp_stream_out *strq = TAILQ_FIRST(&asoc->ss_data.out_wheel); 532 533 if (clear_values) { 534 strq->ss_params.fb.rounds = -1; 535 } 536 TAILQ_REMOVE(&asoc->ss_data.out_wheel, TAILQ_FIRST(&asoc->ss_data.out_wheel), ss_params.fb.next_spoke); 537 strq->ss_params.fb.next_spoke.tqe_next = NULL; 538 strq->ss_params.fb.next_spoke.tqe_prev = NULL; 539 } 540 asoc->last_out_stream = NULL; 541 if (holds_lock == 0) { 542 SCTP_TCB_SEND_UNLOCK(stcb); 543 } 544 return; 545 } 546 547 static void 548 sctp_ss_fb_init_stream(struct sctp_stream_out *strq, struct sctp_stream_out *with_strq) 549 { 550 strq->ss_params.fb.next_spoke.tqe_next = NULL; 551 strq->ss_params.fb.next_spoke.tqe_prev = NULL; 552 if (with_strq != NULL) { 553 strq->ss_params.fb.rounds = with_strq->ss_params.fb.rounds; 554 } else { 555 strq->ss_params.fb.rounds = -1; 556 } 557 return; 558 } 559 560 static void 561 sctp_ss_fb_add(struct sctp_tcb *stcb, struct sctp_association *asoc, 562 struct sctp_stream_out *strq, struct sctp_stream_queue_pending *sp, 563 int holds_lock) 564 { 565 if (holds_lock == 0) { 566 SCTP_TCB_SEND_LOCK(stcb); 567 } 568 if (!TAILQ_EMPTY(&strq->outqueue) && 569 (strq->ss_params.fb.next_spoke.tqe_next == NULL) && 570 (strq->ss_params.fb.next_spoke.tqe_prev == NULL)) { 571 if (strq->ss_params.fb.rounds < 0) 572 strq->ss_params.fb.rounds = TAILQ_FIRST(&strq->outqueue)->length; 573 TAILQ_INSERT_TAIL(&asoc->ss_data.out_wheel, strq, ss_params.fb.next_spoke); 574 } 575 if (holds_lock == 0) { 576 SCTP_TCB_SEND_UNLOCK(stcb); 577 } 578 return; 579 } 580 581 static void 582 sctp_ss_fb_remove(struct sctp_tcb *stcb, struct sctp_association *asoc, 583 struct sctp_stream_out *strq, struct sctp_stream_queue_pending *sp, 584 int holds_lock) 585 { 586 if (holds_lock == 0) { 587 SCTP_TCB_SEND_LOCK(stcb); 588 } 589 /* 590 * Remove from wheel if stream queue is empty and actually is on the 591 * wheel 592 */ 593 if (TAILQ_EMPTY(&strq->outqueue) && 594 (strq->ss_params.fb.next_spoke.tqe_next != NULL || 595 strq->ss_params.fb.next_spoke.tqe_prev != NULL)) { 596 if (asoc->last_out_stream == strq) { 597 asoc->last_out_stream = TAILQ_PREV(asoc->last_out_stream, sctpwheel_listhead, 598 ss_params.fb.next_spoke); 599 if (asoc->last_out_stream == NULL) { 600 asoc->last_out_stream = TAILQ_LAST(&asoc->ss_data.out_wheel, 601 sctpwheel_listhead); 602 } 603 if (asoc->last_out_stream == strq) { 604 asoc->last_out_stream = NULL; 605 } 606 } 607 TAILQ_REMOVE(&asoc->ss_data.out_wheel, strq, ss_params.fb.next_spoke); 608 strq->ss_params.fb.next_spoke.tqe_next = NULL; 609 strq->ss_params.fb.next_spoke.tqe_prev = NULL; 610 } 611 if (holds_lock == 0) { 612 SCTP_TCB_SEND_UNLOCK(stcb); 613 } 614 return; 615 } 616 617 static struct sctp_stream_out * 618 sctp_ss_fb_select(struct sctp_tcb *stcb, struct sctp_nets *net, 619 struct sctp_association *asoc) 620 { 621 struct sctp_stream_out *strq = NULL, *strqt; 622 623 if (asoc->last_out_stream == NULL || 624 TAILQ_FIRST(&asoc->ss_data.out_wheel) == TAILQ_LAST(&asoc->ss_data.out_wheel, sctpwheel_listhead)) { 625 strqt = TAILQ_FIRST(&asoc->ss_data.out_wheel); 626 } else { 627 strqt = TAILQ_NEXT(asoc->last_out_stream, ss_params.fb.next_spoke); 628 } 629 do { 630 if ((strqt != NULL) && 631 ((SCTP_BASE_SYSCTL(sctp_cmt_on_off) > 0) || 632 (SCTP_BASE_SYSCTL(sctp_cmt_on_off) == 0 && 633 (net == NULL || (TAILQ_FIRST(&strqt->outqueue) && TAILQ_FIRST(&strqt->outqueue)->net == NULL) || 634 (net != NULL && TAILQ_FIRST(&strqt->outqueue) && TAILQ_FIRST(&strqt->outqueue)->net != NULL && 635 TAILQ_FIRST(&strqt->outqueue)->net == net))))) { 636 if ((strqt->ss_params.fb.rounds >= 0) && (strq == NULL || 637 strqt->ss_params.fb.rounds < strq->ss_params.fb.rounds)) { 638 strq = strqt; 639 } 640 } 641 if (strqt != NULL) { 642 strqt = TAILQ_NEXT(strqt, ss_params.fb.next_spoke); 643 } else { 644 strqt = TAILQ_FIRST(&asoc->ss_data.out_wheel); 645 } 646 } while (strqt != strq); 647 return (strq); 648 } 649 650 static void 651 sctp_ss_fb_scheduled(struct sctp_tcb *stcb, struct sctp_nets *net, 652 struct sctp_association *asoc, struct sctp_stream_out *strq, 653 int moved_how_much) 654 { 655 struct sctp_stream_out *strqt; 656 int subtract; 657 658 subtract = strq->ss_params.fb.rounds; 659 TAILQ_FOREACH(strqt, &asoc->ss_data.out_wheel, ss_params.fb.next_spoke) { 660 strqt->ss_params.fb.rounds -= subtract; 661 if (strqt->ss_params.fb.rounds < 0) 662 strqt->ss_params.fb.rounds = 0; 663 } 664 if (TAILQ_FIRST(&strq->outqueue)) { 665 strq->ss_params.fb.rounds = TAILQ_FIRST(&strq->outqueue)->length; 666 } else { 667 strq->ss_params.fb.rounds = -1; 668 } 669 asoc->last_out_stream = strq; 670 return; 671 } 672 673 /* 674 * First-come, first-serve algorithm. 675 * Maintains the order provided by the application. 676 */ 677 static void 678 sctp_ss_fcfs_add(struct sctp_tcb *stcb, struct sctp_association *asoc, 679 struct sctp_stream_out *strq, struct sctp_stream_queue_pending *sp, 680 int holds_lock); 681 682 static void 683 sctp_ss_fcfs_init(struct sctp_tcb *stcb, struct sctp_association *asoc, 684 int holds_lock) 685 { 686 uint32_t x, n = 0, add_more = 1; 687 struct sctp_stream_queue_pending *sp; 688 uint16_t i; 689 690 TAILQ_INIT(&asoc->ss_data.out_list); 691 /* 692 * If there is data in the stream queues already, the scheduler of 693 * an existing association has been changed. We can only cycle 694 * through the stream queues and add everything to the FCFS queue. 695 */ 696 while (add_more) { 697 add_more = 0; 698 for (i = 0; i < stcb->asoc.streamoutcnt; i++) { 699 sp = TAILQ_FIRST(&stcb->asoc.strmout[i].outqueue); 700 x = 0; 701 /* Find n. message in current stream queue */ 702 while (sp != NULL && x < n) { 703 sp = TAILQ_NEXT(sp, next); 704 x++; 705 } 706 if (sp != NULL) { 707 sctp_ss_fcfs_add(stcb, &stcb->asoc, &stcb->asoc.strmout[i], sp, holds_lock); 708 add_more = 1; 709 } 710 } 711 n++; 712 } 713 return; 714 } 715 716 static void 717 sctp_ss_fcfs_clear(struct sctp_tcb *stcb, struct sctp_association *asoc, 718 int clear_values, int holds_lock) 719 { 720 if (clear_values) { 721 if (holds_lock == 0) { 722 SCTP_TCB_SEND_LOCK(stcb); 723 } 724 while (!TAILQ_EMPTY(&asoc->ss_data.out_list)) { 725 TAILQ_REMOVE(&asoc->ss_data.out_list, TAILQ_FIRST(&asoc->ss_data.out_list), ss_next); 726 } 727 if (holds_lock == 0) { 728 SCTP_TCB_SEND_UNLOCK(stcb); 729 } 730 } 731 return; 732 } 733 734 static void 735 sctp_ss_fcfs_init_stream(struct sctp_stream_out *strq, struct sctp_stream_out *with_strq) 736 { 737 /* Nothing to be done here */ 738 return; 739 } 740 741 static void 742 sctp_ss_fcfs_add(struct sctp_tcb *stcb, struct sctp_association *asoc, 743 struct sctp_stream_out *strq, struct sctp_stream_queue_pending *sp, 744 int holds_lock) 745 { 746 if (holds_lock == 0) { 747 SCTP_TCB_SEND_LOCK(stcb); 748 } 749 if (sp && (sp->ss_next.tqe_next == NULL) && 750 (sp->ss_next.tqe_prev == NULL)) { 751 TAILQ_INSERT_TAIL(&asoc->ss_data.out_list, sp, ss_next); 752 } 753 if (holds_lock == 0) { 754 SCTP_TCB_SEND_UNLOCK(stcb); 755 } 756 return; 757 } 758 759 static int 760 sctp_ss_fcfs_is_empty(struct sctp_tcb *stcb, struct sctp_association *asoc) 761 { 762 if (TAILQ_EMPTY(&asoc->ss_data.out_list)) { 763 return (1); 764 } else { 765 return (0); 766 } 767 } 768 769 static void 770 sctp_ss_fcfs_remove(struct sctp_tcb *stcb, struct sctp_association *asoc, 771 struct sctp_stream_out *strq, struct sctp_stream_queue_pending *sp, 772 int holds_lock) 773 { 774 if (holds_lock == 0) { 775 SCTP_TCB_SEND_LOCK(stcb); 776 } 777 if (sp && 778 ((sp->ss_next.tqe_next != NULL) || 779 (sp->ss_next.tqe_prev != NULL))) { 780 TAILQ_REMOVE(&asoc->ss_data.out_list, sp, ss_next); 781 } 782 if (holds_lock == 0) { 783 SCTP_TCB_SEND_UNLOCK(stcb); 784 } 785 return; 786 } 787 788 789 static struct sctp_stream_out * 790 sctp_ss_fcfs_select(struct sctp_tcb *stcb, struct sctp_nets *net, 791 struct sctp_association *asoc) 792 { 793 struct sctp_stream_out *strq; 794 struct sctp_stream_queue_pending *sp; 795 796 sp = TAILQ_FIRST(&asoc->ss_data.out_list); 797 default_again: 798 if (sp != NULL) { 799 strq = &asoc->strmout[sp->stream]; 800 } else { 801 strq = NULL; 802 } 803 804 /* 805 * If CMT is off, we must validate that the stream in question has 806 * the first item pointed towards are network destination requested 807 * by the caller. Note that if we turn out to be locked to a stream 808 * (assigning TSN's then we must stop, since we cannot look for 809 * another stream with data to send to that destination). In CMT's 810 * case, by skipping this check, we will send one data packet 811 * towards the requested net. 812 */ 813 if (net != NULL && strq != NULL && 814 SCTP_BASE_SYSCTL(sctp_cmt_on_off) == 0) { 815 if (TAILQ_FIRST(&strq->outqueue) && 816 TAILQ_FIRST(&strq->outqueue)->net != NULL && 817 TAILQ_FIRST(&strq->outqueue)->net != net) { 818 sp = TAILQ_NEXT(sp, ss_next); 819 goto default_again; 820 } 821 } 822 return (strq); 823 } 824 825 struct sctp_ss_functions sctp_ss_functions[] = { 826 /* SCTP_SS_DEFAULT */ 827 { 828 .sctp_ss_init = sctp_ss_default_init, 829 .sctp_ss_clear = sctp_ss_default_clear, 830 .sctp_ss_init_stream = sctp_ss_default_init_stream, 831 .sctp_ss_add_to_stream = sctp_ss_default_add, 832 .sctp_ss_is_empty = sctp_ss_default_is_empty, 833 .sctp_ss_remove_from_stream = sctp_ss_default_remove, 834 .sctp_ss_select_stream = sctp_ss_default_select, 835 .sctp_ss_scheduled = sctp_ss_default_scheduled, 836 .sctp_ss_packet_done = sctp_ss_default_packet_done, 837 .sctp_ss_get_value = sctp_ss_default_get_value, 838 .sctp_ss_set_value = sctp_ss_default_set_value 839 }, 840 /* SCTP_SS_ROUND_ROBIN */ 841 { 842 .sctp_ss_init = sctp_ss_default_init, 843 .sctp_ss_clear = sctp_ss_default_clear, 844 .sctp_ss_init_stream = sctp_ss_default_init_stream, 845 .sctp_ss_add_to_stream = sctp_ss_rr_add, 846 .sctp_ss_is_empty = sctp_ss_default_is_empty, 847 .sctp_ss_remove_from_stream = sctp_ss_default_remove, 848 .sctp_ss_select_stream = sctp_ss_default_select, 849 .sctp_ss_scheduled = sctp_ss_default_scheduled, 850 .sctp_ss_packet_done = sctp_ss_default_packet_done, 851 .sctp_ss_get_value = sctp_ss_default_get_value, 852 .sctp_ss_set_value = sctp_ss_default_set_value 853 }, 854 /* SCTP_SS_ROUND_ROBIN_PACKET */ 855 { 856 .sctp_ss_init = sctp_ss_default_init, 857 .sctp_ss_clear = sctp_ss_default_clear, 858 .sctp_ss_init_stream = sctp_ss_default_init_stream, 859 .sctp_ss_add_to_stream = sctp_ss_rr_add, 860 .sctp_ss_is_empty = sctp_ss_default_is_empty, 861 .sctp_ss_remove_from_stream = sctp_ss_default_remove, 862 .sctp_ss_select_stream = sctp_ss_rrp_select, 863 .sctp_ss_scheduled = sctp_ss_default_scheduled, 864 .sctp_ss_packet_done = sctp_ss_rrp_packet_done, 865 .sctp_ss_get_value = sctp_ss_default_get_value, 866 .sctp_ss_set_value = sctp_ss_default_set_value 867 }, 868 /* SCTP_SS_PRIORITY */ 869 { 870 .sctp_ss_init = sctp_ss_default_init, 871 .sctp_ss_clear = sctp_ss_prio_clear, 872 .sctp_ss_init_stream = sctp_ss_prio_init_stream, 873 .sctp_ss_add_to_stream = sctp_ss_prio_add, 874 .sctp_ss_is_empty = sctp_ss_default_is_empty, 875 .sctp_ss_remove_from_stream = sctp_ss_prio_remove, 876 .sctp_ss_select_stream = sctp_ss_prio_select, 877 .sctp_ss_scheduled = sctp_ss_default_scheduled, 878 .sctp_ss_packet_done = sctp_ss_default_packet_done, 879 .sctp_ss_get_value = sctp_ss_prio_get_value, 880 .sctp_ss_set_value = sctp_ss_prio_set_value 881 }, 882 /* SCTP_SS_FAIR_BANDWITH */ 883 { 884 .sctp_ss_init = sctp_ss_default_init, 885 .sctp_ss_clear = sctp_ss_fb_clear, 886 .sctp_ss_init_stream = sctp_ss_fb_init_stream, 887 .sctp_ss_add_to_stream = sctp_ss_fb_add, 888 .sctp_ss_is_empty = sctp_ss_default_is_empty, 889 .sctp_ss_remove_from_stream = sctp_ss_fb_remove, 890 .sctp_ss_select_stream = sctp_ss_fb_select, 891 .sctp_ss_scheduled = sctp_ss_fb_scheduled, 892 .sctp_ss_packet_done = sctp_ss_default_packet_done, 893 .sctp_ss_get_value = sctp_ss_default_get_value, 894 .sctp_ss_set_value = sctp_ss_default_set_value 895 }, 896 /* SCTP_SS_FIRST_COME */ 897 { 898 .sctp_ss_init = sctp_ss_fcfs_init, 899 .sctp_ss_clear = sctp_ss_fcfs_clear, 900 .sctp_ss_init_stream = sctp_ss_fcfs_init_stream, 901 .sctp_ss_add_to_stream = sctp_ss_fcfs_add, 902 .sctp_ss_is_empty = sctp_ss_fcfs_is_empty, 903 .sctp_ss_remove_from_stream = sctp_ss_fcfs_remove, 904 .sctp_ss_select_stream = sctp_ss_fcfs_select, 905 .sctp_ss_scheduled = sctp_ss_default_scheduled, 906 .sctp_ss_packet_done = sctp_ss_default_packet_done, 907 .sctp_ss_get_value = sctp_ss_default_get_value, 908 .sctp_ss_set_value = sctp_ss_default_set_value 909 } 910 }; 911