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