1 2 /* 3 * ng_ppp.c 4 * 5 * Copyright (c) 1996-1999 Whistle Communications, Inc. 6 * All rights reserved. 7 * 8 * Subject to the following obligations and disclaimer of warranty, use and 9 * redistribution of this software, in source or object code forms, with or 10 * without modifications are expressly permitted by Whistle Communications; 11 * provided, however, that: 12 * 1. Any and all reproductions of the source or object code must include the 13 * copyright notice above and the following disclaimer of warranties; and 14 * 2. No rights are granted, in any manner or form, to use Whistle 15 * Communications, Inc. trademarks, including the mark "WHISTLE 16 * COMMUNICATIONS" on advertising, endorsements, or otherwise except as 17 * such appears in the above copyright notice or in the software. 18 * 19 * THIS SOFTWARE IS BEING PROVIDED BY WHISTLE COMMUNICATIONS "AS IS", AND 20 * TO THE MAXIMUM EXTENT PERMITTED BY LAW, WHISTLE COMMUNICATIONS MAKES NO 21 * REPRESENTATIONS OR WARRANTIES, EXPRESS OR IMPLIED, REGARDING THIS SOFTWARE, 22 * INCLUDING WITHOUT LIMITATION, ANY AND ALL IMPLIED WARRANTIES OF 23 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, OR NON-INFRINGEMENT. 24 * WHISTLE COMMUNICATIONS DOES NOT WARRANT, GUARANTEE, OR MAKE ANY 25 * REPRESENTATIONS REGARDING THE USE OF, OR THE RESULTS OF THE USE OF THIS 26 * SOFTWARE IN TERMS OF ITS CORRECTNESS, ACCURACY, RELIABILITY OR OTHERWISE. 27 * IN NO EVENT SHALL WHISTLE COMMUNICATIONS BE LIABLE FOR ANY DAMAGES 28 * RESULTING FROM OR ARISING OUT OF ANY USE OF THIS SOFTWARE, INCLUDING 29 * WITHOUT LIMITATION, ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, 30 * PUNITIVE, OR CONSEQUENTIAL DAMAGES, PROCUREMENT OF SUBSTITUTE GOODS OR 31 * SERVICES, LOSS OF USE, DATA OR PROFITS, HOWEVER CAUSED AND UNDER ANY 32 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 33 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF 34 * THIS SOFTWARE, EVEN IF WHISTLE COMMUNICATIONS IS ADVISED OF THE POSSIBILITY 35 * OF SUCH DAMAGE. 36 * 37 * Author: Archie Cobbs <archie@whistle.com> 38 * 39 * $FreeBSD$ 40 * $Whistle: ng_ppp.c,v 1.24 1999/11/01 09:24:52 julian Exp $ 41 */ 42 43 /* 44 * PPP node type. 45 */ 46 47 #include <sys/param.h> 48 #include <sys/systm.h> 49 #include <sys/kernel.h> 50 #include <sys/mbuf.h> 51 #include <sys/malloc.h> 52 #include <sys/errno.h> 53 #include <sys/syslog.h> 54 #include <sys/ctype.h> 55 56 #include <netgraph/ng_message.h> 57 #include <netgraph/netgraph.h> 58 #include <netgraph/ng_parse.h> 59 #include <netgraph/ng_ppp.h> 60 #include <netgraph/ng_vjc.h> 61 62 #define PROT_VALID(p) (((p) & 0x0101) == 0x0001) 63 #define PROT_COMPRESSABLE(p) (((p) & 0xff00) == 0x0000) 64 65 /* Some PPP protocol numbers we're interested in */ 66 #define PROT_APPLETALK 0x0029 67 #define PROT_COMPD 0x00fd 68 #define PROT_CRYPTD 0x0053 69 #define PROT_IP 0x0021 70 #define PROT_IPX 0x002b 71 #define PROT_LCP 0xc021 72 #define PROT_MP 0x003d 73 #define PROT_VJCOMP 0x002d 74 #define PROT_VJUNCOMP 0x002f 75 76 /* Multilink PPP definitions */ 77 #define MP_MIN_MRRU 1500 /* per RFC 1990 */ 78 #define MP_INITIAL_SEQ 0 /* per RFC 1990 */ 79 #define MP_MIN_LINK_MRU 32 80 81 #define MP_MAX_SEQ_LINGER 64 /* max frags we will hold */ 82 #define MP_INSANE_SEQ_JUMP 128 /* a sequence # jump too far */ 83 #define MP_MIN_FRAG_LEN 6 /* don't frag smaller frames */ 84 85 #define MP_SHORT_SEQ_MASK 0x00000fff /* short seq # mask */ 86 #define MP_SHORT_SEQ_HIBIT 0x00000800 /* short seq # high bit */ 87 #define MP_SHORT_FIRST_FLAG 0x00008000 /* first fragment in frame */ 88 #define MP_SHORT_LAST_FLAG 0x00004000 /* last fragment in frame */ 89 90 #define MP_LONG_SEQ_MASK 0x00ffffff /* long seq # mask */ 91 #define MP_LONG_SEQ_HIBIT 0x00800000 /* long seq # high bit */ 92 #define MP_LONG_FIRST_FLAG 0x80000000 /* first fragment in frame */ 93 #define MP_LONG_LAST_FLAG 0x40000000 /* last fragment in frame */ 94 95 #define MP_SEQ_MASK(priv) ((priv)->conf.recvShortSeq ? \ 96 MP_SHORT_SEQ_MASK : MP_LONG_SEQ_MASK) 97 98 /* Sign extension of MP sequence numbers */ 99 #define MP_SHORT_EXTEND(s) (((s) & MP_SHORT_SEQ_HIBIT) ? \ 100 ((s) | ~MP_SHORT_SEQ_MASK) : (s)) 101 #define MP_LONG_EXTEND(s) (((s) & MP_LONG_SEQ_HIBIT) ? \ 102 ((s) | ~MP_LONG_SEQ_MASK) : (s)) 103 104 /* Comparision of MP sequence numbers */ 105 #define MP_SHORT_SEQ_DIFF(x,y) (MP_SHORT_EXTEND(x) - MP_SHORT_EXTEND(y)) 106 #define MP_LONG_SEQ_DIFF(x,y) (MP_LONG_EXTEND(x) - MP_LONG_EXTEND(y)) 107 108 #define MP_SEQ_DIFF(priv,x,y) ((priv)->conf.recvShortSeq ? \ 109 MP_SHORT_SEQ_DIFF((x), (y)) : \ 110 MP_LONG_SEQ_DIFF((x), (y))) 111 112 /* We store incoming fragments this way */ 113 struct ng_ppp_frag { 114 int seq; 115 u_char first; 116 u_char last; 117 struct mbuf *data; 118 meta_p meta; 119 CIRCLEQ_ENTRY(ng_ppp_frag) f_qent; 120 }; 121 122 /* We keep track of link queue status this way */ 123 struct ng_ppp_link_qstat { 124 struct timeval lastWrite; /* time of last write */ 125 int bytesInQueue; /* bytes in the queue then */ 126 }; 127 128 /* We use integer indicies to refer to the non-link hooks */ 129 static const char *const ng_ppp_hook_names[] = { 130 NG_PPP_HOOK_ATALK, 131 #define HOOK_INDEX_ATALK 0 132 NG_PPP_HOOK_BYPASS, 133 #define HOOK_INDEX_BYPASS 1 134 NG_PPP_HOOK_COMPRESS, 135 #define HOOK_INDEX_COMPRESS 2 136 NG_PPP_HOOK_ENCRYPT, 137 #define HOOK_INDEX_ENCRYPT 3 138 NG_PPP_HOOK_DECOMPRESS, 139 #define HOOK_INDEX_DECOMPRESS 4 140 NG_PPP_HOOK_DECRYPT, 141 #define HOOK_INDEX_DECRYPT 5 142 NG_PPP_HOOK_INET, 143 #define HOOK_INDEX_INET 6 144 NG_PPP_HOOK_IPX, 145 #define HOOK_INDEX_IPX 7 146 NG_PPP_HOOK_VJC_COMP, 147 #define HOOK_INDEX_VJC_COMP 8 148 NG_PPP_HOOK_VJC_IP, 149 #define HOOK_INDEX_VJC_IP 9 150 NG_PPP_HOOK_VJC_UNCOMP, 151 #define HOOK_INDEX_VJC_UNCOMP 10 152 NG_PPP_HOOK_VJC_VJIP, 153 #define HOOK_INDEX_VJC_VJIP 11 154 NULL 155 #define HOOK_INDEX_MAX 12 156 }; 157 158 /* We store index numbers in the hook private pointer. The HOOK_INDEX() 159 for a hook is either the index (above) for normal hooks, or the ones 160 complement of the link number for link hooks. */ 161 #define HOOK_INDEX(hook) (*((int16_t *) &(hook)->private)) 162 163 /* Node private data */ 164 struct ng_ppp_private { 165 struct ng_ppp_node_config conf; 166 struct ng_ppp_link_stat bundleStats; 167 struct ng_ppp_link_stat linkStats[NG_PPP_MAX_LINKS]; 168 hook_p links[NG_PPP_MAX_LINKS]; 169 hook_p hooks[HOOK_INDEX_MAX]; 170 u_char vjCompHooked; 171 u_char allLinksEqual; 172 u_short activeLinks[NG_PPP_MAX_LINKS]; 173 u_int numActiveLinks; 174 u_int lastLink; /* for round robin */ 175 struct ng_ppp_link_qstat qstat[NG_PPP_MAX_LINKS]; 176 CIRCLEQ_HEAD(ng_ppp_fraglist, ng_ppp_frag) 177 frags; /* incoming fragments */ 178 int mpSeqOut; /* next out MP seq # */ 179 }; 180 typedef struct ng_ppp_private *priv_p; 181 182 /* Netgraph node methods */ 183 static ng_constructor_t ng_ppp_constructor; 184 static ng_rcvmsg_t ng_ppp_rcvmsg; 185 static ng_shutdown_t ng_ppp_rmnode; 186 static ng_newhook_t ng_ppp_newhook; 187 static ng_rcvdata_t ng_ppp_rcvdata; 188 static ng_disconnect_t ng_ppp_disconnect; 189 190 /* Helper functions */ 191 static int ng_ppp_input(node_p node, int bypass, 192 int linkNum, struct mbuf *m, meta_p meta); 193 static int ng_ppp_output(node_p node, int bypass, int proto, 194 int linkNum, struct mbuf *m, meta_p meta); 195 static int ng_ppp_mp_input(node_p node, int linkNum, 196 struct mbuf *m, meta_p meta); 197 static int ng_ppp_mp_output(node_p node, struct mbuf *m, meta_p meta); 198 static void ng_ppp_mp_strategy(node_p node, int len, int *distrib); 199 static int ng_ppp_intcmp(const void *v1, const void *v2); 200 static struct mbuf *ng_ppp_addproto(struct mbuf *m, int proto, int compOK); 201 static struct mbuf *ng_ppp_prepend(struct mbuf *m, const void *buf, int len); 202 static int ng_ppp_config_valid(node_p node, 203 const struct ng_ppp_node_config *newConf); 204 static void ng_ppp_update(node_p node, int newConf); 205 static void ng_ppp_free_frags(node_p node); 206 207 /* Parse type for struct ng_ppp_link_config */ 208 static const struct ng_parse_struct_info 209 ng_ppp_link_type_info = NG_PPP_LINK_TYPE_INFO; 210 static const struct ng_parse_type ng_ppp_link_type = { 211 &ng_parse_struct_type, 212 &ng_ppp_link_type_info, 213 }; 214 215 /* Parse type for struct ng_ppp_node_config */ 216 struct ng_parse_fixedarray_info ng_ppp_array_info = { 217 &ng_ppp_link_type, 218 NG_PPP_MAX_LINKS 219 }; 220 static const struct ng_parse_type ng_ppp_link_array_type = { 221 &ng_parse_fixedarray_type, 222 &ng_ppp_array_info, 223 }; 224 static const struct ng_parse_struct_info ng_ppp_config_type_info 225 = NG_PPP_CONFIG_TYPE_INFO(&ng_ppp_link_array_type); 226 static const struct ng_parse_type ng_ppp_config_type = { 227 &ng_parse_struct_type, 228 &ng_ppp_config_type_info 229 }; 230 231 /* Parse type for struct ng_ppp_link_stat */ 232 static const struct ng_parse_struct_info 233 ng_ppp_stats_type_info = NG_PPP_STATS_TYPE_INFO; 234 static const struct ng_parse_type ng_ppp_stats_type = { 235 &ng_parse_struct_type, 236 &ng_ppp_stats_type_info 237 }; 238 239 /* List of commands and how to convert arguments to/from ASCII */ 240 static const struct ng_cmdlist ng_ppp_cmds[] = { 241 { 242 NGM_PPP_COOKIE, 243 NGM_PPP_SET_CONFIG, 244 "setconfig", 245 &ng_ppp_config_type, 246 NULL 247 }, 248 { 249 NGM_PPP_COOKIE, 250 NGM_PPP_GET_CONFIG, 251 "getconfig", 252 NULL, 253 &ng_ppp_config_type 254 }, 255 { 256 NGM_PPP_COOKIE, 257 NGM_PPP_GET_LINK_STATS, 258 "getstats", 259 &ng_parse_int16_type, 260 &ng_ppp_stats_type 261 }, 262 { 263 NGM_PPP_COOKIE, 264 NGM_PPP_CLR_LINK_STATS, 265 "clrstats", 266 &ng_parse_int16_type, 267 NULL 268 }, 269 { 270 NGM_PPP_COOKIE, 271 NGM_PPP_GETCLR_LINK_STATS, 272 "getclrstats", 273 &ng_parse_int16_type, 274 &ng_ppp_stats_type 275 }, 276 { 0 } 277 }; 278 279 /* Node type descriptor */ 280 static struct ng_type ng_ppp_typestruct = { 281 NG_VERSION, 282 NG_PPP_NODE_TYPE, 283 NULL, 284 ng_ppp_constructor, 285 ng_ppp_rcvmsg, 286 ng_ppp_rmnode, 287 ng_ppp_newhook, 288 NULL, 289 NULL, 290 ng_ppp_rcvdata, 291 ng_ppp_rcvdata, 292 ng_ppp_disconnect, 293 ng_ppp_cmds 294 }; 295 NETGRAPH_INIT(ppp, &ng_ppp_typestruct); 296 297 static int *compareLatencies; /* hack for ng_ppp_intcmp() */ 298 299 /* Address and control field header */ 300 static const u_char ng_ppp_acf[2] = { 0xff, 0x03 }; 301 302 #define ERROUT(x) do { error = (x); goto done; } while (0) 303 304 /************************************************************************ 305 NETGRAPH NODE STUFF 306 ************************************************************************/ 307 308 /* 309 * Node type constructor 310 */ 311 static int 312 ng_ppp_constructor(node_p *nodep) 313 { 314 priv_p priv; 315 int error; 316 317 /* Allocate private structure */ 318 MALLOC(priv, priv_p, sizeof(*priv), M_NETGRAPH, M_WAITOK); 319 if (priv == NULL) 320 return (ENOMEM); 321 bzero(priv, sizeof(*priv)); 322 323 /* Call generic node constructor */ 324 if ((error = ng_make_node_common(&ng_ppp_typestruct, nodep))) { 325 FREE(priv, M_NETGRAPH); 326 return (error); 327 } 328 (*nodep)->private = priv; 329 330 /* Initialize state */ 331 CIRCLEQ_INIT(&priv->frags); 332 333 /* Done */ 334 return (0); 335 } 336 337 /* 338 * Give our OK for a hook to be added 339 */ 340 static int 341 ng_ppp_newhook(node_p node, hook_p hook, const char *name) 342 { 343 const priv_p priv = node->private; 344 int linkNum = -1; 345 hook_p *hookPtr = NULL; 346 int hookIndex = -1; 347 348 /* Figure out which hook it is */ 349 if (strncmp(name, NG_PPP_HOOK_LINK_PREFIX, /* a link hook? */ 350 strlen(NG_PPP_HOOK_LINK_PREFIX)) == 0) { 351 const char *cp; 352 char *eptr; 353 354 cp = name + strlen(NG_PPP_HOOK_LINK_PREFIX); 355 if (!isdigit(*cp) || (cp[0] == '0' && cp[1] != '\0')) 356 return (EINVAL); 357 linkNum = (int)strtoul(cp, &eptr, 10); 358 if (*eptr != '\0' || linkNum < 0 || linkNum >= NG_PPP_MAX_LINKS) 359 return (EINVAL); 360 hookPtr = &priv->links[linkNum]; 361 hookIndex = ~linkNum; 362 } else { /* must be a non-link hook */ 363 int i; 364 365 for (i = 0; ng_ppp_hook_names[i] != NULL; i++) { 366 if (strcmp(name, ng_ppp_hook_names[i]) == 0) { 367 hookPtr = &priv->hooks[i]; 368 hookIndex = i; 369 break; 370 } 371 } 372 if (ng_ppp_hook_names[i] == NULL) 373 return (EINVAL); /* no such hook */ 374 } 375 376 /* See if hook is already connected */ 377 if (*hookPtr != NULL) 378 return (EISCONN); 379 380 /* Disallow more than one link unless multilink is enabled */ 381 if (linkNum != -1 && priv->conf.links[linkNum].enableLink 382 && !priv->conf.enableMultilink && priv->numActiveLinks >= 1) 383 return (ENODEV); 384 385 /* OK */ 386 *hookPtr = hook; 387 HOOK_INDEX(hook) = hookIndex; 388 ng_ppp_update(node, 0); 389 return (0); 390 } 391 392 /* 393 * Receive a control message 394 */ 395 static int 396 ng_ppp_rcvmsg(node_p node, struct ng_mesg *msg, 397 const char *raddr, struct ng_mesg **rptr) 398 { 399 const priv_p priv = node->private; 400 struct ng_mesg *resp = NULL; 401 int error = 0; 402 403 switch (msg->header.typecookie) { 404 case NGM_PPP_COOKIE: 405 switch (msg->header.cmd) { 406 case NGM_PPP_SET_CONFIG: 407 { 408 struct ng_ppp_node_config *const newConf = 409 (struct ng_ppp_node_config *) msg->data; 410 411 /* Check for invalid or illegal config */ 412 if (msg->header.arglen != sizeof(*newConf)) 413 ERROUT(EINVAL); 414 if (!ng_ppp_config_valid(node, newConf)) 415 ERROUT(EINVAL); 416 priv->conf = *newConf; 417 ng_ppp_update(node, 1); 418 break; 419 } 420 case NGM_PPP_GET_CONFIG: 421 NG_MKRESPONSE(resp, msg, sizeof(priv->conf), M_NOWAIT); 422 if (resp == NULL) 423 ERROUT(ENOMEM); 424 bcopy(&priv->conf, resp->data, sizeof(priv->conf)); 425 break; 426 case NGM_PPP_GET_LINK_STATS: 427 case NGM_PPP_CLR_LINK_STATS: 428 case NGM_PPP_GETCLR_LINK_STATS: 429 { 430 struct ng_ppp_link_stat *stats; 431 u_int16_t linkNum; 432 433 if (msg->header.arglen != sizeof(u_int16_t)) 434 ERROUT(EINVAL); 435 linkNum = *((u_int16_t *) msg->data); 436 if (linkNum >= NG_PPP_MAX_LINKS 437 && linkNum != NG_PPP_BUNDLE_LINKNUM) 438 ERROUT(EINVAL); 439 stats = (linkNum == NG_PPP_BUNDLE_LINKNUM) ? 440 &priv->bundleStats : &priv->linkStats[linkNum]; 441 if (msg->header.cmd != NGM_PPP_CLR_LINK_STATS) { 442 NG_MKRESPONSE(resp, msg, 443 sizeof(struct ng_ppp_link_stat), M_NOWAIT); 444 if (resp == NULL) 445 ERROUT(ENOMEM); 446 bcopy(stats, resp->data, sizeof(*stats)); 447 } 448 if (msg->header.cmd != NGM_PPP_GET_LINK_STATS) 449 bzero(stats, sizeof(*stats)); 450 break; 451 } 452 default: 453 error = EINVAL; 454 break; 455 } 456 break; 457 case NGM_VJC_COOKIE: 458 { 459 char path[NG_PATHLEN + 1]; 460 node_p origNode; 461 462 if ((error = ng_path2node(node, raddr, &origNode, NULL)) != 0) 463 ERROUT(error); 464 snprintf(path, sizeof(path), "[%lx]:%s", 465 (long) node, NG_PPP_HOOK_VJC_IP); 466 return ng_send_msg(origNode, msg, path, rptr); 467 break; 468 } 469 default: 470 error = EINVAL; 471 break; 472 } 473 if (rptr) 474 *rptr = resp; 475 else if (resp) 476 FREE(resp, M_NETGRAPH); 477 478 done: 479 FREE(msg, M_NETGRAPH); 480 return (error); 481 } 482 483 /* 484 * Receive data on a hook 485 */ 486 static int 487 ng_ppp_rcvdata(hook_p hook, struct mbuf *m, meta_p meta) 488 { 489 const node_p node = hook->node; 490 const priv_p priv = node->private; 491 const int index = HOOK_INDEX(hook); 492 u_int16_t linkNum = NG_PPP_BUNDLE_LINKNUM; 493 hook_p outHook = NULL; 494 int proto = 0, error; 495 496 /* Did it come from a link hook? */ 497 if (index < 0) { 498 499 /* Convert index into a link number */ 500 linkNum = (u_int16_t)~index; 501 KASSERT(linkNum < NG_PPP_MAX_LINKS, 502 ("%s: bogus index 0x%x", __FUNCTION__, index)); 503 504 /* Stats */ 505 priv->linkStats[linkNum].recvFrames++; 506 priv->linkStats[linkNum].recvOctets += m->m_pkthdr.len; 507 508 /* Strip address and control fields, if present */ 509 if (m->m_pkthdr.len >= 2) { 510 if (m->m_len < 2 && (m = m_pullup(m, 2)) == NULL) { 511 NG_FREE_DATA(m, meta); 512 return (ENOBUFS); 513 } 514 if (bcmp(mtod(m, u_char *), &ng_ppp_acf, 2) == 0) 515 m_adj(m, 2); 516 } 517 518 /* Dispatch incoming frame (if not enabled, to bypass) */ 519 return ng_ppp_input(node, 520 !priv->conf.links[linkNum].enableLink, linkNum, m, meta); 521 } 522 523 /* Get protocol & check if data allowed from this hook */ 524 switch (index) { 525 526 /* Outgoing data */ 527 case HOOK_INDEX_ATALK: 528 if (!priv->conf.enableAtalk) { 529 NG_FREE_DATA(m, meta); 530 return (ENXIO); 531 } 532 proto = PROT_APPLETALK; 533 break; 534 case HOOK_INDEX_IPX: 535 if (!priv->conf.enableIPX) { 536 NG_FREE_DATA(m, meta); 537 return (ENXIO); 538 } 539 proto = PROT_IPX; 540 break; 541 case HOOK_INDEX_INET: 542 case HOOK_INDEX_VJC_VJIP: 543 if (!priv->conf.enableIP) { 544 NG_FREE_DATA(m, meta); 545 return (ENXIO); 546 } 547 proto = PROT_IP; 548 break; 549 case HOOK_INDEX_VJC_COMP: 550 if (!priv->conf.enableVJCompression) { 551 NG_FREE_DATA(m, meta); 552 return (ENXIO); 553 } 554 proto = PROT_VJCOMP; 555 break; 556 case HOOK_INDEX_VJC_UNCOMP: 557 if (!priv->conf.enableVJCompression) { 558 NG_FREE_DATA(m, meta); 559 return (ENXIO); 560 } 561 proto = PROT_VJUNCOMP; 562 break; 563 case HOOK_INDEX_COMPRESS: 564 if (!priv->conf.enableCompression) { 565 NG_FREE_DATA(m, meta); 566 return (ENXIO); 567 } 568 proto = PROT_COMPD; 569 break; 570 case HOOK_INDEX_ENCRYPT: 571 if (!priv->conf.enableEncryption) { 572 NG_FREE_DATA(m, meta); 573 return (ENXIO); 574 } 575 proto = PROT_CRYPTD; 576 break; 577 case HOOK_INDEX_BYPASS: 578 if (m->m_pkthdr.len < 4) { 579 NG_FREE_DATA(m, meta); 580 return (EINVAL); 581 } 582 if (m->m_len < 4 && (m = m_pullup(m, 4)) == NULL) { 583 NG_FREE_META(meta); 584 return (ENOBUFS); 585 } 586 linkNum = ntohs(mtod(m, u_int16_t *)[0]); 587 proto = ntohs(mtod(m, u_int16_t *)[1]); 588 m_adj(m, 4); 589 if (linkNum >= NG_PPP_MAX_LINKS 590 && linkNum != NG_PPP_BUNDLE_LINKNUM) { 591 NG_FREE_DATA(m, meta); 592 return (EINVAL); 593 } 594 break; 595 596 /* Incoming data */ 597 case HOOK_INDEX_VJC_IP: 598 if (!priv->conf.enableIP || !priv->conf.enableVJDecompression) { 599 NG_FREE_DATA(m, meta); 600 return (ENXIO); 601 } 602 break; 603 case HOOK_INDEX_DECOMPRESS: 604 if (!priv->conf.enableDecompression) { 605 NG_FREE_DATA(m, meta); 606 return (ENXIO); 607 } 608 break; 609 case HOOK_INDEX_DECRYPT: 610 if (!priv->conf.enableDecryption) { 611 NG_FREE_DATA(m, meta); 612 return (ENXIO); 613 } 614 break; 615 default: 616 panic("%s: bogus index 0x%x", __FUNCTION__, index); 617 } 618 619 /* Now figure out what to do with the frame */ 620 switch (index) { 621 622 /* Outgoing data */ 623 case HOOK_INDEX_INET: 624 if (priv->conf.enableVJCompression && priv->vjCompHooked) { 625 outHook = priv->hooks[HOOK_INDEX_VJC_IP]; 626 break; 627 } 628 /* FALLTHROUGH */ 629 case HOOK_INDEX_ATALK: 630 case HOOK_INDEX_IPX: 631 case HOOK_INDEX_VJC_COMP: 632 case HOOK_INDEX_VJC_UNCOMP: 633 case HOOK_INDEX_VJC_VJIP: 634 if (priv->conf.enableCompression 635 && priv->hooks[HOOK_INDEX_COMPRESS] != NULL) { 636 if ((m = ng_ppp_addproto(m, proto, 1)) == NULL) { 637 NG_FREE_META(meta); 638 return (ENOBUFS); 639 } 640 outHook = priv->hooks[HOOK_INDEX_COMPRESS]; 641 break; 642 } 643 /* FALLTHROUGH */ 644 case HOOK_INDEX_COMPRESS: 645 if (priv->conf.enableEncryption 646 && priv->hooks[HOOK_INDEX_ENCRYPT] != NULL) { 647 if ((m = ng_ppp_addproto(m, proto, 1)) == NULL) { 648 NG_FREE_META(meta); 649 return (ENOBUFS); 650 } 651 outHook = priv->hooks[HOOK_INDEX_ENCRYPT]; 652 break; 653 } 654 /* FALLTHROUGH */ 655 case HOOK_INDEX_ENCRYPT: 656 return ng_ppp_output(node, 0, 657 proto, NG_PPP_BUNDLE_LINKNUM, m, meta); 658 659 case HOOK_INDEX_BYPASS: 660 return ng_ppp_output(node, 1, proto, linkNum, m, meta); 661 662 /* Incoming data */ 663 case HOOK_INDEX_DECRYPT: 664 case HOOK_INDEX_DECOMPRESS: 665 return ng_ppp_input(node, 0, NG_PPP_BUNDLE_LINKNUM, m, meta); 666 667 case HOOK_INDEX_VJC_IP: 668 outHook = priv->hooks[HOOK_INDEX_INET]; 669 break; 670 } 671 672 /* Send packet out hook */ 673 NG_SEND_DATA(error, outHook, m, meta); 674 return (error); 675 } 676 677 /* 678 * Destroy node 679 */ 680 static int 681 ng_ppp_rmnode(node_p node) 682 { 683 const priv_p priv = node->private; 684 685 /* Take down netgraph node */ 686 node->flags |= NG_INVALID; 687 ng_cutlinks(node); 688 ng_unname(node); 689 ng_ppp_free_frags(node); 690 bzero(priv, sizeof(*priv)); 691 FREE(priv, M_NETGRAPH); 692 node->private = NULL; 693 ng_unref(node); /* let the node escape */ 694 return (0); 695 } 696 697 /* 698 * Hook disconnection 699 */ 700 static int 701 ng_ppp_disconnect(hook_p hook) 702 { 703 const node_p node = hook->node; 704 const priv_p priv = node->private; 705 const int index = HOOK_INDEX(hook); 706 707 /* Zero out hook pointer */ 708 if (index < 0) 709 priv->links[~index] = NULL; 710 else 711 priv->hooks[index] = NULL; 712 713 /* Update derived info (or go away if no hooks left) */ 714 if (node->numhooks > 0) 715 ng_ppp_update(node, 0); 716 else 717 ng_rmnode(hook->node); 718 return (0); 719 } 720 721 /************************************************************************ 722 HELPER STUFF 723 ************************************************************************/ 724 725 /* 726 * Handle an incoming frame. Extract the PPP protocol number 727 * and dispatch accordingly. 728 */ 729 static int 730 ng_ppp_input(node_p node, int bypass, int linkNum, struct mbuf *m, meta_p meta) 731 { 732 const priv_p priv = node->private; 733 hook_p outHook = NULL; 734 int proto, error; 735 736 /* Extract protocol number */ 737 for (proto = 0; !PROT_VALID(proto) && m->m_pkthdr.len > 0; ) { 738 if (m->m_len < 1 && (m = m_pullup(m, 1)) == NULL) { 739 NG_FREE_META(meta); 740 return (ENOBUFS); 741 } 742 proto = (proto << 8) + *mtod(m, u_char *); 743 m_adj(m, 1); 744 } 745 if (!PROT_VALID(proto)) { 746 if (linkNum == NG_PPP_BUNDLE_LINKNUM) 747 priv->bundleStats.badProtos++; 748 else 749 priv->linkStats[linkNum].badProtos++; 750 NG_FREE_DATA(m, meta); 751 return (EINVAL); 752 } 753 754 /* Bypass frame? */ 755 if (bypass) 756 goto bypass; 757 758 /* Check protocol */ 759 switch (proto) { 760 case PROT_COMPD: 761 if (priv->conf.enableDecompression) 762 outHook = priv->hooks[HOOK_INDEX_DECOMPRESS]; 763 break; 764 case PROT_CRYPTD: 765 if (priv->conf.enableDecryption) 766 outHook = priv->hooks[HOOK_INDEX_DECRYPT]; 767 break; 768 case PROT_VJCOMP: 769 if (priv->conf.enableVJDecompression && priv->vjCompHooked) 770 outHook = priv->hooks[HOOK_INDEX_VJC_COMP]; 771 break; 772 case PROT_VJUNCOMP: 773 if (priv->conf.enableVJDecompression && priv->vjCompHooked) 774 outHook = priv->hooks[HOOK_INDEX_VJC_UNCOMP]; 775 break; 776 case PROT_MP: 777 if (priv->conf.enableMultilink) 778 return ng_ppp_mp_input(node, linkNum, m, meta); 779 break; 780 case PROT_APPLETALK: 781 if (priv->conf.enableAtalk) 782 outHook = priv->hooks[HOOK_INDEX_ATALK]; 783 break; 784 case PROT_IPX: 785 if (priv->conf.enableIPX) 786 outHook = priv->hooks[HOOK_INDEX_IPX]; 787 break; 788 case PROT_IP: 789 if (priv->conf.enableIP) 790 outHook = priv->hooks[HOOK_INDEX_INET]; 791 break; 792 } 793 794 bypass: 795 /* For unknown/inactive protocols, forward out the bypass hook */ 796 if (outHook == NULL) { 797 u_int16_t hdr[2]; 798 799 hdr[0] = htons(linkNum); 800 hdr[1] = htons((u_int16_t)proto); 801 if ((m = ng_ppp_prepend(m, &hdr, 4)) == NULL) { 802 NG_FREE_META(meta); 803 return (ENOBUFS); 804 } 805 outHook = priv->hooks[HOOK_INDEX_BYPASS]; 806 } 807 808 /* Forward frame */ 809 NG_SEND_DATA(error, outHook, m, meta); 810 return (error); 811 } 812 813 /* 814 * Deliver a frame out a link, either a real one or NG_PPP_BUNDLE_LINKNUM 815 * If the link is not enabled then ENXIO is returned, unless "bypass" is != 0. 816 */ 817 static int 818 ng_ppp_output(node_p node, int bypass, 819 int proto, int linkNum, struct mbuf *m, meta_p meta) 820 { 821 const priv_p priv = node->private; 822 int len, error; 823 824 /* If not doing MP, map bundle virtual link to (the only) link */ 825 if (linkNum == NG_PPP_BUNDLE_LINKNUM && !priv->conf.enableMultilink) 826 linkNum = priv->activeLinks[0]; 827 828 /* Check link status (if real) */ 829 if (linkNum != NG_PPP_BUNDLE_LINKNUM) { 830 if (!bypass && !priv->conf.links[linkNum].enableLink) { 831 NG_FREE_DATA(m, meta); 832 return (ENXIO); 833 } 834 if (priv->links[linkNum] == NULL) { 835 NG_FREE_DATA(m, meta); 836 return (ENETDOWN); 837 } 838 } 839 840 /* Prepend protocol number, possibly compressed */ 841 if ((m = ng_ppp_addproto(m, proto, 842 linkNum == NG_PPP_BUNDLE_LINKNUM 843 || priv->conf.links[linkNum].enableProtoComp)) == NULL) { 844 NG_FREE_META(meta); 845 return (ENOBUFS); 846 } 847 848 /* Special handling for the MP virtual link */ 849 if (linkNum == NG_PPP_BUNDLE_LINKNUM) 850 return ng_ppp_mp_output(node, m, meta); 851 852 /* Prepend address and control field (unless compressed) */ 853 if (proto == PROT_LCP || !priv->conf.links[linkNum].enableACFComp) { 854 if ((m = ng_ppp_prepend(m, &ng_ppp_acf, 2)) == NULL) { 855 NG_FREE_META(meta); 856 return (ENOBUFS); 857 } 858 } 859 860 /* Deliver frame */ 861 len = m->m_pkthdr.len; 862 NG_SEND_DATA(error, priv->links[linkNum], m, meta); 863 864 /* Update stats and 'bytes in queue' counter */ 865 if (error == 0) { 866 priv->linkStats[linkNum].xmitFrames++; 867 priv->linkStats[linkNum].xmitOctets += len; 868 priv->qstat[linkNum].bytesInQueue += len; 869 microtime(&priv->qstat[linkNum].lastWrite); 870 } 871 return error; 872 } 873 874 /* 875 * Handle an incoming multi-link fragment 876 */ 877 static int 878 ng_ppp_mp_input(node_p node, int linkNum, struct mbuf *m, meta_p meta) 879 { 880 const priv_p priv = node->private; 881 struct ng_ppp_frag frag0, *frag = &frag0; 882 struct ng_ppp_frag *qent, *qnext; 883 struct ng_ppp_frag *first, *last; 884 int diff, highSeq, nextSeq, inserted; 885 struct mbuf *tail; 886 887 /* Extract fragment information from MP header */ 888 if (priv->conf.recvShortSeq) { 889 u_int16_t shdr; 890 891 if (m->m_pkthdr.len < 2) { 892 NG_FREE_DATA(m, meta); 893 return (EINVAL); 894 } 895 if (m->m_len < 2 && (m = m_pullup(m, 2)) == NULL) { 896 NG_FREE_META(meta); 897 return (ENOBUFS); 898 } 899 shdr = ntohs(*mtod(m, u_int16_t *)); 900 frag->seq = shdr & MP_SHORT_SEQ_MASK; 901 frag->first = (shdr & MP_SHORT_FIRST_FLAG) != 0; 902 frag->last = (shdr & MP_SHORT_LAST_FLAG) != 0; 903 highSeq = CIRCLEQ_EMPTY(&priv->frags) ? 904 frag->seq : CIRCLEQ_FIRST(&priv->frags)->seq; 905 diff = MP_SHORT_SEQ_DIFF(frag->seq, highSeq); 906 m_adj(m, 2); 907 } else { 908 u_int32_t lhdr; 909 910 if (m->m_pkthdr.len < 4) { 911 NG_FREE_DATA(m, meta); 912 return (EINVAL); 913 } 914 if (m->m_len < 4 && (m = m_pullup(m, 4)) == NULL) { 915 NG_FREE_META(meta); 916 return (ENOBUFS); 917 } 918 lhdr = ntohl(*mtod(m, u_int32_t *)); 919 frag->seq = lhdr & MP_LONG_SEQ_MASK; 920 frag->first = (lhdr & MP_LONG_FIRST_FLAG) != 0; 921 frag->last = (lhdr & MP_LONG_LAST_FLAG) != 0; 922 highSeq = CIRCLEQ_EMPTY(&priv->frags) ? 923 frag->seq : CIRCLEQ_FIRST(&priv->frags)->seq; 924 diff = MP_LONG_SEQ_DIFF(frag->seq, highSeq); 925 m_adj(m, 4); 926 } 927 frag->data = m; 928 frag->meta = meta; 929 930 /* If the sequence number makes a large jump, empty the queue */ 931 if (diff <= -MP_INSANE_SEQ_JUMP || diff >= MP_INSANE_SEQ_JUMP) 932 ng_ppp_free_frags(node); 933 934 /* Optimization: handle a frame that's all in one fragment */ 935 if (frag->first && frag->last) 936 return ng_ppp_input(node, 0, NG_PPP_BUNDLE_LINKNUM, m, meta); 937 938 /* Allocate a new frag struct for the queue */ 939 MALLOC(frag, struct ng_ppp_frag *, sizeof(*frag), M_NETGRAPH, M_NOWAIT); 940 if (frag == NULL) { 941 NG_FREE_DATA(m, meta); 942 return (ENOMEM); 943 } 944 *frag = frag0; 945 meta = NULL; 946 m = NULL; 947 948 /* Add fragment to queue, which is reverse sorted by sequence number */ 949 inserted = 0; 950 CIRCLEQ_FOREACH(qent, &priv->frags, f_qent) { 951 diff = MP_SEQ_DIFF(priv, frag->seq, qent->seq); 952 if (diff > 0) { 953 CIRCLEQ_INSERT_BEFORE(&priv->frags, qent, frag, f_qent); 954 inserted = 1; 955 break; 956 } else if (diff == 0) { /* should never happen! */ 957 log(LOG_ERR, "%s: rec'd dup MP fragment\n", node->name); 958 if (linkNum == NG_PPP_BUNDLE_LINKNUM) 959 priv->linkStats[linkNum].dupFragments++; 960 else 961 priv->bundleStats.dupFragments++; 962 NG_FREE_DATA(frag->data, frag->meta); 963 FREE(frag, M_NETGRAPH); 964 return (EINVAL); 965 } 966 } 967 if (!inserted) 968 CIRCLEQ_INSERT_TAIL(&priv->frags, frag, f_qent); 969 970 /* Find the last fragment in the possibly newly completed frame */ 971 last = NULL; 972 qent = frag; 973 nextSeq = frag->seq; 974 while (qent != (void *)&priv->frags && qent->seq == nextSeq) { 975 if (qent->last) { 976 last = qent; 977 break; 978 } 979 qent = CIRCLEQ_PREV(qent, f_qent); 980 nextSeq = (nextSeq + 1) & MP_SEQ_MASK(priv); 981 } 982 if (last == NULL) 983 goto incomplete; 984 985 /* Find the first fragment in the possibly newly completed frame */ 986 first = NULL; 987 qent = frag; 988 nextSeq = frag->seq; 989 while (qent != (void *)&priv->frags && qent->seq == nextSeq) { 990 if (qent->first) { 991 first = qent; 992 break; 993 } 994 qent = CIRCLEQ_NEXT(qent, f_qent); 995 nextSeq = (nextSeq - 1) & MP_SEQ_MASK(priv); 996 } 997 if (first == NULL) 998 goto incomplete; 999 1000 /* We have a complete frame, extract it from the queue */ 1001 for (tail = NULL, qent = first; qent != NULL; qent = qnext) { 1002 qnext = CIRCLEQ_PREV(qent, f_qent); 1003 CIRCLEQ_REMOVE(&priv->frags, qent, f_qent); 1004 if (tail == NULL) { 1005 tail = m = qent->data; 1006 meta = qent->meta; /* inherit first frag's meta */ 1007 } else { 1008 m->m_pkthdr.len += qent->data->m_pkthdr.len; 1009 tail->m_next = qent->data; 1010 NG_FREE_META(qent->meta); /* drop other frags' metas */ 1011 } 1012 while (tail->m_next != NULL) 1013 tail = tail->m_next; 1014 if (qent == last) 1015 qnext = NULL; 1016 FREE(qent, M_NETGRAPH); 1017 } 1018 1019 incomplete: 1020 /* Prune out stale entries in the queue */ 1021 for (qent = CIRCLEQ_LAST(&priv->frags); 1022 qent != (void *)&priv->frags; qent = qnext) { 1023 if (MP_SEQ_DIFF(priv, highSeq, qent->seq) <= MP_MAX_SEQ_LINGER) 1024 break; 1025 qnext = CIRCLEQ_PREV(qent, f_qent); 1026 CIRCLEQ_REMOVE(&priv->frags, qent, f_qent); 1027 NG_FREE_DATA(qent->data, qent->meta); 1028 FREE(qent, M_NETGRAPH); 1029 } 1030 1031 /* Deliver newly completed frame, if any */ 1032 return m ? ng_ppp_input(node, 0, NG_PPP_BUNDLE_LINKNUM, m, meta) : 0; 1033 } 1034 1035 /* 1036 * Deliver a frame out on the bundle, i.e., figure out how to fragment 1037 * the frame across the individual PPP links and do so. 1038 */ 1039 static int 1040 ng_ppp_mp_output(node_p node, struct mbuf *m, meta_p meta) 1041 { 1042 const priv_p priv = node->private; 1043 int distrib[NG_PPP_MAX_LINKS]; 1044 int firstFragment; 1045 int activeLinkNum; 1046 1047 /* At least one link must be active */ 1048 if (priv->numActiveLinks == 0) { 1049 NG_FREE_DATA(m, meta); 1050 return (ENETDOWN); 1051 } 1052 1053 /* Round-robin strategy */ 1054 if (priv->conf.enableRoundRobin || m->m_pkthdr.len < MP_MIN_FRAG_LEN) { 1055 activeLinkNum = priv->lastLink++ % priv->numActiveLinks; 1056 bzero(&distrib, priv->numActiveLinks * sizeof(distrib[0])); 1057 distrib[activeLinkNum] = m->m_pkthdr.len; 1058 goto deliver; 1059 } 1060 1061 /* Strategy when all links are equivalent (optimize the common case) */ 1062 if (priv->allLinksEqual) { 1063 const int fraction = m->m_pkthdr.len / priv->numActiveLinks; 1064 int i, remain; 1065 1066 for (i = 0; i < priv->numActiveLinks; i++) 1067 distrib[priv->lastLink++ % priv->numActiveLinks] 1068 = fraction; 1069 remain = m->m_pkthdr.len - (fraction * priv->numActiveLinks); 1070 while (remain > 0) { 1071 distrib[priv->lastLink++ % priv->numActiveLinks]++; 1072 remain--; 1073 } 1074 goto deliver; 1075 } 1076 1077 /* Strategy when all links are not equivalent */ 1078 ng_ppp_mp_strategy(node, m->m_pkthdr.len, distrib); 1079 1080 deliver: 1081 /* Update stats */ 1082 priv->bundleStats.xmitFrames++; 1083 priv->bundleStats.xmitOctets += m->m_pkthdr.len; 1084 1085 /* Send alloted portions of frame out on the link(s) */ 1086 for (firstFragment = 1, activeLinkNum = priv->numActiveLinks - 1; 1087 activeLinkNum >= 0; activeLinkNum--) { 1088 const int linkNum = priv->activeLinks[activeLinkNum]; 1089 1090 /* Deliver fragment(s) out the next link */ 1091 for ( ; distrib[activeLinkNum] > 0; firstFragment = 0) { 1092 int len, lastFragment, error; 1093 struct mbuf *m2; 1094 meta_p meta2; 1095 1096 /* Calculate fragment length; don't exceed link MTU */ 1097 len = distrib[activeLinkNum]; 1098 if (len > priv->conf.links[linkNum].mru) 1099 len = priv->conf.links[linkNum].mru; 1100 distrib[activeLinkNum] -= len; 1101 lastFragment = (len == m->m_pkthdr.len); 1102 1103 /* Split off next fragment as "m2" */ 1104 m2 = m; 1105 if (!lastFragment) { 1106 struct mbuf *n = m_split(m, len, M_NOWAIT); 1107 1108 if (n == NULL) { 1109 NG_FREE_DATA(m, meta); 1110 return (ENOMEM); 1111 } 1112 m = n; 1113 } 1114 1115 /* Prepend MP header */ 1116 if (priv->conf.xmitShortSeq) { 1117 u_int16_t shdr; 1118 1119 shdr = priv->mpSeqOut; 1120 priv->mpSeqOut = 1121 (priv->mpSeqOut + 1) % MP_SHORT_SEQ_MASK; 1122 if (firstFragment) 1123 shdr |= MP_SHORT_FIRST_FLAG; 1124 if (lastFragment) 1125 shdr |= MP_SHORT_LAST_FLAG; 1126 shdr = htons(shdr); 1127 m2 = ng_ppp_prepend(m2, &shdr, 2); 1128 } else { 1129 u_int32_t lhdr; 1130 1131 lhdr = priv->mpSeqOut; 1132 priv->mpSeqOut = 1133 (priv->mpSeqOut + 1) % MP_LONG_SEQ_MASK; 1134 if (firstFragment) 1135 lhdr |= MP_LONG_FIRST_FLAG; 1136 if (lastFragment) 1137 lhdr |= MP_LONG_LAST_FLAG; 1138 lhdr = htonl(lhdr); 1139 m2 = ng_ppp_prepend(m2, &lhdr, 4); 1140 } 1141 if (m2 == NULL) { 1142 if (!lastFragment) 1143 m_freem(m); 1144 NG_FREE_META(meta); 1145 return (ENOBUFS); 1146 } 1147 1148 /* Copy the meta information, if any */ 1149 if (meta != NULL && !lastFragment) { 1150 MALLOC(meta2, meta_p, 1151 meta->used_len, M_NETGRAPH, M_NOWAIT); 1152 if (meta2 == NULL) { 1153 m_freem(m2); 1154 NG_FREE_DATA(m, meta); 1155 return (ENOMEM); 1156 } 1157 meta2->allocated_len = meta->used_len; 1158 bcopy(meta, meta2, meta->used_len); 1159 } else 1160 meta2 = meta; 1161 1162 /* Send fragment */ 1163 error = ng_ppp_output(node, 0, 1164 PROT_MP, linkNum, m2, meta2); 1165 if (error != 0) { 1166 if (!lastFragment) 1167 NG_FREE_DATA(m, meta); 1168 return (error); 1169 } 1170 } 1171 } 1172 1173 /* Done */ 1174 return (0); 1175 } 1176 1177 /* 1178 * Computing the optimal fragmentation 1179 * ----------------------------------- 1180 * 1181 * This routine tries to compute the optimal fragmentation pattern based 1182 * on each link's latency, bandwidth, and calculated additional latency. 1183 * The latter quantity is the additional latency caused by previously 1184 * written data that has not been transmitted yet. 1185 * 1186 * This algorithm is only useful when not all of the links have the 1187 * same latency and bandwidth values. 1188 * 1189 * The essential idea is to make the last bit of each fragment of the 1190 * frame arrive at the opposite end at the exact same time. This greedy 1191 * algorithm is optimal, in that no other scheduling could result in any 1192 * packet arriving any sooner unless packets are delivered out of order. 1193 * 1194 * Suppose link i has bandwidth b_i (in tens of bytes per milisecond) and 1195 * latency l_i (in miliseconds). Consider the function function f_i(t) 1196 * which is equal to the number of bytes that will have arrived at 1197 * the peer after t miliseconds if we start writing continuously at 1198 * time t = 0. Then f_i(t) = b_i * (t - l_i) = ((b_i * t) - (l_i * b_i). 1199 * That is, f_i(t) is a line with slope b_i and y-intersect -(l_i * b_i). 1200 * Note that the y-intersect is always <= zero because latency can't be 1201 * negative. Note also that really the function is f_i(t) except when 1202 * f_i(t) is negative, in which case the function is zero. To take 1203 * care of this, let Q_i(t) = { if (f_i(t) > 0) return 1; else return 0; }. 1204 * So the actual number of bytes that will have arrived at the peer after 1205 * t miliseconds is f_i(t) * Q_i(t). 1206 * 1207 * At any given time, each link has some additional latency a_i >= 0 1208 * due to previously written fragment(s) which are still in the queue. 1209 * This value is easily computed from the time since last transmission, 1210 * the previous latency value, the number of bytes written, and the 1211 * link's bandwidth. 1212 * 1213 * Assume that l_i includes any a_i already, and that the links are 1214 * sorted by latency, so that l_i <= l_{i+1}. 1215 * 1216 * Let N be the total number of bytes in the current frame we are sending. 1217 * 1218 * Suppose we were to start writing bytes at time t = 0 on all links 1219 * simultaneously, which is the most we can possibly do. Then let 1220 * F(t) be equal to the total number of bytes received by the peer 1221 * after t miliseconds. Then F(t) = Sum_i (f_i(t) * Q_i(t)). 1222 * 1223 * Our goal is simply this: fragment the frame across the links such 1224 * that the peer is able to reconstruct the completed frame as soon as 1225 * possible, i.e., at the least possible value of t. Call this value t_0. 1226 * 1227 * Then it follows that F(t_0) = N. Our strategy is first to find the value 1228 * of t_0, and then deduce how many bytes to write to each link. 1229 * 1230 * Rewriting F(t_0): 1231 * 1232 * t_0 = ( N + Sum_i ( l_i * b_i * Q_i(t_0) ) ) / Sum_i ( b_i * Q_i(t_0) ) 1233 * 1234 * Now, we note that Q_i(t) is constant for l_i <= t <= l_{i+1}. t_0 will 1235 * lie in one of these ranges. To find it, we just need to find the i such 1236 * that F(l_i) <= N <= F(l_{i+1}). Then we compute all the constant values 1237 * for Q_i() in this range, plug in the remaining values, solving for t_0. 1238 * 1239 * Once t_0 is known, then the number of bytes to send on link i is 1240 * just f_i(t_0) * Q_i(t_0). 1241 * 1242 * In other words, we start allocating bytes to the links one at a time. 1243 * We keep adding links until the frame is completely sent. Some links 1244 * may not get any bytes because their latency is too high. 1245 * 1246 * Is all this work really worth the trouble? Depends on the situation. 1247 * The bigger the ratio of computer speed to link speed, and the more 1248 * important total bundle latency is (e.g., for interactive response time), 1249 * the more it's worth it. There is however the cost of calling this 1250 * function for every frame. The running time is O(n^2) where n is the 1251 * number of links that receive a non-zero number of bytes. 1252 * 1253 * Since latency is measured in miliseconds, the "resolution" of this 1254 * algorithm is one milisecond. 1255 * 1256 * To avoid this algorithm altogether, configure all links to have the 1257 * same latency and bandwidth. 1258 */ 1259 static void 1260 ng_ppp_mp_strategy(node_p node, int len, int *distrib) 1261 { 1262 const priv_p priv = node->private; 1263 int latency[NG_PPP_MAX_LINKS]; 1264 int sortByLatency[NG_PPP_MAX_LINKS]; 1265 int activeLinkNum, linkNum; 1266 int t0, total, topSum, botSum; 1267 struct timeval now; 1268 int i, numFragments; 1269 1270 /* If only one link, this gets real easy */ 1271 if (priv->numActiveLinks == 1) { 1272 distrib[0] = len; 1273 return; 1274 } 1275 1276 /* Get current time */ 1277 microtime(&now); 1278 1279 /* Compute latencies for each link at this point in time */ 1280 for (activeLinkNum = 0; 1281 activeLinkNum < priv->numActiveLinks; activeLinkNum++) { 1282 struct timeval diff; 1283 int xmitBytes; 1284 1285 /* Start with base latency value */ 1286 linkNum = priv->activeLinks[activeLinkNum]; 1287 latency[activeLinkNum] = priv->conf.links[linkNum].latency; 1288 sortByLatency[activeLinkNum] = activeLinkNum; /* see below */ 1289 1290 /* Any additional latency? */ 1291 if (priv->qstat[activeLinkNum].bytesInQueue == 0) 1292 continue; 1293 1294 /* Compute time delta since last write */ 1295 diff = now; 1296 timevalsub(&diff, &priv->qstat[activeLinkNum].lastWrite); 1297 if (now.tv_sec < 0 || diff.tv_sec >= 10) { /* sanity */ 1298 priv->qstat[activeLinkNum].bytesInQueue = 0; 1299 continue; 1300 } 1301 1302 /* How many bytes could have transmitted since last write? */ 1303 xmitBytes = priv->conf.links[linkNum].bandwidth * diff.tv_sec 1304 + (priv->conf.links[linkNum].bandwidth 1305 * (diff.tv_usec / 1000)) / 100; 1306 priv->qstat[activeLinkNum].bytesInQueue -= xmitBytes; 1307 if (priv->qstat[activeLinkNum].bytesInQueue < 0) 1308 priv->qstat[activeLinkNum].bytesInQueue = 0; 1309 else 1310 latency[activeLinkNum] += 1311 (100 * priv->qstat[activeLinkNum].bytesInQueue) 1312 / priv->conf.links[linkNum].bandwidth; 1313 } 1314 1315 /* Sort links by latency */ 1316 compareLatencies = latency; 1317 qsort(sortByLatency, 1318 priv->numActiveLinks, sizeof(*sortByLatency), ng_ppp_intcmp); 1319 compareLatencies = NULL; 1320 1321 /* Find the interval we need (add links in sortByLatency[] order) */ 1322 for (numFragments = 1; 1323 numFragments < priv->numActiveLinks; numFragments++) { 1324 for (total = i = 0; i < numFragments; i++) { 1325 int flowTime; 1326 1327 flowTime = latency[sortByLatency[numFragments]] 1328 - latency[sortByLatency[i]]; 1329 total += ((flowTime * priv->conf.links[ 1330 priv->activeLinks[sortByLatency[i]]].bandwidth) 1331 + 99) / 100; 1332 } 1333 if (total >= len) 1334 break; 1335 } 1336 1337 /* Solve for t_0 in that interval */ 1338 for (topSum = botSum = i = 0; i < numFragments; i++) { 1339 int bw = priv->conf.links[ 1340 priv->activeLinks[sortByLatency[i]]].bandwidth; 1341 1342 topSum += latency[sortByLatency[i]] * bw; /* / 100 */ 1343 botSum += bw; /* / 100 */ 1344 } 1345 t0 = ((len * 100) + topSum + botSum / 2) / botSum; 1346 1347 /* Compute f_i(t_0) all i */ 1348 bzero(distrib, priv->numActiveLinks * sizeof(*distrib)); 1349 for (total = i = 0; i < numFragments; i++) { 1350 int bw = priv->conf.links[ 1351 priv->activeLinks[sortByLatency[i]]].bandwidth; 1352 1353 distrib[sortByLatency[i]] = 1354 (bw * (t0 - latency[sortByLatency[i]]) + 50) / 100; 1355 total += distrib[sortByLatency[i]]; 1356 } 1357 1358 /* Deal with any rounding error */ 1359 if (total < len) { 1360 int fast = 0; 1361 1362 /* Find the fastest link */ 1363 for (i = 1; i < numFragments; i++) { 1364 if (priv->conf.links[ 1365 priv->activeLinks[sortByLatency[i]]].bandwidth > 1366 priv->conf.links[ 1367 priv->activeLinks[sortByLatency[fast]]].bandwidth) 1368 fast = i; 1369 } 1370 distrib[sortByLatency[fast]] += len - total; 1371 } else while (total > len) { 1372 int delta, slow = 0; 1373 1374 /* Find the slowest link that still has bytes to remove */ 1375 for (i = 1; i < numFragments; i++) { 1376 if (distrib[sortByLatency[slow]] == 0 1377 || (distrib[sortByLatency[i]] > 0 1378 && priv->conf.links[priv->activeLinks[ 1379 sortByLatency[i]]].bandwidth < 1380 priv->conf.links[priv->activeLinks[ 1381 sortByLatency[slow]]].bandwidth)) 1382 slow = i; 1383 } 1384 delta = total - len; 1385 if (delta > distrib[sortByLatency[slow]]) 1386 delta = distrib[sortByLatency[slow]]; 1387 distrib[sortByLatency[slow]] -= delta; 1388 total -= delta; 1389 } 1390 } 1391 1392 /* 1393 * Compare two integers 1394 */ 1395 static int 1396 ng_ppp_intcmp(const void *v1, const void *v2) 1397 { 1398 const int index1 = *((const int *) v1); 1399 const int index2 = *((const int *) v2); 1400 1401 return compareLatencies[index1] - compareLatencies[index2]; 1402 } 1403 1404 /* 1405 * Prepend a possibly compressed PPP protocol number in front of a frame 1406 */ 1407 static struct mbuf * 1408 ng_ppp_addproto(struct mbuf *m, int proto, int compOK) 1409 { 1410 if (compOK && PROT_COMPRESSABLE(proto)) { 1411 u_char pbyte = (u_char)proto; 1412 1413 return ng_ppp_prepend(m, &pbyte, 1); 1414 } else { 1415 u_int16_t pword = htons((u_int16_t)proto); 1416 1417 return ng_ppp_prepend(m, &pword, 2); 1418 } 1419 } 1420 1421 /* 1422 * Prepend some bytes to an mbuf 1423 */ 1424 static struct mbuf * 1425 ng_ppp_prepend(struct mbuf *m, const void *buf, int len) 1426 { 1427 M_PREPEND(m, len, M_NOWAIT); 1428 if (m == NULL || (m->m_len < len && (m = m_pullup(m, len)) == NULL)) 1429 return (NULL); 1430 bcopy(buf, mtod(m, u_char *), len); 1431 return (m); 1432 } 1433 1434 /* 1435 * Update private information that is derived from other private information 1436 */ 1437 static void 1438 ng_ppp_update(node_p node, int newConf) 1439 { 1440 const priv_p priv = node->private; 1441 int i; 1442 1443 /* Update active status for VJ Compression */ 1444 priv->vjCompHooked = priv->hooks[HOOK_INDEX_VJC_IP] != NULL 1445 && priv->hooks[HOOK_INDEX_VJC_COMP] != NULL 1446 && priv->hooks[HOOK_INDEX_VJC_UNCOMP] != NULL 1447 && priv->hooks[HOOK_INDEX_VJC_VJIP] != NULL; 1448 1449 /* Increase latency for each link an amount equal to one MP header */ 1450 if (newConf) { 1451 for (i = 0; i < NG_PPP_MAX_LINKS; i++) { 1452 int hdrBytes; 1453 1454 hdrBytes = (priv->conf.links[i].enableACFComp ? 0 : 2) 1455 + (priv->conf.links[i].enableProtoComp ? 1 : 2) 1456 + (priv->conf.xmitShortSeq ? 2 : 4); 1457 priv->conf.links[i].latency += 1458 ((hdrBytes * priv->conf.links[i].bandwidth) + 50) 1459 / 100; 1460 } 1461 } 1462 1463 /* Update list of active links */ 1464 bzero(&priv->activeLinks, sizeof(priv->activeLinks)); 1465 priv->numActiveLinks = 0; 1466 priv->allLinksEqual = 1; 1467 for (i = 0; i < NG_PPP_MAX_LINKS; i++) { 1468 struct ng_ppp_link_config *const lc = &priv->conf.links[i]; 1469 1470 if (lc->enableLink && priv->links[i] != NULL) { 1471 priv->activeLinks[priv->numActiveLinks++] = i; 1472 if (lc->latency != 1473 priv->conf.links[priv->activeLinks[0]].latency 1474 || lc->bandwidth != 1475 priv->conf.links[priv->activeLinks[0]].bandwidth) 1476 priv->allLinksEqual = 0; 1477 } 1478 } 1479 1480 /* Reset MP state if multi-link is no longer active */ 1481 if (!priv->conf.enableMultilink || priv->numActiveLinks == 0) { 1482 ng_ppp_free_frags(node); 1483 priv->mpSeqOut = MP_INITIAL_SEQ; 1484 bzero(&priv->qstat, sizeof(priv->qstat)); 1485 } 1486 } 1487 1488 /* 1489 * Determine if a new configuration would represent a valid change 1490 * from the current configuration and link activity status. 1491 */ 1492 static int 1493 ng_ppp_config_valid(node_p node, const struct ng_ppp_node_config *newConf) 1494 { 1495 const priv_p priv = node->private; 1496 int i, newNumLinksActive; 1497 1498 /* Check per-link config and count how many links would be active */ 1499 for (newNumLinksActive = i = 0; i < NG_PPP_MAX_LINKS; i++) { 1500 if (newConf->links[i].enableLink && priv->links[i] != NULL) 1501 newNumLinksActive++; 1502 if (!newConf->links[i].enableLink) 1503 continue; 1504 if (newConf->links[i].mru < MP_MIN_LINK_MRU) 1505 return (0); 1506 if (newConf->links[i].bandwidth == 0) 1507 return (0); 1508 if (newConf->links[i].bandwidth > NG_PPP_MAX_BANDWIDTH) 1509 return (0); 1510 if (newConf->links[i].latency > NG_PPP_MAX_LATENCY) 1511 return (0); 1512 } 1513 1514 /* Check bundle parameters */ 1515 if (newConf->enableMultilink && newConf->mrru < MP_MIN_MRRU) 1516 return (0); 1517 1518 /* Disallow changes to multi-link configuration while MP is active */ 1519 if (priv->numActiveLinks > 0 && newNumLinksActive > 0) { 1520 if (!priv->conf.enableMultilink != !newConf->enableMultilink 1521 || !priv->conf.xmitShortSeq != !newConf->xmitShortSeq 1522 || !priv->conf.recvShortSeq != !newConf->recvShortSeq) 1523 return (0); 1524 } 1525 1526 /* At most one link can be active unless multi-link is enabled */ 1527 if (!newConf->enableMultilink && newNumLinksActive > 1) 1528 return (0); 1529 1530 /* Configuration change would be valid */ 1531 return (1); 1532 } 1533 1534 /* 1535 * Free all entries in the fragment queue 1536 */ 1537 static void 1538 ng_ppp_free_frags(node_p node) 1539 { 1540 const priv_p priv = node->private; 1541 struct ng_ppp_frag *qent, *qnext; 1542 1543 for (qent = CIRCLEQ_FIRST(&priv->frags); 1544 qent != (void *)&priv->frags; qent = qnext) { 1545 qnext = CIRCLEQ_NEXT(qent, f_qent); 1546 NG_FREE_DATA(qent->data, qent->meta); 1547 FREE(qent, M_NETGRAPH); 1548 } 1549 CIRCLEQ_INIT(&priv->frags); 1550 } 1551 1552