1 /* 2 * CDDL HEADER START 3 * 4 * The contents of this file are subject to the terms of the 5 * Common Development and Distribution License (the "License"). 6 * You may not use this file except in compliance with the License. 7 * 8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 9 * or http://www.opensolaris.org/os/licensing. 10 * See the License for the specific language governing permissions 11 * and limitations under the License. 12 * 13 * When distributing Covered Code, include this CDDL HEADER in each 14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 15 * If applicable, add the following below this CDDL HEADER, with the 16 * fields enclosed by brackets "[]" replaced with your own identifying 17 * information: Portions Copyright [yyyy] [name of copyright owner] 18 * 19 * CDDL HEADER END 20 */ 21 /* 22 * Copyright 2007 Sun Microsystems, Inc. All rights reserved. 23 * Use is subject to license terms. 24 */ 25 26 #pragma ident "%Z%%M% %I% %E% SMI" 27 28 #include <sys/types.h> 29 #include <sys/socket.h> 30 #include <net/if.h> 31 #include <stdlib.h> 32 #include <sys/sockio.h> 33 #include <netinet/in.h> 34 #include <netinet/dhcp.h> 35 #include <string.h> 36 #include <unistd.h> 37 #include <search.h> 38 #include <libdevinfo.h> 39 #include <libdlpi.h> 40 #include <netinet/if_ether.h> 41 #include <arpa/inet.h> 42 #include <dhcpmsg.h> 43 #include <dhcp_inittab.h> 44 45 #include "agent.h" 46 #include "interface.h" 47 #include "util.h" 48 #include "packet.h" 49 #include "states.h" 50 51 dhcp_pif_t *v4root; 52 dhcp_pif_t *v6root; 53 54 static uint_t cached_v4_max_mtu, cached_v6_max_mtu; 55 56 /* 57 * Interface flags to watch: things that should be under our direct control. 58 */ 59 #define DHCP_IFF_WATCH (IFF_DHCPRUNNING | IFF_DEPRECATED | IFF_ADDRCONF | \ 60 IFF_TEMPORARY) 61 62 static void clear_lif_dhcp(dhcp_lif_t *); 63 64 /* 65 * insert_pif(): creates a new physical interface structure and chains it on 66 * the list. Initializes state that remains consistent across 67 * all use of the physical interface entry. 68 * 69 * input: const char *: the name of the physical interface 70 * boolean_t: if B_TRUE, this is DHCPv6 71 * int *: ignored on input; if insert_pif fails, set to a DHCP_IPC_E_* 72 * error code with the reason why 73 * output: dhcp_pif_t *: a pointer to the new entry, or NULL on failure 74 */ 75 76 dhcp_pif_t * 77 insert_pif(const char *pname, boolean_t isv6, int *error) 78 { 79 dhcp_pif_t *pif; 80 struct lifreq lifr; 81 dlpi_handle_t dh = NULL; 82 83 if ((pif = calloc(1, sizeof (*pif))) == NULL) { 84 dhcpmsg(MSG_ERR, "insert_pif: cannot allocate pif entry for " 85 "%s", pname); 86 *error = DHCP_IPC_E_MEMORY; 87 return (NULL); 88 } 89 90 pif->pif_isv6 = isv6; 91 pif->pif_hold_count = 1; 92 pif->pif_running = B_TRUE; 93 94 if (strlcpy(pif->pif_name, pname, LIFNAMSIZ) >= LIFNAMSIZ) { 95 dhcpmsg(MSG_ERROR, "insert_pif: interface name %s is too long", 96 pname); 97 *error = DHCP_IPC_E_INVIF; 98 goto failure; 99 } 100 101 /* We do not use DLPI with DHCPv6 */ 102 if (!isv6) { 103 int rc; 104 dlpi_info_t dlinfo; 105 106 /* 107 * Do the allocations necessary for IPv4 DHCP. 108 * 109 * 1. open the interface using DLPI 110 * 2. get the interface hardware type and hardware length 111 * 3. get the interface hardware address 112 */ 113 114 /* step 1 */ 115 if ((rc = dlpi_open(pname, &dh, 0)) != DLPI_SUCCESS) { 116 dhcpmsg(MSG_ERROR, "insert_pif: dlpi_open: %s", 117 dlpi_strerror(rc)); 118 *error = DHCP_IPC_E_INVIF; 119 goto failure; 120 } 121 122 if ((rc = dlpi_bind(dh, ETHERTYPE_IP, NULL)) != DLPI_SUCCESS) { 123 dhcpmsg(MSG_ERROR, "insert_pif: dlpi_bind: %s", 124 dlpi_strerror(rc)); 125 *error = DHCP_IPC_E_INVIF; 126 goto failure; 127 } 128 129 /* step 2 */ 130 rc = dlpi_info(dh, &dlinfo, 0); 131 if (rc != DLPI_SUCCESS) { 132 dhcpmsg(MSG_ERROR, "insert_pif: dlpi_info: %s", 133 dlpi_strerror(rc)); 134 *error = DHCP_IPC_E_INVIF; 135 goto failure; 136 } 137 138 pif->pif_hwtype = dlpi_arptype(dlinfo.di_mactype); 139 pif->pif_hwlen = dlinfo.di_physaddrlen; 140 141 dhcpmsg(MSG_DEBUG, "insert_pif: %s: hwtype %d, hwlen %d", 142 pname, pif->pif_hwtype, pif->pif_hwlen); 143 144 /* step 3 */ 145 if (pif->pif_hwlen > 0) { 146 pif->pif_hwaddr = malloc(pif->pif_hwlen); 147 if (pif->pif_hwaddr == NULL) { 148 dhcpmsg(MSG_ERR, "insert_pif: cannot allocate " 149 "pif_hwaddr for %s", pname); 150 *error = DHCP_IPC_E_MEMORY; 151 goto failure; 152 } 153 (void) memcpy(pif->pif_hwaddr, dlinfo.di_physaddr, 154 pif->pif_hwlen); 155 } 156 157 dlpi_close(dh); 158 dh = NULL; 159 } 160 161 /* 162 * This is a bit gross, but IP has a confused interface. We must 163 * assume that the zeroth LIF is plumbed, and must query there to get 164 * the interface index number. 165 */ 166 (void) strlcpy(lifr.lifr_name, pname, LIFNAMSIZ); 167 168 if (ioctl(isv6 ? v6_sock_fd : v4_sock_fd, SIOCGLIFINDEX, &lifr) == -1) { 169 *error = (errno == ENXIO) ? DHCP_IPC_E_INVIF : DHCP_IPC_E_INT; 170 dhcpmsg(MSG_ERR, "insert_pif: SIOCGLIFINDEX for %s", pname); 171 goto failure; 172 } 173 pif->pif_index = lifr.lifr_index; 174 175 if (ioctl(isv6 ? v6_sock_fd : v4_sock_fd, SIOCGLIFMTU, &lifr) == -1) { 176 *error = (errno == ENXIO) ? DHCP_IPC_E_INVIF : DHCP_IPC_E_INT; 177 dhcpmsg(MSG_ERR, "insert_pif: SIOCGLIFMTU for %s", pname); 178 goto failure; 179 } 180 pif->pif_max = lifr.lifr_mtu; 181 182 if (pif->pif_max < DHCP_DEF_MAX_SIZE) { 183 dhcpmsg(MSG_ERROR, "insert_pif: MTU of %s is too small to " 184 "support DHCP (%u < %u)", pname, pif->pif_max, 185 DHCP_DEF_MAX_SIZE); 186 *error = DHCP_IPC_E_INVIF; 187 goto failure; 188 } 189 190 insque(pif, isv6 ? &v6root : &v4root); 191 192 return (pif); 193 194 failure: 195 if (dh != NULL) 196 dlpi_close(dh); 197 release_pif(pif); 198 return (NULL); 199 } 200 201 /* 202 * hold_pif(): acquire a hold on a physical interface structure. 203 * 204 * input: dhcp_pif_t *: a pointer to the PIF structure 205 * output: none 206 */ 207 208 void 209 hold_pif(dhcp_pif_t *pif) 210 { 211 pif->pif_hold_count++; 212 dhcpmsg(MSG_DEBUG2, "hold_pif: hold count on %s: %u", pif->pif_name, 213 pif->pif_hold_count); 214 } 215 216 /* 217 * release_pif(): release a hold on a physical interface structure; will 218 * destroy the structure on the last hold removed. 219 * 220 * input: dhcp_pif_t *: a pointer to the PIF structure 221 * output: none 222 */ 223 224 void 225 release_pif(dhcp_pif_t *pif) 226 { 227 if (pif->pif_hold_count == 0) { 228 dhcpmsg(MSG_CRIT, "release_pif: extraneous release"); 229 return; 230 } 231 232 if (--pif->pif_hold_count == 0) { 233 dhcpmsg(MSG_DEBUG, "release_pif: freeing PIF %s", 234 pif->pif_name); 235 236 remque(pif); 237 free(pif->pif_hwaddr); 238 free(pif); 239 } else { 240 dhcpmsg(MSG_DEBUG2, "release_pif: hold count on %s: %u", 241 pif->pif_name, pif->pif_hold_count); 242 } 243 } 244 245 /* 246 * lookup_pif_by_index(): Looks up PIF entries given regular ifIndex. 247 * 248 * input: uint_t: the interface index 249 * boolean_t: B_TRUE if using DHCPv6, B_FALSE otherwise 250 * output: dhcp_pif_t *: the matching PIF, or NULL if not found 251 */ 252 253 dhcp_pif_t * 254 lookup_pif_by_index(uint_t ifindex, boolean_t isv6) 255 { 256 dhcp_pif_t *pif; 257 258 for (pif = isv6 ? v6root : v4root; pif != NULL; pif = pif->pif_next) { 259 if (pif->pif_index == ifindex) 260 break; 261 } 262 263 return (pif); 264 } 265 266 /* 267 * lookup_pif_by_uindex(): Looks up PIF entries given truncated index and 268 * previous PIF pointer (or NULL for list start). 269 * Caller is expected to iterate through all 270 * potential matches to find interface of interest. 271 * 272 * input: uint16_t: the interface index (truncated) 273 * dhcp_pif_t *: the previous PIF, or NULL for list start 274 * boolean_t: B_TRUE if using DHCPv6, B_FALSE otherwise 275 * output: dhcp_pif_t *: the next matching PIF, or NULL if not found 276 * note: This operates using the 'truncated' (16-bit) ifindex as seen by 277 * routing socket clients. The value stored in pif_index is the 278 * 32-bit ifindex from the ioctl interface. 279 */ 280 281 dhcp_pif_t * 282 lookup_pif_by_uindex(uint16_t ifindex, dhcp_pif_t *pif, boolean_t isv6) 283 { 284 if (pif == NULL) 285 pif = isv6 ? v6root : v4root; 286 else 287 pif = pif->pif_next; 288 289 for (; pif != NULL; pif = pif->pif_next) { 290 if ((pif->pif_index & 0xffff) == ifindex) 291 break; 292 } 293 294 return (pif); 295 } 296 297 /* 298 * lookup_pif_by_name(): Looks up a physical interface entry given a name. 299 * 300 * input: const char *: the physical interface name 301 * boolean_t: B_TRUE if using DHCPv6, B_FALSE otherwise 302 * output: dhcp_pif_t *: the matching PIF, or NULL if not found 303 */ 304 305 dhcp_pif_t * 306 lookup_pif_by_name(const char *pname, boolean_t isv6) 307 { 308 dhcp_pif_t *pif; 309 310 pif = isv6 ? v6root : v4root; 311 312 for (; pif != NULL; pif = pif->pif_next) { 313 if (strcmp(pif->pif_name, pname) == 0) 314 break; 315 } 316 317 return (pif); 318 } 319 320 /* 321 * pif_status(): update the physical interface up/down status. 322 * 323 * input: dhcp_pif_t *: the physical interface to be updated 324 * boolean_t: B_TRUE if the interface is going up 325 * output: none 326 */ 327 328 void 329 pif_status(dhcp_pif_t *pif, boolean_t isup) 330 { 331 dhcp_lif_t *lif; 332 dhcp_smach_t *dsmp; 333 334 pif->pif_running = isup; 335 dhcpmsg(MSG_DEBUG, "interface %s has %s", pif->pif_name, 336 isup ? "come back up" : "gone down"); 337 for (lif = pif->pif_lifs; lif != NULL; lif = lif->lif_next) { 338 for (dsmp = lif->lif_smachs; dsmp != NULL; 339 dsmp = dsmp->dsm_next) { 340 if (isup) 341 refresh_smach(dsmp); 342 else 343 remove_default_routes(dsmp); 344 } 345 } 346 } 347 348 /* Helper for insert_lif: extract addresses as defined */ 349 #define ASSIGN_ADDR(v4, v6, lf) \ 350 if (pif->pif_isv6) { \ 351 lif->v6 = ((struct sockaddr_in6 *)&lifr.lf)->sin6_addr; \ 352 } else { \ 353 lif->v4 = ((struct sockaddr_in *)&lifr.lf)->sin_addr.s_addr; \ 354 } 355 356 /* 357 * insert_lif(): Creates a new logical interface structure and chains it on 358 * the list for a given physical interface. Initializes state 359 * that remains consistent across all use of the logical 360 * interface entry. Caller's PIF hold is transferred to the 361 * LIF on success, and is dropped on failure. 362 * 363 * input: dhcp_pif_t *: pointer to the physical interface for this LIF 364 * const char *: the name of the logical interface 365 * int *: ignored on input; if insert_pif fails, set to a DHCP_IPC_E_* 366 * error code with the reason why 367 * output: dhcp_lif_t *: a pointer to the new entry, or NULL on failure 368 */ 369 370 dhcp_lif_t * 371 insert_lif(dhcp_pif_t *pif, const char *lname, int *error) 372 { 373 dhcp_lif_t *lif; 374 int fd; 375 struct lifreq lifr; 376 377 if ((lif = calloc(1, sizeof (*lif))) == NULL) { 378 dhcpmsg(MSG_ERR, "insert_lif: cannot allocate lif entry for " 379 "%s", lname); 380 *error = DHCP_IPC_E_MEMORY; 381 return (NULL); 382 } 383 384 lif->lif_sock_ip_fd = -1; 385 lif->lif_packet_id = -1; 386 lif->lif_iaid_id = -1; 387 lif->lif_hold_count = 1; 388 lif->lif_pif = pif; 389 lif->lif_removed = B_TRUE; 390 init_timer(&lif->lif_preferred, 0); 391 init_timer(&lif->lif_expire, 0); 392 393 if (strlcpy(lif->lif_name, lname, LIFNAMSIZ) >= LIFNAMSIZ) { 394 dhcpmsg(MSG_ERROR, "insert_lif: interface name %s is too long", 395 lname); 396 *error = DHCP_IPC_E_INVIF; 397 goto failure; 398 } 399 400 (void) strlcpy(lifr.lifr_name, lname, LIFNAMSIZ); 401 402 fd = pif->pif_isv6 ? v6_sock_fd : v4_sock_fd; 403 404 if (ioctl(fd, SIOCGLIFMTU, &lifr) == -1) 405 lif->lif_max = 1024; 406 else 407 lif->lif_max = lifr.lifr_mtu; 408 409 if (ioctl(fd, SIOCGLIFADDR, &lifr) == -1) { 410 if (errno == ENXIO) 411 *error = DHCP_IPC_E_INVIF; 412 else 413 *error = DHCP_IPC_E_INT; 414 dhcpmsg(MSG_ERR, "insert_lif: SIOCGLIFADDR for %s", lname); 415 goto failure; 416 } 417 ASSIGN_ADDR(lif_addr, lif_v6addr, lifr_addr); 418 419 if (ioctl(fd, SIOCGLIFNETMASK, &lifr) == -1) { 420 if (errno == ENXIO) 421 *error = DHCP_IPC_E_INVIF; 422 else 423 *error = DHCP_IPC_E_INT; 424 dhcpmsg(MSG_ERR, "insert_lif: SIOCGLIFNETMASK for %s", lname); 425 goto failure; 426 } 427 ASSIGN_ADDR(lif_netmask, lif_v6mask, lifr_addr); 428 429 if (ioctl(fd, SIOCGLIFFLAGS, &lifr) == -1) { 430 *error = DHCP_IPC_E_INT; 431 dhcpmsg(MSG_ERR, "insert_lif: SIOCGLIFFLAGS for %s", lname); 432 goto failure; 433 } 434 lif->lif_flags = lifr.lifr_flags; 435 436 /* 437 * If we've just detected the interface going up or down, then signal 438 * an appropriate action. There may be other state machines here. 439 */ 440 if ((lifr.lifr_flags & IFF_RUNNING) && !pif->pif_running) { 441 pif_status(pif, B_TRUE); 442 } else if (!(lifr.lifr_flags & IFF_RUNNING) && pif->pif_running) { 443 pif_status(pif, B_FALSE); 444 } 445 446 if (lifr.lifr_flags & IFF_POINTOPOINT) { 447 if (ioctl(fd, SIOCGLIFDSTADDR, &lifr) == -1) { 448 *error = DHCP_IPC_E_INT; 449 dhcpmsg(MSG_ERR, "insert_lif: SIOCGLIFDSTADDR for %s", 450 lname); 451 goto failure; 452 } 453 ASSIGN_ADDR(lif_peer, lif_v6peer, lifr_dstaddr); 454 } else if (!pif->pif_isv6 && (lifr.lifr_flags & IFF_BROADCAST)) { 455 if (ioctl(fd, SIOCGLIFBRDADDR, &lifr) == -1) { 456 *error = DHCP_IPC_E_INT; 457 dhcpmsg(MSG_ERR, "insert_lif: SIOCGLIFBRDADDR for %s", 458 lname); 459 goto failure; 460 } 461 lif->lif_broadcast = 462 ((struct sockaddr_in *)&lifr.lifr_broadaddr)->sin_addr. 463 s_addr; 464 } 465 466 if (pif->pif_isv6) 467 cached_v6_max_mtu = 0; 468 else 469 cached_v4_max_mtu = 0; 470 471 lif->lif_removed = B_FALSE; 472 insque(lif, &pif->pif_lifs); 473 474 return (lif); 475 476 failure: 477 release_lif(lif); 478 return (NULL); 479 } 480 481 /* 482 * hold_lif(): acquire a hold on a logical interface structure. 483 * 484 * input: dhcp_lif_t *: a pointer to the LIF structure 485 * output: none 486 */ 487 488 void 489 hold_lif(dhcp_lif_t *lif) 490 { 491 lif->lif_hold_count++; 492 dhcpmsg(MSG_DEBUG2, "hold_lif: hold count on %s: %u", lif->lif_name, 493 lif->lif_hold_count); 494 } 495 496 /* 497 * release_lif(): release a hold on a logical interface structure; will 498 * destroy the structure on the last hold removed. 499 * 500 * input: dhcp_lif_t *: a pointer to the LIF structure 501 * output: none 502 */ 503 504 void 505 release_lif(dhcp_lif_t *lif) 506 { 507 if (lif->lif_hold_count == 0) { 508 dhcpmsg(MSG_CRIT, "release_lif: extraneous release on %s", 509 lif->lif_name); 510 return; 511 } 512 513 if (lif->lif_hold_count == 1 && !lif->lif_removed) { 514 unplumb_lif(lif); 515 return; 516 } 517 518 if (--lif->lif_hold_count == 0) { 519 dhcp_pif_t *pif; 520 521 dhcpmsg(MSG_DEBUG, "release_lif: freeing LIF %s", 522 lif->lif_name); 523 524 if (lif->lif_lease != NULL) 525 dhcpmsg(MSG_CRIT, 526 "release_lif: still holding lease at last hold!"); 527 close_ip_lif(lif); 528 pif = lif->lif_pif; 529 if (pif->pif_isv6) 530 cached_v6_max_mtu = 0; 531 else 532 cached_v4_max_mtu = 0; 533 release_pif(pif); 534 free(lif); 535 } else { 536 dhcpmsg(MSG_DEBUG2, "release_lif: hold count on %s: %u", 537 lif->lif_name, lif->lif_hold_count); 538 } 539 } 540 541 /* 542 * remove_lif(): remove a logical interface from its PIF and lease (if any) and 543 * the lease's hold on the LIF. Assumes that we did not plumb 544 * the interface. 545 * 546 * input: dhcp_lif_t *: a pointer to the LIF structure 547 * output: none 548 */ 549 550 void 551 remove_lif(dhcp_lif_t *lif) 552 { 553 if (lif->lif_plumbed) { 554 dhcpmsg(MSG_CRIT, "remove_lif: attempted invalid removal of %s", 555 lif->lif_name); 556 return; 557 } 558 if (lif->lif_removed) { 559 dhcpmsg(MSG_CRIT, "remove_lif: extraneous removal of %s", 560 lif->lif_name); 561 } else { 562 dhcp_lif_t *lifnext; 563 dhcp_lease_t *dlp; 564 565 dhcpmsg(MSG_DEBUG2, "remove_lif: removing %s", lif->lif_name); 566 lif->lif_removed = B_TRUE; 567 lifnext = lif->lif_next; 568 clear_lif_dhcp(lif); 569 cancel_lif_timers(lif); 570 if (lif->lif_iaid_id != -1 && 571 iu_cancel_timer(tq, lif->lif_iaid_id, NULL) == 1) { 572 lif->lif_iaid_id = -1; 573 release_lif(lif); 574 } 575 576 /* Remove from PIF list */ 577 remque(lif); 578 579 /* If we were part of a lease, then remove ourselves */ 580 if ((dlp = lif->lif_lease) != NULL) { 581 if (--dlp->dl_nlifs == 0) 582 dlp->dl_lifs = NULL; 583 else if (dlp->dl_lifs == lif) 584 dlp->dl_lifs = lifnext; 585 if (lif->lif_flags & IFF_DHCPRUNNING) 586 clear_lif_dhcp(lif); 587 if (lif->lif_declined != NULL) { 588 dlp->dl_smach->dsm_lif_down--; 589 lif->lif_declined = NULL; 590 } 591 lif->lif_lease = NULL; 592 release_lif(lif); 593 } 594 } 595 } 596 597 /* 598 * lookup_lif_by_name(): Looks up a logical interface entry given a name and 599 * a physical interface. 600 * 601 * input: const char *: the logical interface name 602 * const dhcp_pif_t *: the physical interface 603 * output: dhcp_lif_t *: the matching LIF, or NULL if not found 604 */ 605 606 dhcp_lif_t * 607 lookup_lif_by_name(const char *lname, const dhcp_pif_t *pif) 608 { 609 dhcp_lif_t *lif; 610 611 for (lif = pif->pif_lifs; lif != NULL; lif = lif->lif_next) { 612 if (strcmp(lif->lif_name, lname) == 0) 613 break; 614 } 615 616 return (lif); 617 } 618 619 /* 620 * checkaddr(): checks if the given address is still set on the given LIF 621 * 622 * input: const dhcp_lif_t *: the LIF to check 623 * int: the address to look up on the interface (ioctl) 624 * const in6_addr_t *: the address to compare to 625 * const char *: name of the address for logging purposes 626 * output: boolean_t: B_TRUE if the address is still set; B_FALSE if not 627 */ 628 629 static boolean_t 630 checkaddr(const dhcp_lif_t *lif, int ioccmd, const in6_addr_t *addr, 631 const char *aname) 632 { 633 boolean_t isv6; 634 int fd; 635 struct lifreq lifr; 636 char abuf1[INET6_ADDRSTRLEN]; 637 char abuf2[INET6_ADDRSTRLEN]; 638 639 (void) memset(&lifr, 0, sizeof (struct lifreq)); 640 (void) strlcpy(lifr.lifr_name, lif->lif_name, LIFNAMSIZ); 641 642 isv6 = lif->lif_pif->pif_isv6; 643 fd = isv6 ? v6_sock_fd : v4_sock_fd; 644 645 if (ioctl(fd, ioccmd, &lifr) == -1) { 646 if (errno == ENXIO) { 647 dhcpmsg(MSG_WARNING, "checkaddr: interface %s is gone", 648 lif->lif_name); 649 return (B_FALSE); 650 } 651 dhcpmsg(MSG_DEBUG, 652 "checkaddr: ignoring ioctl error on %s %x: %s", 653 lif->lif_name, ioccmd, strerror(errno)); 654 } else if (isv6) { 655 struct sockaddr_in6 *sin6 = 656 (struct sockaddr_in6 *)&lifr.lifr_addr; 657 658 if (!IN6_ARE_ADDR_EQUAL(&sin6->sin6_addr, addr)) { 659 dhcpmsg(MSG_WARNING, 660 "checkaddr: expected %s %s on %s, have %s", aname, 661 inet_ntop(AF_INET6, addr, abuf1, sizeof (abuf1)), 662 lif->lif_name, inet_ntop(AF_INET6, &sin6->sin6_addr, 663 abuf2, sizeof (abuf2))); 664 return (B_FALSE); 665 } 666 } else { 667 struct sockaddr_in *sinp = 668 (struct sockaddr_in *)&lifr.lifr_addr; 669 ipaddr_t v4addr; 670 671 IN6_V4MAPPED_TO_IPADDR(addr, v4addr); 672 if (sinp->sin_addr.s_addr != v4addr) { 673 dhcpmsg(MSG_WARNING, 674 "checkaddr: expected %s %s on %s, have %s", aname, 675 inet_ntop(AF_INET, &v4addr, abuf1, sizeof (abuf1)), 676 lif->lif_name, inet_ntop(AF_INET, &sinp->sin_addr, 677 abuf2, sizeof (abuf2))); 678 return (B_FALSE); 679 } 680 } 681 return (B_TRUE); 682 } 683 684 /* 685 * verify_lif(): verifies than a LIF is still valid (i.e., has not been 686 * explicitly or implicitly dropped or released) 687 * 688 * input: const dhcp_lif_t *: the LIF to verify 689 * output: boolean_t: B_TRUE if the LIF is still valid, B_FALSE otherwise 690 */ 691 692 boolean_t 693 verify_lif(const dhcp_lif_t *lif) 694 { 695 boolean_t isv6; 696 int fd; 697 struct lifreq lifr; 698 699 (void) memset(&lifr, 0, sizeof (struct lifreq)); 700 (void) strlcpy(lifr.lifr_name, lif->lif_name, LIFNAMSIZ); 701 702 isv6 = lif->lif_pif->pif_isv6; 703 fd = isv6 ? v6_sock_fd : v4_sock_fd; 704 705 if (ioctl(fd, SIOCGLIFFLAGS, &lifr) == -1) { 706 if (errno != ENXIO) { 707 dhcpmsg(MSG_ERR, 708 "verify_lif: SIOCGLIFFLAGS failed on %s", 709 lif->lif_name); 710 } 711 return (B_FALSE); 712 } 713 714 /* 715 * If important flags have changed, then abandon the interface. 716 */ 717 if ((lif->lif_flags ^ lifr.lifr_flags) & DHCP_IFF_WATCH) { 718 dhcpmsg(MSG_DEBUG, "verify_lif: unexpected flag change on %s: " 719 "%llx to %llx (%llx)", lif->lif_name, lif->lif_flags, 720 lifr.lifr_flags, (lif->lif_flags ^ lifr.lifr_flags) & 721 DHCP_IFF_WATCH); 722 return (B_FALSE); 723 } 724 725 /* 726 * Special case: if the interface has gone down as a duplicate, then 727 * this alone does _not_ mean that we're abandoning it just yet. Allow 728 * the state machine to handle this normally by trying to get a new 729 * lease. 730 */ 731 if ((lifr.lifr_flags & (IFF_UP|IFF_DUPLICATE)) == IFF_DUPLICATE) { 732 dhcpmsg(MSG_DEBUG, "verify_lif: duplicate address on %s", 733 lif->lif_name); 734 return (B_TRUE); 735 } 736 737 /* 738 * If the user has torn down or started up the interface manually, then 739 * abandon the lease. 740 */ 741 if ((lif->lif_flags ^ lifr.lifr_flags) & IFF_UP) { 742 dhcpmsg(MSG_DEBUG, "verify_lif: user has %s %s", 743 lifr.lifr_flags & IFF_UP ? "started up" : "shut down", 744 lif->lif_name); 745 return (B_FALSE); 746 } 747 748 /* 749 * Check for delete and recreate. 750 */ 751 if (ioctl(fd, SIOCGLIFINDEX, &lifr) == -1) { 752 dhcpmsg(MSG_ERR, "verify_lif: SIOCGLIFINDEX failed on %s", 753 lif->lif_name); 754 return (B_FALSE); 755 } 756 if (lifr.lifr_index != lif->lif_pif->pif_index) { 757 dhcpmsg(MSG_DEBUG, 758 "verify_lif: ifindex on %s changed: %u to %u", 759 lif->lif_name, lif->lif_pif->pif_index, lifr.lifr_index); 760 return (B_FALSE); 761 } 762 763 /* 764 * If the IP address, netmask, or broadcast address have changed, or 765 * the interface has been unplumbed, then we act like there has been an 766 * implicit drop. (Note that the netmask is under DHCP control for 767 * IPv4, but not for DHCPv6, and that IPv6 doesn't have broadcast 768 * addresses.) 769 */ 770 771 if (!checkaddr(lif, SIOCGLIFADDR, &lif->lif_v6addr, "local address")) 772 return (B_FALSE); 773 774 if (isv6) { 775 /* 776 * If it's not point-to-point, we're done. If it is, then 777 * check the peer's address as well. 778 */ 779 return (!(lif->lif_flags & IFF_POINTOPOINT) || 780 checkaddr(lif, SIOCGLIFDSTADDR, &lif->lif_v6peer, 781 "peer address")); 782 } else { 783 if (!checkaddr(lif, SIOCGLIFNETMASK, &lif->lif_v6mask, 784 "netmask")) 785 return (B_FALSE); 786 787 return (checkaddr(lif, 788 (lif->lif_flags & IFF_POINTOPOINT) ? SIOCGLIFDSTADDR : 789 SIOCGLIFBRDADDR, &lif->lif_v6peer, "peer address")); 790 } 791 } 792 793 /* 794 * canonize_lif(): puts the interface in a canonical (zeroed) form. This is 795 * used only on the "main" LIF for IPv4. All other interfaces 796 * are under dhcpagent control and are removed using 797 * unplumb_lif(). 798 * 799 * input: dhcp_lif_t *: the interface to canonize 800 * boolean_t: only canonize lif if it's under DHCP control 801 * output: none 802 */ 803 804 static void 805 canonize_lif(dhcp_lif_t *lif, boolean_t dhcponly) 806 { 807 boolean_t isv6; 808 int fd; 809 struct lifreq lifr; 810 811 /* 812 * If there's nothing here, then don't touch the interface. This can 813 * happen when an already-canonized LIF is recanonized. 814 */ 815 if (IN6_IS_ADDR_UNSPECIFIED(&lif->lif_v6addr)) 816 return; 817 818 isv6 = lif->lif_pif->pif_isv6; 819 dhcpmsg(MSG_VERBOSE, "canonizing IPv%d interface %s", 820 isv6 ? 6 : 4, lif->lif_name); 821 822 lif->lif_v6addr = my_in6addr_any; 823 lif->lif_v6mask = my_in6addr_any; 824 lif->lif_v6peer = my_in6addr_any; 825 826 (void) memset(&lifr, 0, sizeof (struct lifreq)); 827 (void) strlcpy(lifr.lifr_name, lif->lif_name, LIFNAMSIZ); 828 829 fd = isv6 ? v6_sock_fd : v4_sock_fd; 830 831 if (ioctl(fd, SIOCGLIFFLAGS, &lifr) == -1) { 832 if (errno != ENXIO) { 833 dhcpmsg(MSG_ERR, "canonize_lif: can't get flags for %s", 834 lif->lif_name); 835 } 836 return; 837 } 838 839 if (dhcponly && !(lifr.lifr_flags & IFF_DHCPRUNNING)) { 840 dhcpmsg(MSG_INFO, 841 "canonize_lif: cannot clear %s; flags are %llx", 842 lif->lif_name, lifr.lifr_flags); 843 return; 844 } 845 846 /* 847 * clear the UP flag, but don't clear DHCPRUNNING since 848 * that should only be done when the interface is removed 849 * (see clear_lif_dhcp() and remove_lif()) 850 */ 851 852 lif->lif_flags = lifr.lifr_flags &= ~IFF_UP; 853 854 if (ioctl(fd, SIOCSLIFFLAGS, &lifr) == -1) { 855 dhcpmsg(MSG_ERR, "canonize_lif: can't set flags for %s", 856 lif->lif_name); 857 return; 858 } 859 860 (void) memset(&lifr.lifr_addr, 0, sizeof (lifr.lifr_addr)); 861 if (isv6) { 862 struct sockaddr_in6 *sin6 = 863 (struct sockaddr_in6 *)&lifr.lifr_addr; 864 865 sin6->sin6_family = AF_INET6; 866 sin6->sin6_addr = my_in6addr_any; 867 } else { 868 struct sockaddr_in *sinv = 869 (struct sockaddr_in *)&lifr.lifr_addr; 870 871 sinv->sin_family = AF_INET; 872 sinv->sin_addr.s_addr = htonl(INADDR_ANY); 873 } 874 875 if (ioctl(fd, SIOCSLIFADDR, &lifr) == -1) { 876 dhcpmsg(MSG_ERR, 877 "canonize_lif: can't clear local address on %s", 878 lif->lif_name); 879 } 880 881 /* Netmask is under in.ndpd control with IPv6 */ 882 if (!isv6 && ioctl(fd, SIOCSLIFNETMASK, &lifr) == -1) { 883 dhcpmsg(MSG_ERR, "canonize_lif: can't clear netmask on %s", 884 lif->lif_name); 885 } 886 887 if (lif->lif_flags & IFF_POINTOPOINT) { 888 if (ioctl(fd, SIOCSLIFDSTADDR, &lifr) == -1) { 889 dhcpmsg(MSG_ERR, 890 "canonize_lif: can't clear remote address on %s", 891 lif->lif_name); 892 } 893 } else if (!isv6) { 894 if (ioctl(fd, SIOCSLIFBRDADDR, &lifr) == -1) { 895 dhcpmsg(MSG_ERR, 896 "canonize_lif: can't clear broadcast address on %s", 897 lif->lif_name); 898 } 899 } 900 } 901 902 /* 903 * plumb_lif(): Adds the LIF to the system. This is used for all 904 * DHCPv6-derived interfaces. The returned LIF has a hold 905 * on it. 906 * 907 * input: dhcp_lif_t *: the interface to unplumb 908 * output: none 909 */ 910 911 dhcp_lif_t * 912 plumb_lif(dhcp_pif_t *pif, const in6_addr_t *addr) 913 { 914 dhcp_lif_t *lif; 915 char abuf[INET6_ADDRSTRLEN]; 916 struct lifreq lifr; 917 struct sockaddr_in6 *sin6; 918 int error; 919 920 (void) inet_ntop(AF_INET6, addr, abuf, sizeof (abuf)); 921 922 for (lif = pif->pif_lifs; lif != NULL; lif = lif->lif_next) { 923 if (IN6_ARE_ADDR_EQUAL(&lif->lif_v6addr, addr)) { 924 dhcpmsg(MSG_ERR, 925 "plumb_lif: entry for %s already exists!", abuf); 926 return (NULL); 927 } 928 } 929 930 /* First, create a new zero-address logical interface */ 931 (void) memset(&lifr, 0, sizeof (lifr)); 932 (void) strlcpy(lifr.lifr_name, pif->pif_name, sizeof (lifr.lifr_name)); 933 if (ioctl(v6_sock_fd, SIOCLIFADDIF, &lifr) == -1) { 934 dhcpmsg(MSG_ERR, "plumb_lif: SIOCLIFADDIF %s", pif->pif_name); 935 return (NULL); 936 } 937 938 /* Next, set the netmask to all ones */ 939 sin6 = (struct sockaddr_in6 *)&lifr.lifr_addr; 940 sin6->sin6_family = AF_INET6; 941 (void) memset(&sin6->sin6_addr, 0xff, sizeof (sin6->sin6_addr)); 942 if (ioctl(v6_sock_fd, SIOCSLIFNETMASK, &lifr) == -1) { 943 dhcpmsg(MSG_ERR, "plumb_lif: SIOCSLIFNETMASK %s", 944 lifr.lifr_name); 945 goto failure; 946 } 947 948 /* Now set the interface address */ 949 sin6->sin6_addr = *addr; 950 if (ioctl(v6_sock_fd, SIOCSLIFADDR, &lifr) == -1) { 951 dhcpmsg(MSG_ERR, "plumb_lif: SIOCSLIFADDR %s %s", 952 lifr.lifr_name, abuf); 953 goto failure; 954 } 955 956 /* Mark the interface up */ 957 if (ioctl(v6_sock_fd, SIOCGLIFFLAGS, &lifr) == -1) { 958 dhcpmsg(MSG_ERR, "plumb_lif: SIOCGLIFFLAGS %s", 959 lifr.lifr_name); 960 goto failure; 961 } 962 lifr.lifr_flags |= IFF_UP | IFF_DHCPRUNNING; 963 if (ioctl(v6_sock_fd, SIOCSLIFFLAGS, &lifr) == -1) { 964 dhcpmsg(MSG_ERR, "plumb_lif: SIOCSLIFFLAGS %s", 965 lifr.lifr_name); 966 goto failure; 967 } 968 969 /* Now we can create the internal LIF structure */ 970 hold_pif(pif); 971 if ((lif = insert_lif(pif, lifr.lifr_name, &error)) == NULL) 972 goto failure; 973 974 dhcpmsg(MSG_DEBUG, "plumb_lif: plumbed up %s on %s", abuf, 975 lif->lif_name); 976 lif->lif_plumbed = B_TRUE; 977 978 return (lif); 979 980 failure: 981 if (ioctl(v6_sock_fd, SIOCLIFREMOVEIF, &lifr) == -1 && 982 errno != ENXIO) { 983 dhcpmsg(MSG_ERR, "plumb_lif: SIOCLIFREMOVEIF %s", 984 lifr.lifr_name); 985 } 986 return (NULL); 987 } 988 989 /* 990 * unplumb_lif(): Removes the LIF from dhcpagent and the system. This is used 991 * for all interfaces configured by DHCP (those in leases). 992 * 993 * input: dhcp_lif_t *: the interface to unplumb 994 * output: none 995 */ 996 997 void 998 unplumb_lif(dhcp_lif_t *lif) 999 { 1000 dhcp_lease_t *dlp; 1001 1002 if (lif->lif_plumbed) { 1003 struct lifreq lifr; 1004 1005 (void) memset(&lifr, 0, sizeof (lifr)); 1006 (void) strlcpy(lifr.lifr_name, lif->lif_name, 1007 sizeof (lifr.lifr_name)); 1008 if (ioctl(v6_sock_fd, SIOCLIFREMOVEIF, &lifr) == -1 && 1009 errno != ENXIO) { 1010 dhcpmsg(MSG_ERR, "unplumb_lif: SIOCLIFREMOVEIF %s", 1011 lif->lif_name); 1012 } 1013 lif->lif_plumbed = B_FALSE; 1014 } 1015 lif->lif_flags = 0; 1016 /* 1017 * Special case: if we're "unplumbing" the main LIF for DHCPv4, then 1018 * just canonize it and remove it from the lease. 1019 */ 1020 if ((dlp = lif->lif_lease) != NULL && dlp->dl_smach->dsm_lif == lif) { 1021 canonize_lif(lif, B_TRUE); 1022 cancel_lif_timers(lif); 1023 if (lif->lif_declined != NULL) { 1024 dlp->dl_smach->dsm_lif_down--; 1025 lif->lif_declined = NULL; 1026 } 1027 dlp->dl_nlifs = 0; 1028 dlp->dl_lifs = NULL; 1029 lif->lif_lease = NULL; 1030 release_lif(lif); 1031 } else { 1032 remove_lif(lif); 1033 } 1034 } 1035 1036 /* 1037 * attach_lif(): create a new logical interface, creating the physical 1038 * interface as necessary. 1039 * 1040 * input: const char *: the logical interface name 1041 * boolean_t: B_TRUE for IPv6 1042 * int *: set to DHCP_IPC_E_* if creation fails 1043 * output: dhcp_lif_t *: pointer to new entry, or NULL on failure 1044 */ 1045 1046 dhcp_lif_t * 1047 attach_lif(const char *lname, boolean_t isv6, int *error) 1048 { 1049 dhcp_pif_t *pif; 1050 char pname[LIFNAMSIZ], *cp; 1051 1052 (void) strlcpy(pname, lname, sizeof (pname)); 1053 if ((cp = strchr(pname, ':')) != NULL) 1054 *cp = '\0'; 1055 1056 if ((pif = lookup_pif_by_name(pname, isv6)) != NULL) 1057 hold_pif(pif); 1058 else if ((pif = insert_pif(pname, isv6, error)) == NULL) 1059 return (NULL); 1060 1061 if (lookup_lif_by_name(lname, pif) != NULL) { 1062 dhcpmsg(MSG_ERROR, "attach_lif: entry for %s already exists!", 1063 lname); 1064 release_pif(pif); 1065 *error = DHCP_IPC_E_INVIF; 1066 return (NULL); 1067 } 1068 1069 /* If LIF creation fails, then insert_lif discards our PIF hold */ 1070 return (insert_lif(pif, lname, error)); 1071 } 1072 1073 /* 1074 * set_lif_dhcp(): Set logical interface flags to show that it's managed 1075 * by DHCP. 1076 * 1077 * input: dhcp_lif_t *: the logical interface 1078 * boolean_t: B_TRUE if adopting 1079 * output: int: set to DHCP_IPC_E_* if operation fails 1080 */ 1081 1082 int 1083 set_lif_dhcp(dhcp_lif_t *lif, boolean_t is_adopting) 1084 { 1085 int fd; 1086 int err; 1087 struct lifreq lifr; 1088 1089 fd = lif->lif_pif->pif_isv6 ? v6_sock_fd : v4_sock_fd; 1090 1091 (void) strlcpy(lifr.lifr_name, lif->lif_name, LIFNAMSIZ); 1092 1093 if (ioctl(fd, SIOCGLIFFLAGS, &lifr) == -1) { 1094 err = errno; 1095 dhcpmsg(MSG_ERR, "set_lif_dhcp: SIOCGLIFFLAGS for %s", 1096 lif->lif_name); 1097 return (err == ENXIO ? DHCP_IPC_E_INVIF : DHCP_IPC_E_INT); 1098 } 1099 lif->lif_flags = lifr.lifr_flags; 1100 1101 /* 1102 * Check for conflicting sources of address control, and other 1103 * unacceptable configurations. 1104 */ 1105 if (lifr.lifr_flags & (IFF_LOOPBACK|IFF_ADDRCONF|IFF_TEMPORARY| 1106 IFF_VIRTUAL)) { 1107 dhcpmsg(MSG_ERR, "set_lif_dhcp: cannot use %s: flags are %llx", 1108 lif->lif_name, lifr.lifr_flags); 1109 return (DHCP_IPC_E_INVIF); 1110 } 1111 1112 /* 1113 * if DHCPRUNNING is already set on the interface and we're 1114 * not adopting it, the agent probably crashed and burned. 1115 * note it, but don't let it stop the proceedings. we're 1116 * pretty sure we're not already running, since we wouldn't 1117 * have been able to bind to our IPC port. 1118 */ 1119 1120 if (lifr.lifr_flags & IFF_DHCPRUNNING) { 1121 if (!is_adopting) { 1122 dhcpmsg(MSG_WARNING, "set_lif_dhcp: DHCP flag already " 1123 "set on %s", lif->lif_name); 1124 } 1125 } else { 1126 lifr.lifr_flags |= IFF_DHCPRUNNING; 1127 if (ioctl(fd, SIOCSLIFFLAGS, &lifr) == -1) { 1128 dhcpmsg(MSG_ERR, "set_lif_dhcp: SIOCSLIFFLAGS for %s", 1129 lif->lif_name); 1130 return (DHCP_IPC_E_INT); 1131 } 1132 lif->lif_flags = lifr.lifr_flags; 1133 } 1134 return (DHCP_IPC_SUCCESS); 1135 } 1136 1137 /* 1138 * clear_lif_dhcp(): Clear logical interface flags to show that it's no longer 1139 * managed by DHCP. 1140 * 1141 * input: dhcp_lif_t *: the logical interface 1142 * output: none 1143 */ 1144 1145 static void 1146 clear_lif_dhcp(dhcp_lif_t *lif) 1147 { 1148 int fd; 1149 struct lifreq lifr; 1150 1151 fd = lif->lif_pif->pif_isv6 ? v6_sock_fd : v4_sock_fd; 1152 1153 (void) strlcpy(lifr.lifr_name, lif->lif_name, LIFNAMSIZ); 1154 1155 if (ioctl(fd, SIOCGLIFFLAGS, &lifr) == -1) 1156 return; 1157 1158 if (!(lifr.lifr_flags & IFF_DHCPRUNNING)) 1159 return; 1160 1161 lif->lif_flags = lifr.lifr_flags &= ~IFF_DHCPRUNNING; 1162 (void) ioctl(fd, SIOCSLIFFLAGS, &lifr); 1163 } 1164 1165 /* 1166 * set_lif_deprecated(): Set the "deprecated" flag to tell users that this 1167 * address will be going away. As the interface is 1168 * going away, we don't care if there are errors. 1169 * 1170 * input: dhcp_lif_t *: the logical interface 1171 * output: none 1172 */ 1173 1174 void 1175 set_lif_deprecated(dhcp_lif_t *lif) 1176 { 1177 int fd; 1178 struct lifreq lifr; 1179 1180 if (lif->lif_flags & IFF_DEPRECATED) 1181 return; 1182 1183 fd = lif->lif_pif->pif_isv6 ? v6_sock_fd : v4_sock_fd; 1184 1185 (void) strlcpy(lifr.lifr_name, lif->lif_name, LIFNAMSIZ); 1186 1187 if (ioctl(fd, SIOCGLIFFLAGS, &lifr) == -1) 1188 return; 1189 1190 if (lifr.lifr_flags & IFF_DEPRECATED) 1191 return; 1192 1193 lifr.lifr_flags |= IFF_DEPRECATED; 1194 (void) ioctl(fd, SIOCSLIFFLAGS, &lifr); 1195 lif->lif_flags = lifr.lifr_flags; 1196 } 1197 1198 /* 1199 * clear_lif_deprecated(): Clear the "deprecated" flag to tell users that this 1200 * address will not be going away. This happens if we 1201 * get a renewal after preferred lifetime but before 1202 * the valid lifetime. 1203 * 1204 * input: dhcp_lif_t *: the logical interface 1205 * output: boolean_t: B_TRUE on success. 1206 */ 1207 1208 boolean_t 1209 clear_lif_deprecated(dhcp_lif_t *lif) 1210 { 1211 int fd; 1212 struct lifreq lifr; 1213 1214 fd = lif->lif_pif->pif_isv6 ? v6_sock_fd : v4_sock_fd; 1215 1216 (void) strlcpy(lifr.lifr_name, lif->lif_name, LIFNAMSIZ); 1217 1218 if (ioctl(fd, SIOCGLIFFLAGS, &lifr) == -1) { 1219 dhcpmsg(MSG_ERR, "clear_lif_deprecated: SIOCGLIFFLAGS for %s", 1220 lif->lif_name); 1221 return (B_FALSE); 1222 } 1223 1224 /* 1225 * Check for conflicting sources of address control, and other 1226 * unacceptable configurations. 1227 */ 1228 if (lifr.lifr_flags & (IFF_LOOPBACK|IFF_ADDRCONF|IFF_TEMPORARY| 1229 IFF_VIRTUAL)) { 1230 dhcpmsg(MSG_ERR, "clear_lif_deprecated: cannot use %s: flags " 1231 "are %llx", lif->lif_name, lifr.lifr_flags); 1232 return (B_FALSE); 1233 } 1234 1235 if (!(lifr.lifr_flags & IFF_DEPRECATED)) 1236 return (B_TRUE); 1237 1238 lifr.lifr_flags &= ~IFF_DEPRECATED; 1239 if (ioctl(fd, SIOCSLIFFLAGS, &lifr) == -1) { 1240 dhcpmsg(MSG_ERR, "clear_lif_deprecated: SIOCSLIFFLAGS for %s", 1241 lif->lif_name); 1242 return (B_FALSE); 1243 } else { 1244 lif->lif_flags = lifr.lifr_flags; 1245 return (B_TRUE); 1246 } 1247 } 1248 1249 /* 1250 * open_ip_lif(): open up an IP socket for I/O on a given LIF (v4 only). 1251 * 1252 * input: dhcp_lif_t *: the logical interface to operate on 1253 * in_addr_t: the address the socket will be bound to (in hbo) 1254 * output: boolean_t: B_TRUE if the socket was opened successfully. 1255 */ 1256 1257 boolean_t 1258 open_ip_lif(dhcp_lif_t *lif, in_addr_t addr_hbo) 1259 { 1260 const char *errmsg; 1261 struct lifreq lifr; 1262 int on = 1; 1263 1264 if (lif->lif_sock_ip_fd != -1) { 1265 dhcpmsg(MSG_WARNING, "open_ip_lif: socket already open on %s", 1266 lif->lif_name); 1267 return (B_FALSE); 1268 } 1269 1270 lif->lif_sock_ip_fd = socket(AF_INET, SOCK_DGRAM, 0); 1271 if (lif->lif_sock_ip_fd == -1) { 1272 errmsg = "cannot create v4 socket"; 1273 goto failure; 1274 } 1275 1276 if (!bind_sock(lif->lif_sock_ip_fd, IPPORT_BOOTPC, addr_hbo)) { 1277 errmsg = "cannot bind v4 socket"; 1278 goto failure; 1279 } 1280 1281 /* 1282 * If we bound to INADDR_ANY, we have no IFF_UP source address to use. 1283 * Thus, enable IP_UNSPEC_SRC so that we can send packets with an 1284 * unspecified (0.0.0.0) address. Also, enable IP_DHCPINIT_IF so that 1285 * the IP module will accept unicast DHCP traffic regardless of the IP 1286 * address it's sent to. (We'll then figure out which packets are 1287 * ours based on the xid.) 1288 */ 1289 if (addr_hbo == INADDR_ANY) { 1290 if (setsockopt(lif->lif_sock_ip_fd, IPPROTO_IP, IP_UNSPEC_SRC, 1291 &on, sizeof (int)) == -1) { 1292 errmsg = "cannot set IP_UNSPEC_SRC"; 1293 goto failure; 1294 } 1295 1296 if (setsockopt(lif->lif_sock_ip_fd, IPPROTO_IP, IP_DHCPINIT_IF, 1297 &lif->lif_pif->pif_index, sizeof (int)) == -1) { 1298 errmsg = "cannot set IP_DHCPINIT_IF"; 1299 goto failure; 1300 } 1301 } 1302 1303 if (setsockopt(lif->lif_sock_ip_fd, IPPROTO_IP, IP_BOUND_IF, 1304 &lif->lif_pif->pif_index, sizeof (int)) == -1) { 1305 errmsg = "cannot set IP_BOUND_IF"; 1306 goto failure; 1307 } 1308 1309 /* 1310 * Make sure at least one lif on the interface we used in IP_BOUND_IF 1311 * is IFF_UP so that we can send and receive IP packets. 1312 */ 1313 (void) strlcpy(lifr.lifr_name, lif->lif_name, LIFNAMSIZ); 1314 if (ioctl(v4_sock_fd, SIOCGLIFFLAGS, &lifr) == -1) { 1315 errmsg = "cannot get interface flags"; 1316 goto failure; 1317 } 1318 1319 if (!(lifr.lifr_flags & IFF_UP)) { 1320 /* 1321 * Start from a clean slate. 1322 */ 1323 canonize_lif(lif, B_FALSE); 1324 1325 lifr.lifr_flags |= IFF_UP; 1326 if (ioctl(v4_sock_fd, SIOCSLIFFLAGS, &lifr) == -1) { 1327 errmsg = "cannot bring up"; 1328 goto failure; 1329 } 1330 1331 /* 1332 * When bringing 0.0.0.0 IFF_UP, the kernel changes the 1333 * netmask to 255.0.0.0, so re-fetch our expected netmask. 1334 */ 1335 if (ioctl(v4_sock_fd, SIOCGLIFNETMASK, &lifr) == -1) { 1336 errmsg = "cannot get netmask"; 1337 goto failure; 1338 } 1339 1340 lif->lif_netmask = 1341 ((struct sockaddr_in *)&lifr.lifr_addr)->sin_addr.s_addr; 1342 } 1343 1344 lif->lif_packet_id = iu_register_event(eh, lif->lif_sock_ip_fd, POLLIN, 1345 dhcp_packet_lif, lif); 1346 if (lif->lif_packet_id == -1) { 1347 errmsg = "cannot register to receive DHCP packets"; 1348 goto failure; 1349 } 1350 1351 return (B_TRUE); 1352 failure: 1353 dhcpmsg(MSG_ERR, "open_ip_lif: %s: %s", lif->lif_name, errmsg); 1354 close_ip_lif(lif); 1355 return (B_FALSE); 1356 } 1357 1358 /* 1359 * close_ip_lif(): close an IP socket for I/O on a given LIF. 1360 * 1361 * input: dhcp_lif_t *: the logical interface to operate on 1362 * output: none 1363 */ 1364 1365 void 1366 close_ip_lif(dhcp_lif_t *lif) 1367 { 1368 if (lif->lif_packet_id != -1) { 1369 (void) iu_unregister_event(eh, lif->lif_packet_id, NULL); 1370 lif->lif_packet_id = -1; 1371 } 1372 if (lif->lif_sock_ip_fd != -1) { 1373 (void) close(lif->lif_sock_ip_fd); 1374 lif->lif_sock_ip_fd = -1; 1375 } 1376 } 1377 1378 /* 1379 * lif_mark_decline(): mark a LIF as having been declined due to a duplicate 1380 * address or some other conflict. This is used in 1381 * send_declines() to report failure back to the server. 1382 * 1383 * input: dhcp_lif_t *: the logical interface to operate on 1384 * const char *: text string explaining why the address is declined 1385 * output: none 1386 */ 1387 1388 void 1389 lif_mark_decline(dhcp_lif_t *lif, const char *reason) 1390 { 1391 if (lif->lif_declined == NULL) { 1392 dhcp_lease_t *dlp; 1393 1394 lif->lif_declined = reason; 1395 if ((dlp = lif->lif_lease) != NULL) 1396 dlp->dl_smach->dsm_lif_down++; 1397 } 1398 } 1399 1400 /* 1401 * schedule_lif_timer(): schedules the LIF-related timer 1402 * 1403 * input: dhcp_lif_t *: the logical interface to operate on 1404 * dhcp_timer_t *: the timer to schedule 1405 * iu_tq_callback_t *: the callback to call upon firing 1406 * output: boolean_t: B_TRUE if the timer was scheduled successfully 1407 */ 1408 1409 boolean_t 1410 schedule_lif_timer(dhcp_lif_t *lif, dhcp_timer_t *dt, iu_tq_callback_t *expire) 1411 { 1412 /* 1413 * If there's a timer running, cancel it and release its lease 1414 * reference. 1415 */ 1416 if (dt->dt_id != -1) { 1417 if (!cancel_timer(dt)) 1418 return (B_FALSE); 1419 release_lif(lif); 1420 } 1421 1422 if (schedule_timer(dt, expire, lif)) { 1423 hold_lif(lif); 1424 return (B_TRUE); 1425 } else { 1426 dhcpmsg(MSG_WARNING, 1427 "schedule_lif_timer: cannot schedule timer"); 1428 return (B_FALSE); 1429 } 1430 } 1431 1432 /* 1433 * cancel_lif_timer(): cancels a LIF-related timer 1434 * 1435 * input: dhcp_lif_t *: the logical interface to operate on 1436 * dhcp_timer_t *: the timer to cancel 1437 * output: none 1438 */ 1439 1440 static void 1441 cancel_lif_timer(dhcp_lif_t *lif, dhcp_timer_t *dt) 1442 { 1443 if (dt->dt_id == -1) 1444 return; 1445 if (cancel_timer(dt)) { 1446 dhcpmsg(MSG_DEBUG2, 1447 "cancel_lif_timer: canceled expiry timer on %s", 1448 lif->lif_name); 1449 release_lif(lif); 1450 } else { 1451 dhcpmsg(MSG_WARNING, 1452 "cancel_lif_timer: cannot cancel timer on %s", 1453 lif->lif_name); 1454 } 1455 } 1456 1457 /* 1458 * cancel_lif_timers(): cancels the LIF-related timers 1459 * 1460 * input: dhcp_lif_t *: the logical interface to operate on 1461 * output: none 1462 */ 1463 1464 void 1465 cancel_lif_timers(dhcp_lif_t *lif) 1466 { 1467 cancel_lif_timer(lif, &lif->lif_preferred); 1468 cancel_lif_timer(lif, &lif->lif_expire); 1469 } 1470 1471 /* 1472 * get_max_mtu(): find the maximum MTU of all interfaces for I/O on common 1473 * file descriptors (v4_sock_fd and v6_sock_fd). 1474 * 1475 * input: boolean_t: B_TRUE for IPv6, B_FALSE for IPv4 1476 * output: none 1477 */ 1478 1479 uint_t 1480 get_max_mtu(boolean_t isv6) 1481 { 1482 uint_t *mtup = isv6 ? &cached_v6_max_mtu : &cached_v4_max_mtu; 1483 1484 if (*mtup == 0) { 1485 dhcp_pif_t *pif; 1486 dhcp_lif_t *lif; 1487 struct lifreq lifr; 1488 1489 /* Set an arbitrary lower bound */ 1490 *mtup = 1024; 1491 pif = isv6 ? v6root : v4root; 1492 for (; pif != NULL; pif = pif->pif_next) { 1493 for (lif = pif->pif_lifs; lif != NULL; 1494 lif = lif->lif_next) { 1495 (void) strlcpy(lifr.lifr_name, lif->lif_name, 1496 LIFNAMSIZ); 1497 if (ioctl(v4_sock_fd, SIOCGLIFMTU, &lifr) != 1498 -1 && lifr.lifr_mtu > *mtup) { 1499 *mtup = lifr.lifr_mtu; 1500 } 1501 } 1502 } 1503 } 1504 return (*mtup); 1505 } 1506 1507 /* 1508 * expired_lif_state(): summarize the state of expired LIFs on a given state 1509 * machine. 1510 * 1511 * input: dhcp_smach_t *: the state machine to scan 1512 * output: dhcp_expire_t: overall state 1513 */ 1514 1515 dhcp_expire_t 1516 expired_lif_state(dhcp_smach_t *dsmp) 1517 { 1518 dhcp_lease_t *dlp; 1519 dhcp_lif_t *lif; 1520 uint_t nlifs; 1521 uint_t numlifs; 1522 uint_t numexp; 1523 1524 numlifs = numexp = 0; 1525 for (dlp = dsmp->dsm_leases; dlp != NULL; dlp = dlp->dl_next) { 1526 lif = dlp->dl_lifs; 1527 nlifs = dlp->dl_nlifs; 1528 numlifs += nlifs; 1529 for (; nlifs > 0; nlifs--, lif = lif->lif_next) { 1530 if (lif->lif_expired) 1531 numexp++; 1532 } 1533 } 1534 if (numlifs == 0) 1535 return (DHCP_EXP_NOLIFS); 1536 else if (numexp == 0) 1537 return (DHCP_EXP_NOEXP); 1538 else if (numlifs == numexp) 1539 return (DHCP_EXP_ALLEXP); 1540 else 1541 return (DHCP_EXP_SOMEEXP); 1542 } 1543 1544 /* 1545 * find_expired_lif(): find the first expired LIF on a given state machine 1546 * 1547 * input: dhcp_smach_t *: the state machine to scan 1548 * output: dhcp_lif_t *: the first expired LIF, or NULL if none. 1549 */ 1550 1551 dhcp_lif_t * 1552 find_expired_lif(dhcp_smach_t *dsmp) 1553 { 1554 dhcp_lease_t *dlp; 1555 dhcp_lif_t *lif; 1556 uint_t nlifs; 1557 1558 for (dlp = dsmp->dsm_leases; dlp != NULL; dlp = dlp->dl_next) { 1559 lif = dlp->dl_lifs; 1560 nlifs = dlp->dl_nlifs; 1561 for (; nlifs > 0; nlifs--, lif = lif->lif_next) { 1562 if (lif->lif_expired) 1563 return (lif); 1564 } 1565 } 1566 return (NULL); 1567 } 1568 1569 /* 1570 * remove_v6_strays(): remove any stray interfaces marked as DHCPRUNNING. Used 1571 * only for DHCPv6. 1572 * 1573 * input: none 1574 * output: none 1575 */ 1576 1577 void 1578 remove_v6_strays(void) 1579 { 1580 struct lifnum lifn; 1581 struct lifconf lifc; 1582 struct lifreq *lifrp, *lifrmax; 1583 uint_t numifs; 1584 uint64_t flags; 1585 1586 /* 1587 * Get the approximate number of interfaces in the system. It's only 1588 * approximate because the system is dynamic -- interfaces may be 1589 * plumbed or unplumbed at any time. This is also the reason for the 1590 * "+ 10" fudge factor: we're trying to avoid unnecessary looping. 1591 */ 1592 (void) memset(&lifn, 0, sizeof (lifn)); 1593 lifn.lifn_family = AF_INET6; 1594 lifn.lifn_flags = LIFC_ALLZONES | LIFC_NOXMIT | LIFC_TEMPORARY; 1595 if (ioctl(v6_sock_fd, SIOCGLIFNUM, &lifn) == -1) { 1596 dhcpmsg(MSG_ERR, 1597 "remove_v6_strays: cannot read number of interfaces"); 1598 numifs = 10; 1599 } else { 1600 numifs = lifn.lifn_count + 10; 1601 } 1602 1603 /* 1604 * Get the interface information. We do this in a loop so that we can 1605 * recover from EINVAL from the kernel -- delivered when the buffer is 1606 * too small. 1607 */ 1608 (void) memset(&lifc, 0, sizeof (lifc)); 1609 lifc.lifc_family = AF_INET6; 1610 lifc.lifc_flags = LIFC_ALLZONES | LIFC_NOXMIT | LIFC_TEMPORARY; 1611 for (;;) { 1612 lifc.lifc_len = numifs * sizeof (*lifrp); 1613 lifrp = realloc(lifc.lifc_buf, lifc.lifc_len); 1614 if (lifrp == NULL) { 1615 dhcpmsg(MSG_ERR, 1616 "remove_v6_strays: cannot allocate memory"); 1617 free(lifc.lifc_buf); 1618 return; 1619 } 1620 lifc.lifc_buf = (caddr_t)lifrp; 1621 errno = 0; 1622 if (ioctl(v6_sock_fd, SIOCGLIFCONF, &lifc) == 0 && 1623 lifc.lifc_len < numifs * sizeof (*lifrp)) 1624 break; 1625 if (errno == 0 || errno == EINVAL) { 1626 numifs <<= 1; 1627 } else { 1628 dhcpmsg(MSG_ERR, "remove_v6_strays: SIOCGLIFCONF"); 1629 free(lifc.lifc_buf); 1630 return; 1631 } 1632 } 1633 1634 lifrmax = lifrp + lifc.lifc_len / sizeof (*lifrp); 1635 for (; lifrp < lifrmax; lifrp++) { 1636 /* 1637 * Get the interface flags; we're interested in the DHCP ones. 1638 */ 1639 if (ioctl(v6_sock_fd, SIOCGLIFFLAGS, lifrp) == -1) 1640 continue; 1641 flags = lifrp->lifr_flags; 1642 if (!(flags & IFF_DHCPRUNNING)) 1643 continue; 1644 /* 1645 * If the interface has a link-local address, then we don't 1646 * control it. Just remove the flag. 1647 */ 1648 if (ioctl(v6_sock_fd, SIOCGLIFADDR, lifrp) == -1) 1649 continue; 1650 if (IN6_IS_ADDR_LINKLOCAL(&((struct sockaddr_in6 *)&lifrp-> 1651 lifr_addr)->sin6_addr)) { 1652 lifrp->lifr_flags = flags & ~IFF_DHCPRUNNING; 1653 (void) ioctl(v6_sock_fd, SIOCSLIFFLAGS, lifrp); 1654 continue; 1655 } 1656 /* 1657 * All others are (or were) under our control. Clean up by 1658 * removing them. 1659 */ 1660 if (ioctl(v6_sock_fd, SIOCLIFREMOVEIF, lifrp) == 0) { 1661 dhcpmsg(MSG_DEBUG, "remove_v6_strays: removed %s", 1662 lifrp->lifr_name); 1663 } else if (errno != ENXIO) { 1664 dhcpmsg(MSG_ERR, 1665 "remove_v6_strays: SIOCLIFREMOVEIF %s", 1666 lifrp->lifr_name); 1667 } 1668 } 1669 free(lifc.lifc_buf); 1670 } 1671