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 2009 Sun Microsystems, Inc. All rights reserved. 23 * Use is subject to license terms. 24 */ 25 26 27 /* 28 * Config routines common to idmap(1M) and idmapd(1M) 29 */ 30 31 #include <stdlib.h> 32 #include <strings.h> 33 #include <libintl.h> 34 #include <ctype.h> 35 #include <errno.h> 36 #include "idmapd.h" 37 #include <stdio.h> 38 #include <stdarg.h> 39 #include <uuid/uuid.h> 40 #include <pthread.h> 41 #include <port.h> 42 #include <net/route.h> 43 #include <sys/u8_textprep.h> 44 #include "addisc.h" 45 46 #define MACHINE_SID_LEN (9 + 3 * 11) 47 #define FMRI_BASE "svc:/system/idmap" 48 #define CONFIG_PG "config" 49 #define GENERAL_PG "general" 50 #define RECONFIGURE 1 51 #define POKE_AUTO_DISCOVERY 2 52 53 /*LINTLIBRARY*/ 54 55 56 static pthread_t update_thread_handle = 0; 57 58 static int idmapd_ev_port = -1; 59 static int rt_sock = -1; 60 61 static int 62 generate_machine_sid(char **machine_sid) 63 { 64 char *p; 65 uuid_t uu; 66 int i, j, len, rlen; 67 uint32_t rid; 68 69 /* 70 * Generate and split 128-bit UUID into three 32-bit RIDs The 71 * machine_sid will be of the form S-1-5-21-N1-N2-N3 (that's 72 * four RIDs altogether). 73 * 74 * Technically we could use upto 14 random RIDs here, but it 75 * turns out that with some versions of Windows using SIDs with 76 * more than five RIDs in security descriptors causes problems. 77 */ 78 79 *machine_sid = calloc(1, MACHINE_SID_LEN); 80 if (*machine_sid == NULL) { 81 idmapdlog(LOG_ERR, "Out of memory"); 82 return (-1); 83 } 84 (void) strcpy(*machine_sid, "S-1-5-21"); 85 p = *machine_sid + strlen("S-1-5-21"); 86 len = MACHINE_SID_LEN - strlen("S-1-5-21"); 87 88 uuid_clear(uu); 89 uuid_generate_random(uu); 90 91 #if UUID_LEN != 16 92 #error UUID size is not 16! 93 #endif 94 95 for (i = 0; i < 3; i++) { 96 j = i * 4; 97 rid = (uu[j] << 24) | (uu[j + 1] << 16) | 98 (uu[j + 2] << 8) | (uu[j + 3]); 99 rlen = snprintf(p, len, "-%u", rid); 100 p += rlen; 101 len -= rlen; 102 } 103 104 return (0); 105 } 106 107 108 /* In the case of error, exists is set to FALSE anyway */ 109 static int 110 prop_exists(idmap_cfg_handles_t *handles, char *name, boolean_t *exists) 111 { 112 113 scf_property_t *scf_prop; 114 scf_value_t *value; 115 116 *exists = B_FALSE; 117 118 scf_prop = scf_property_create(handles->main); 119 if (scf_prop == NULL) { 120 idmapdlog(LOG_ERR, "scf_property_create() failed: %s", 121 scf_strerror(scf_error())); 122 return (-1); 123 } 124 value = scf_value_create(handles->main); 125 if (value == NULL) { 126 idmapdlog(LOG_ERR, "scf_value_create() failed: %s", 127 scf_strerror(scf_error())); 128 scf_property_destroy(scf_prop); 129 return (-1); 130 } 131 132 if (scf_pg_get_property(handles->config_pg, name, scf_prop) == 0) 133 *exists = B_TRUE; 134 135 scf_value_destroy(value); 136 scf_property_destroy(scf_prop); 137 138 return (0); 139 } 140 141 /* Check if in the case of failure the original value of *val is preserved */ 142 static int 143 get_val_int(idmap_cfg_handles_t *handles, char *name, 144 void *val, scf_type_t type) 145 { 146 int rc = 0; 147 148 scf_property_t *scf_prop; 149 scf_value_t *value; 150 151 switch (type) { 152 case SCF_TYPE_BOOLEAN: 153 *(uint8_t *)val = 0; 154 break; 155 case SCF_TYPE_COUNT: 156 *(uint64_t *)val = 0; 157 break; 158 case SCF_TYPE_INTEGER: 159 *(int64_t *)val = 0; 160 break; 161 default: 162 idmapdlog(LOG_ERR, "Invalid scf integer type (%d)", 163 type); 164 abort(); 165 } 166 167 scf_prop = scf_property_create(handles->main); 168 if (scf_prop == NULL) { 169 idmapdlog(LOG_ERR, "scf_property_create() failed: %s", 170 scf_strerror(scf_error())); 171 return (-1); 172 } 173 value = scf_value_create(handles->main); 174 if (value == NULL) { 175 idmapdlog(LOG_ERR, "scf_value_create() failed: %s", 176 scf_strerror(scf_error())); 177 scf_property_destroy(scf_prop); 178 return (-1); 179 } 180 181 if (scf_pg_get_property(handles->config_pg, name, scf_prop) < 0) 182 /* this is OK: the property is just undefined */ 183 goto destruction; 184 185 186 if (scf_property_get_value(scf_prop, value) < 0) 187 /* It is still OK when a property doesn't have any value */ 188 goto destruction; 189 190 switch (type) { 191 case SCF_TYPE_BOOLEAN: 192 rc = scf_value_get_boolean(value, val); 193 break; 194 case SCF_TYPE_COUNT: 195 rc = scf_value_get_count(value, val); 196 break; 197 case SCF_TYPE_INTEGER: 198 rc = scf_value_get_integer(value, val); 199 break; 200 } 201 202 203 destruction: 204 scf_value_destroy(value); 205 scf_property_destroy(scf_prop); 206 207 return (rc); 208 } 209 210 static char * 211 scf_value2string(scf_value_t *value) 212 { 213 int rc = -1; 214 char buf_size = 127; 215 int length; 216 char *buf = NULL; 217 buf = (char *) malloc(sizeof (char) * buf_size); 218 219 for (;;) { 220 length = scf_value_get_astring(value, buf, buf_size); 221 if (length < 0) { 222 rc = -1; 223 goto destruction; 224 } 225 226 if (length == buf_size - 1) { 227 buf_size *= 2; 228 buf = (char *)realloc(buf, buf_size * sizeof (char)); 229 if (!buf) { 230 idmapdlog(LOG_ERR, "Out of memory"); 231 rc = -1; 232 goto destruction; 233 } 234 } else { 235 rc = 0; 236 break; 237 } 238 } 239 240 destruction: 241 if (rc < 0) { 242 if (buf) 243 free(buf); 244 buf = NULL; 245 } 246 247 return (buf); 248 } 249 250 static int 251 get_val_ds(idmap_cfg_handles_t *handles, const char *name, int defport, 252 idmap_ad_disc_ds_t **val) 253 { 254 idmap_ad_disc_ds_t *servers = NULL; 255 scf_property_t *scf_prop; 256 scf_value_t *value; 257 scf_iter_t *iter; 258 char *host, *portstr; 259 int len, i; 260 int count = 0; 261 int rc = -1; 262 263 *val = NULL; 264 265 restart: 266 scf_prop = scf_property_create(handles->main); 267 if (scf_prop == NULL) { 268 idmapdlog(LOG_ERR, "scf_property_create() failed: %s", 269 scf_strerror(scf_error())); 270 return (-1); 271 } 272 273 value = scf_value_create(handles->main); 274 if (value == NULL) { 275 idmapdlog(LOG_ERR, "scf_value_create() failed: %s", 276 scf_strerror(scf_error())); 277 scf_property_destroy(scf_prop); 278 return (-1); 279 } 280 281 iter = scf_iter_create(handles->main); 282 if (iter == NULL) { 283 idmapdlog(LOG_ERR, "scf_iter_create() failed: %s", 284 scf_strerror(scf_error())); 285 scf_value_destroy(value); 286 scf_property_destroy(scf_prop); 287 return (-1); 288 } 289 290 if (scf_pg_get_property(handles->config_pg, name, scf_prop) < 0) { 291 /* this is OK: the property is just undefined */ 292 rc = 0; 293 goto destruction; 294 } 295 296 if (scf_iter_property_values(iter, scf_prop) < 0) { 297 idmapdlog(LOG_ERR, 298 "scf_iter_property_values(%s) failed: %s", 299 name, scf_strerror(scf_error())); 300 goto destruction; 301 } 302 303 /* Workaround scf bugs -- can't reset an iteration */ 304 if (count == 0) { 305 while (scf_iter_next_value(iter, value) > 0) 306 count++; 307 308 if (count == 0) { 309 /* no values */ 310 rc = 0; 311 goto destruction; 312 } 313 314 scf_value_destroy(value); 315 scf_iter_destroy(iter); 316 scf_property_destroy(scf_prop); 317 goto restart; 318 } 319 320 if ((servers = calloc(count + 1, sizeof (*servers))) == NULL) { 321 idmapdlog(LOG_ERR, "Out of memory"); 322 goto destruction; 323 } 324 325 i = 0; 326 while (i < count && scf_iter_next_value(iter, value) > 0) { 327 servers[i].priority = 0; 328 servers[i].weight = 100; 329 servers[i].port = defport; 330 if ((host = scf_value2string(value)) == NULL) { 331 goto destruction; 332 } 333 if ((portstr = strchr(host, ':')) != NULL) { 334 *portstr++ = '\0'; 335 servers[i].port = strtol(portstr, 336 (char **)NULL, 10); 337 if (servers[i].port == 0) 338 servers[i].port = defport; 339 } 340 len = strlcpy(servers[i].host, host, 341 sizeof (servers->host)); 342 343 free(host); 344 345 /* Ignore this server if the hostname is too long */ 346 if (len < sizeof (servers->host)) 347 i++; 348 } 349 350 *val = servers; 351 352 rc = 0; 353 354 destruction: 355 scf_value_destroy(value); 356 scf_iter_destroy(iter); 357 scf_property_destroy(scf_prop); 358 359 if (rc < 0) { 360 if (servers) 361 free(servers); 362 *val = NULL; 363 } 364 365 return (rc); 366 } 367 368 369 static int 370 get_val_astring(idmap_cfg_handles_t *handles, char *name, char **val) 371 { 372 int rc = 0; 373 374 scf_property_t *scf_prop; 375 scf_value_t *value; 376 377 scf_prop = scf_property_create(handles->main); 378 if (scf_prop == NULL) { 379 idmapdlog(LOG_ERR, "scf_property_create() failed: %s", 380 scf_strerror(scf_error())); 381 return (-1); 382 } 383 value = scf_value_create(handles->main); 384 if (value == NULL) { 385 idmapdlog(LOG_ERR, "scf_value_create() failed: %s", 386 scf_strerror(scf_error())); 387 scf_property_destroy(scf_prop); 388 return (-1); 389 } 390 391 *val = NULL; 392 393 if (scf_pg_get_property(handles->config_pg, name, scf_prop) < 0) 394 /* this is OK: the property is just undefined */ 395 goto destruction; 396 397 if (scf_property_get_value(scf_prop, value) < 0) { 398 idmapdlog(LOG_ERR, 399 "scf_property_get_value(%s) failed: %s", 400 name, scf_strerror(scf_error())); 401 rc = -1; 402 goto destruction; 403 } 404 405 if (!(*val = scf_value2string(value))) 406 rc = -1; 407 408 destruction: 409 scf_value_destroy(value); 410 scf_property_destroy(scf_prop); 411 412 if (rc < 0) { 413 if (*val) 414 free(*val); 415 *val = NULL; 416 } 417 418 return (rc); 419 } 420 421 422 static int 423 set_val_astring(idmap_cfg_handles_t *handles, char *name, const char *val) 424 { 425 int rc = -1; 426 int ret = -2; 427 int i; 428 scf_property_t *scf_prop = NULL; 429 scf_value_t *value = NULL; 430 scf_transaction_t *tx = NULL; 431 scf_transaction_entry_t *ent = NULL; 432 433 if ((scf_prop = scf_property_create(handles->main)) == NULL || 434 (value = scf_value_create(handles->main)) == NULL || 435 (tx = scf_transaction_create(handles->main)) == NULL || 436 (ent = scf_entry_create(handles->main)) == NULL) { 437 idmapdlog(LOG_ERR, "Unable to set property %s", 438 name, scf_strerror(scf_error())); 439 goto destruction; 440 } 441 442 for (i = 0; i < MAX_TRIES && (ret == -2 || ret == 0); i++) { 443 if (scf_transaction_start(tx, handles->config_pg) == -1) { 444 idmapdlog(LOG_ERR, 445 "scf_transaction_start(%s) failed: %s", 446 name, scf_strerror(scf_error())); 447 goto destruction; 448 } 449 450 if (scf_transaction_property_new(tx, ent, name, 451 SCF_TYPE_ASTRING) < 0) { 452 idmapdlog(LOG_ERR, 453 "scf_transaction_property_new() failed: %s", 454 scf_strerror(scf_error())); 455 goto destruction; 456 } 457 458 if (scf_value_set_astring(value, val) == -1) { 459 idmapdlog(LOG_ERR, 460 "scf_value_set_astring() failed: %s", 461 scf_strerror(scf_error())); 462 goto destruction; 463 } 464 465 if (scf_entry_add_value(ent, value) == -1) { 466 idmapdlog(LOG_ERR, 467 "scf_entry_add_value() failed: %s", 468 scf_strerror(scf_error())); 469 goto destruction; 470 } 471 472 if ((ret = scf_transaction_commit(tx)) == 1) 473 break; 474 475 if (ret == 0 && i < MAX_TRIES - 1) { 476 /* 477 * Property group set in scf_transaction_start() 478 * is not the most recent. Update pg, reset tx and 479 * retry tx. 480 */ 481 idmapdlog(LOG_WARNING, 482 "scf_transaction_commit(%s) failed - Retry: %s", 483 name, scf_strerror(scf_error())); 484 if (scf_pg_update(handles->config_pg) == -1) { 485 idmapdlog(LOG_ERR, 486 "scf_pg_update() failed: %s", 487 scf_strerror(scf_error())); 488 goto destruction; 489 } 490 scf_transaction_reset(tx); 491 } 492 } 493 494 495 if (ret == 1) 496 rc = 0; 497 else if (ret != -2) 498 idmapdlog(LOG_ERR, 499 "scf_transaction_commit(%s) failed: %s", 500 name, scf_strerror(scf_error())); 501 502 destruction: 503 scf_value_destroy(value); 504 scf_entry_destroy(ent); 505 scf_transaction_destroy(tx); 506 scf_property_destroy(scf_prop); 507 return (rc); 508 } 509 510 511 512 /* 513 * This function updates a boolean value. 514 * If nothing has changed it returns 0 else 1 515 */ 516 static int 517 update_bool(boolean_t *value, boolean_t *new, char *name) 518 { 519 if (*value == *new) 520 return (0); 521 522 idmapdlog(LOG_INFO, "change %s=%s", name, *new ? "true" : "false"); 523 *value = *new; 524 return (1); 525 } 526 527 528 /* 529 * This function updates a string value. 530 * If nothing has changed it returns 0 else 1 531 */ 532 static int 533 update_string(char **value, char **new, char *name) 534 { 535 if (*new == NULL) 536 return (0); 537 538 if (*value != NULL && strcmp(*new, *value) == 0) { 539 free(*new); 540 *new = NULL; 541 return (0); 542 } 543 544 idmapdlog(LOG_INFO, "change %s=%s", name, CHECK_NULL(*new)); 545 if (*value != NULL) 546 free(*value); 547 *value = *new; 548 *new = NULL; 549 return (1); 550 } 551 552 553 /* 554 * This function updates a directory service structure. 555 * If nothing has changed it returns 0 else 1 556 */ 557 static int 558 update_dirs(idmap_ad_disc_ds_t **value, idmap_ad_disc_ds_t **new, char *name) 559 { 560 int i; 561 562 if (*value == *new) 563 /* Nothing to do */ 564 return (0); 565 566 if (*value != NULL && *new != NULL && 567 ad_disc_compare_ds(*value, *new) == 0) { 568 free(*new); 569 *new = NULL; 570 return (0); 571 } 572 573 if (*value != NULL) 574 free(*value); 575 576 *value = *new; 577 *new = NULL; 578 579 if (*value == NULL) { 580 /* We're unsetting this DS property */ 581 idmapdlog(LOG_INFO, "change %s=<none>", name); 582 return (1); 583 } 584 585 /* List all the new DSs */ 586 for (i = 0; (*value)[i].host[0] != '\0'; i++) 587 idmapdlog(LOG_INFO, "change %s=%s port=%d", name, 588 (*value)[i].host, (*value)[i].port); 589 return (1); 590 } 591 592 /* 593 * This function updates a trusted domains structure. 594 * If nothing has changed it returns 0 else 1 595 */ 596 static int 597 update_trusted_domains(ad_disc_trusteddomains_t **value, 598 ad_disc_trusteddomains_t **new, char *name) 599 { 600 int i; 601 602 if (*value == *new) 603 /* Nothing to do */ 604 return (0); 605 606 if (*value != NULL && *new != NULL && 607 ad_disc_compare_trusteddomains(*value, *new) == 0) { 608 free(*new); 609 *new = NULL; 610 return (0); 611 } 612 613 if (*value != NULL) 614 free(*value); 615 616 *value = *new; 617 *new = NULL; 618 619 if (*value == NULL) { 620 /* We're unsetting this DS property */ 621 idmapdlog(LOG_INFO, "change %s=<none>", name); 622 return (1); 623 } 624 625 /* List all the new domains */ 626 for (i = 0; (*value)[i].domain[0] != '\0'; i++) 627 idmapdlog(LOG_INFO, "change %s=%s direction=%s", name, 628 (*value)[i].domain, 629 (*value)[i].direction == 3 ? "bi-directional" : "inbound"); 630 return (1); 631 } 632 633 634 /* 635 * This function updates a domains in a forest structure. 636 * If nothing has changed it returns 0 else 1 637 */ 638 static int 639 update_domains_in_forest(ad_disc_domainsinforest_t **value, 640 ad_disc_domainsinforest_t **new, char *name) 641 { 642 int i; 643 644 if (*value == *new) 645 /* Nothing to do */ 646 return (0); 647 648 if (*value != NULL && *new != NULL && 649 ad_disc_compare_domainsinforest(*value, *new) == 0) { 650 free(*new); 651 *new = NULL; 652 return (0); 653 } 654 655 if (*value != NULL) 656 free(*value); 657 658 *value = *new; 659 *new = NULL; 660 661 if (*value == NULL) { 662 /* We're unsetting this DS property */ 663 idmapdlog(LOG_INFO, "change %s=<none>", name); 664 return (1); 665 } 666 667 /* List all the new domains */ 668 for (i = 0; (*value)[i].domain[0] != '\0'; i++) 669 idmapdlog(LOG_INFO, "change %s=%s", name, 670 (*value)[i].domain); 671 return (1); 672 } 673 674 675 static void 676 free_trusted_forests(idmap_trustedforest_t **value, int *num_values) 677 { 678 int i; 679 680 for (i = 0; i < *num_values; i++) { 681 free((*value)[i].forest_name); 682 free((*value)[i].global_catalog); 683 free((*value)[i].domains_in_forest); 684 } 685 free(*value); 686 *value = NULL; 687 *num_values = 0; 688 } 689 690 691 static int 692 compare_trusteddomainsinforest(ad_disc_domainsinforest_t *df1, 693 ad_disc_domainsinforest_t *df2) 694 { 695 int i, j; 696 int num_df1 = 0; 697 int num_df2 = 0; 698 boolean_t match; 699 int err; 700 701 for (i = 0; df1[i].domain[0] != '\0'; i++) 702 if (df1[i].trusted) 703 num_df1++; 704 705 for (j = 0; df2[j].domain[0] != '\0'; j++) 706 if (df2[j].trusted) 707 num_df2++; 708 709 if (num_df1 != num_df2) 710 return (1); 711 712 for (i = 0; df1[i].domain[0] != '\0'; i++) { 713 if (df1[i].trusted) { 714 match = B_FALSE; 715 for (j = 0; df2[j].domain[0] != '\0'; j++) { 716 if (df2[j].trusted && 717 u8_strcmp(df1[i].domain, df2[i].domain, 0, 718 U8_STRCMP_CI_LOWER, U8_UNICODE_LATEST, &err) 719 == 0 && err == 0 && 720 strcmp(df1[i].sid, df2[i].sid) == 0) { 721 match = B_TRUE; 722 break; 723 } 724 } 725 if (!match) 726 return (1); 727 } 728 } 729 return (0); 730 } 731 732 733 734 /* 735 * This function updates trusted forest structure. 736 * If nothing has changed it returns 0 else 1 737 */ 738 static int 739 update_trusted_forest(idmap_trustedforest_t **value, int *num_value, 740 idmap_trustedforest_t **new, int *num_new, char *name) 741 { 742 int i, j; 743 boolean_t match; 744 745 if (*value == *new) 746 /* Nothing to do */ 747 return (0); 748 749 if (*value != NULL && *new != NULL) { 750 if (*num_value != *num_new) 751 goto not_equal; 752 for (i = 0; i < *num_value; i++) { 753 match = B_FALSE; 754 for (j = 0; j < *num_new; j++) { 755 if (strcmp((*value)[i].forest_name, 756 (*new)[j].forest_name) == 0 && 757 ad_disc_compare_ds( 758 (*value)[i].global_catalog, 759 (*new)[i].global_catalog) == 0 && 760 compare_trusteddomainsinforest( 761 (*value)[i].domains_in_forest, 762 (*new)[i].domains_in_forest) == 0) { 763 match = B_TRUE; 764 break; 765 } 766 } 767 if (!match) 768 goto not_equal; 769 } 770 free_trusted_forests(new, num_new); 771 return (0); 772 } 773 not_equal: 774 if (*value != NULL) 775 free_trusted_forests(value, num_value); 776 *value = *new; 777 *num_value = *num_new; 778 *new = NULL; 779 *num_new = 0; 780 781 if (*value == NULL) { 782 /* We're unsetting this DS property */ 783 idmapdlog(LOG_INFO, "change %s=<none>", name); 784 return (1); 785 } 786 787 /* List all the trusted forests */ 788 for (i = 0; i < *num_value; i++) { 789 for (j = 0; (*value)[i].domains_in_forest[j].domain[0] != '\0'; 790 j++) { 791 /* List trusted Domains in the forest. */ 792 if ((*value)[i].domains_in_forest[j].trusted) 793 idmapdlog(LOG_INFO, "change %s=%s domain=%s", 794 name, (*value)[i].forest_name, 795 (*value)[i].domains_in_forest[j].domain); 796 } 797 /* List the hosts */ 798 for (j = 0; (*value)[i].global_catalog[j].host[0] != '\0'; j++) 799 idmapdlog(LOG_INFO, "change %s=%s host=%s port=%d", 800 name, (*value)[i].forest_name, 801 (*value)[i].global_catalog[j].host, 802 (*value)[i].global_catalog[j].port); 803 } 804 return (1); 805 } 806 807 808 #define MAX_CHECK_TIME (20 * 60) 809 810 /* 811 * Returns 1 if the PF_ROUTE socket event indicates that we should rescan the 812 * interfaces. 813 * 814 * Shamelessly based on smb_nics_changed() and other PF_ROUTE uses in ON. 815 */ 816 static 817 int 818 pfroute_event_is_interesting(int rt_sock) 819 { 820 int nbytes; 821 int64_t msg[2048 / 8]; 822 struct rt_msghdr *rtm; 823 int is_interesting = FALSE; 824 825 for (;;) { 826 if ((nbytes = read(rt_sock, msg, sizeof (msg))) <= 0) 827 break; 828 rtm = (struct rt_msghdr *)msg; 829 if (rtm->rtm_version != RTM_VERSION) 830 continue; 831 if (nbytes < rtm->rtm_msglen) 832 continue; 833 switch (rtm->rtm_type) { 834 case RTM_NEWADDR: 835 case RTM_DELADDR: 836 case RTM_IFINFO: 837 is_interesting = TRUE; 838 break; 839 default: 840 break; 841 } 842 } 843 return (is_interesting); 844 } 845 846 /* 847 * Returns 1 if SIGHUP has been received (see hup_handler() elsewhere) or if an 848 * interface address was added or removed; otherwise it returns 0. 849 * 850 * Note that port_get() does not update its timeout argument when EINTR, unlike 851 * nanosleep(). We probably don't care very much here, but if we did care then 852 * we could always use a timer event and associate it with the same event port, 853 * then we could get accurate waiting regardless of EINTRs. 854 */ 855 static 856 int 857 wait_for_event(int poke_is_interesting, struct timespec *timeoutp) 858 { 859 port_event_t pe; 860 861 retry: 862 memset(&pe, 0, sizeof (pe)); 863 if (port_get(idmapd_ev_port, &pe, timeoutp) != 0) { 864 switch (errno) { 865 case EINTR: 866 goto retry; 867 case ETIME: 868 /* Timeout */ 869 return (FALSE); 870 default: 871 /* EBADF, EBADFD, EFAULT, EINVAL (end of time?)? */ 872 idmapdlog(LOG_ERR, "Event port failed: %s", 873 strerror(errno)); 874 exit(1); 875 /* NOTREACHED */ 876 break; 877 } 878 } 879 880 if (pe.portev_source == PORT_SOURCE_USER && 881 pe.portev_events == POKE_AUTO_DISCOVERY) 882 return (poke_is_interesting ? TRUE : FALSE); 883 884 if (pe.portev_source == PORT_SOURCE_FD && pe.portev_object == rt_sock) { 885 /* PF_ROUTE socket read event, re-associate fd, handle event */ 886 if (port_associate(idmapd_ev_port, PORT_SOURCE_FD, rt_sock, 887 POLLIN, NULL) != 0) { 888 idmapdlog(LOG_ERR, "Failed to re-associate the " 889 "routing socket with the event port: %s", 890 strerror(errno)); 891 exit(1); 892 } 893 /* 894 * The network configuration may still be in flux. No matter, 895 * the resolver will re-transmit and timout if need be. 896 */ 897 return (pfroute_event_is_interesting(rt_sock)); 898 } 899 900 if (pe.portev_source == PORT_SOURCE_USER && 901 pe.portev_events == RECONFIGURE) { 902 int rc; 903 904 /* 905 * Blow away the ccache, we might have re-joined the 906 * domain or joined a new one 907 */ 908 (void) unlink(IDMAP_CACHEDIR "/ccache"); 909 /* HUP is the refresh method, so re-read SMF config */ 910 idmapdlog(LOG_INFO, "SMF refresh"); 911 rc = idmap_cfg_load(_idmapdstate.cfg, CFG_DISCOVER|CFG_LOG); 912 if (rc < -1) { 913 idmapdlog(LOG_ERR, "Fatal errors while reading " 914 "SMF properties"); 915 exit(1); 916 } else if (rc == -1) { 917 idmapdlog(LOG_WARNING, "Various errors " 918 "re-loading configuration may cause AD lookups " 919 "to fail"); 920 } 921 return (FALSE); 922 } 923 924 return (FALSE); 925 } 926 927 void * 928 idmap_cfg_update_thread(void *arg) 929 { 930 931 int ttl, changed, poke_is_interesting; 932 idmap_cfg_handles_t *handles = &_idmapdstate.cfg->handles; 933 ad_disc_t ad_ctx = handles->ad_ctx; 934 struct timespec timeout, *timeoutp; 935 936 poke_is_interesting = 1; 937 for (ttl = 0, changed = TRUE; ; ttl = ad_disc_get_TTL(ad_ctx)) { 938 /* 939 * If ttl < 0 then we can wait for an event without timing out. 940 * If idmapd needs to notice that the system has been joined to 941 * a Windows domain then idmapd needs to be refreshed. 942 */ 943 timeoutp = (ttl < 0) ? NULL : &timeout; 944 if (ttl > MAX_CHECK_TIME) 945 ttl = MAX_CHECK_TIME; 946 timeout.tv_sec = ttl; 947 timeout.tv_nsec = 0; 948 changed = wait_for_event(poke_is_interesting, timeoutp); 949 950 /* 951 * If there are no interesting events, and this is not the first 952 * time through the loop, and we haven't waited the most that 953 * we're willing to wait, so do nothing but wait some more. 954 */ 955 if (changed == FALSE && ttl > 0 && ttl < MAX_CHECK_TIME) 956 continue; 957 958 (void) ad_disc_SubnetChanged(ad_ctx); 959 960 if (idmap_cfg_load(_idmapdstate.cfg, CFG_DISCOVER) < -1) { 961 idmapdlog(LOG_ERR, "Fatal errors while reading " 962 "SMF properties"); 963 exit(1); 964 } 965 966 if (_idmapdstate.cfg->pgcfg.global_catalog == NULL || 967 _idmapdstate.cfg->pgcfg.global_catalog[0].host[0] == '\0') 968 poke_is_interesting = 1; 969 else 970 poke_is_interesting = 0; 971 } 972 /*NOTREACHED*/ 973 return (NULL); 974 } 975 976 int 977 idmap_cfg_start_updates(void) 978 { 979 if ((idmapd_ev_port = port_create()) < 0) { 980 idmapdlog(LOG_ERR, "Failed to create event port: %s", 981 strerror(errno)); 982 return (-1); 983 } 984 985 if ((rt_sock = socket(PF_ROUTE, SOCK_RAW, 0)) < 0) { 986 idmapdlog(LOG_ERR, "Failed to open routing socket: %s", 987 strerror(errno)); 988 (void) close(idmapd_ev_port); 989 return (-1); 990 } 991 992 if (fcntl(rt_sock, F_SETFL, O_NDELAY|O_NONBLOCK) < 0) { 993 idmapdlog(LOG_ERR, "Failed to set routing socket flags: %s", 994 strerror(errno)); 995 (void) close(rt_sock); 996 (void) close(idmapd_ev_port); 997 return (-1); 998 } 999 1000 if (port_associate(idmapd_ev_port, PORT_SOURCE_FD, 1001 rt_sock, POLLIN, NULL) != 0) { 1002 idmapdlog(LOG_ERR, "Failed to associate the routing " 1003 "socket with the event port: %s", strerror(errno)); 1004 (void) close(rt_sock); 1005 (void) close(idmapd_ev_port); 1006 return (-1); 1007 } 1008 1009 if ((errno = pthread_create(&update_thread_handle, NULL, 1010 idmap_cfg_update_thread, NULL)) != 0) { 1011 idmapdlog(LOG_ERR, "Failed to start update thread: %s", 1012 strerror(errno)); 1013 (void) port_dissociate(idmapd_ev_port, PORT_SOURCE_FD, rt_sock); 1014 (void) close(rt_sock); 1015 (void) close(idmapd_ev_port); 1016 return (-1); 1017 } 1018 1019 return (0); 1020 } 1021 1022 /* 1023 * Reject attribute names with invalid characters. 1024 */ 1025 static 1026 int 1027 valid_ldap_attr(const char *attr) { 1028 for (; *attr; attr++) { 1029 if (!isalnum(*attr) && *attr != '-' && 1030 *attr != '_' && *attr != '.' && *attr != ';') 1031 return (0); 1032 } 1033 return (1); 1034 } 1035 1036 /* 1037 * This is the half of idmap_cfg_load() that loads property values from 1038 * SMF (using the config/ property group of the idmap FMRI). 1039 * 1040 * Return values: 0 -> success, -1 -> failure, -2 -> hard failures 1041 * -3 -> hard smf config failures 1042 * reading from SMF. 1043 */ 1044 static 1045 int 1046 idmap_cfg_load_smf(idmap_cfg_handles_t *handles, idmap_pg_config_t *pgcfg, 1047 int *errors) 1048 { 1049 int rc; 1050 uint8_t bool_val; 1051 char *str = NULL; 1052 boolean_t new_debug_mode; 1053 1054 if (scf_pg_update(handles->config_pg) < 0) { 1055 idmapdlog(LOG_ERR, "scf_pg_update() failed: %s", 1056 scf_strerror(scf_error())); 1057 return (-2); 1058 } 1059 1060 if (scf_pg_update(handles->general_pg) < 0) { 1061 idmapdlog(LOG_ERR, "scf_pg_update() failed: %s", 1062 scf_strerror(scf_error())); 1063 return (-2); 1064 } 1065 1066 1067 rc = prop_exists(handles, "debug", &new_debug_mode); 1068 if (rc != 0) 1069 errors++; 1070 1071 if (_idmapdstate.debug_mode != new_debug_mode) { 1072 if (!_idmapdstate.debug_mode) { 1073 _idmapdstate.debug_mode = new_debug_mode; 1074 idmap_log_stderr(LOG_DEBUG); 1075 idmapdlog(LOG_DEBUG, "debug mode enabled"); 1076 } else { 1077 idmapdlog(LOG_DEBUG, "debug mode disabled"); 1078 idmap_log_stderr(-1); 1079 _idmapdstate.debug_mode = new_debug_mode; 1080 } 1081 } 1082 1083 rc = get_val_int(handles, "unresolvable_sid_mapping", 1084 &pgcfg->eph_map_unres_sids, SCF_TYPE_BOOLEAN); 1085 if (rc != 0) 1086 errors++; 1087 1088 rc = get_val_int(handles, "list_size_limit", 1089 &pgcfg->list_size_limit, SCF_TYPE_COUNT); 1090 if (rc != 0) 1091 errors++; 1092 1093 rc = get_val_astring(handles, "domain_name", 1094 &pgcfg->domain_name); 1095 if (rc != 0) 1096 errors++; 1097 else { 1098 (void) ad_disc_set_DomainName(handles->ad_ctx, 1099 pgcfg->domain_name); 1100 pgcfg->domain_name_auto_disc = B_FALSE; 1101 } 1102 1103 rc = get_val_astring(handles, "default_domain", 1104 &pgcfg->default_domain); 1105 if (rc != 0) { 1106 /* 1107 * SCF failures fetching config/default_domain we treat 1108 * as fatal as they may leave ID mapping rules that 1109 * match unqualified winnames flapping in the wind. 1110 */ 1111 return (-2); 1112 } 1113 1114 rc = get_val_astring(handles, "mapping_domain", &str); 1115 if (rc != 0) 1116 errors++; 1117 1118 /* 1119 * We treat default_domain as having been specified in SMF IFF 1120 * either (the config/default_domain property was set) or (the 1121 * old, obsolete, never documented config/mapping_domain 1122 * property was set and the new config/domain_name property was 1123 * not set). 1124 */ 1125 pgcfg->dflt_dom_set_in_smf = B_TRUE; 1126 if (pgcfg->default_domain == NULL) { 1127 1128 pgcfg->dflt_dom_set_in_smf = B_FALSE; 1129 1130 if (pgcfg->domain_name != NULL) { 1131 pgcfg->default_domain = strdup(pgcfg->domain_name); 1132 if (str != NULL) { 1133 idmapdlog(LOG_WARNING, 1134 "Ignoring obsolete, undocumented " 1135 "config/mapping_domain property"); 1136 } 1137 } else if (str != NULL) { 1138 pgcfg->default_domain = strdup(str); 1139 pgcfg->dflt_dom_set_in_smf = B_TRUE; 1140 idmapdlog(LOG_WARNING, 1141 "The config/mapping_domain property is " 1142 "obsolete; support for it will be removed, " 1143 "please use config/default_domain instead"); 1144 } 1145 } 1146 1147 if (str != NULL) 1148 free(str); 1149 1150 rc = get_val_astring(handles, "machine_sid", &pgcfg->machine_sid); 1151 if (rc != 0) 1152 errors++; 1153 if (pgcfg->machine_sid == NULL) { 1154 /* If machine_sid not configured, generate one */ 1155 if (generate_machine_sid(&pgcfg->machine_sid) < 0) 1156 return (-2); 1157 rc = set_val_astring(handles, "machine_sid", 1158 pgcfg->machine_sid); 1159 if (rc != 0) 1160 errors++; 1161 } 1162 1163 str = NULL; 1164 rc = get_val_ds(handles, "domain_controller", 389, 1165 &pgcfg->domain_controller); 1166 if (rc != 0) 1167 errors++; 1168 else { 1169 (void) ad_disc_set_DomainController(handles->ad_ctx, 1170 pgcfg->domain_controller); 1171 pgcfg->domain_controller_auto_disc = B_FALSE; 1172 } 1173 1174 rc = get_val_astring(handles, "forest_name", &pgcfg->forest_name); 1175 if (rc != 0) 1176 errors++; 1177 else { 1178 (void) ad_disc_set_ForestName(handles->ad_ctx, 1179 pgcfg->forest_name); 1180 pgcfg->forest_name_auto_disc = B_FALSE; 1181 } 1182 1183 rc = get_val_astring(handles, "site_name", &pgcfg->site_name); 1184 if (rc != 0) 1185 errors++; 1186 else 1187 (void) ad_disc_set_SiteName(handles->ad_ctx, pgcfg->site_name); 1188 1189 str = NULL; 1190 rc = get_val_ds(handles, "global_catalog", 3268, 1191 &pgcfg->global_catalog); 1192 if (rc != 0) 1193 errors++; 1194 else { 1195 (void) ad_disc_set_GlobalCatalog(handles->ad_ctx, 1196 pgcfg->global_catalog); 1197 pgcfg->global_catalog_auto_disc = B_FALSE; 1198 } 1199 1200 /* 1201 * Read directory-based name mappings related SMF properties 1202 */ 1203 rc = get_val_int(handles, "ds_name_mapping_enabled", 1204 &bool_val, SCF_TYPE_BOOLEAN); 1205 if (rc != 0) 1206 return (-2); 1207 1208 if (!bool_val) 1209 return (rc); 1210 1211 pgcfg->ds_name_mapping_enabled = B_TRUE; 1212 rc = get_val_astring(handles, "ad_unixuser_attr", 1213 &pgcfg->ad_unixuser_attr); 1214 if (rc != 0) 1215 return (-2); 1216 if (pgcfg->ad_unixuser_attr != NULL && 1217 !valid_ldap_attr(pgcfg->ad_unixuser_attr)) { 1218 idmapdlog(LOG_ERR, "config/ad_unixuser_attr=%s is not a " 1219 "valid LDAP attribute name", pgcfg->ad_unixuser_attr); 1220 return (-3); 1221 } 1222 1223 rc = get_val_astring(handles, "ad_unixgroup_attr", 1224 &pgcfg->ad_unixgroup_attr); 1225 if (rc != 0) 1226 return (-2); 1227 if (pgcfg->ad_unixgroup_attr != NULL && 1228 !valid_ldap_attr(pgcfg->ad_unixgroup_attr)) { 1229 idmapdlog(LOG_ERR, "config/ad_unixgroup_attr=%s is not a " 1230 "valid LDAP attribute name", pgcfg->ad_unixgroup_attr); 1231 return (-3); 1232 } 1233 1234 rc = get_val_astring(handles, "nldap_winname_attr", 1235 &pgcfg->nldap_winname_attr); 1236 if (rc != 0) 1237 return (-2); 1238 if (pgcfg->nldap_winname_attr != NULL && 1239 !valid_ldap_attr(pgcfg->nldap_winname_attr)) { 1240 idmapdlog(LOG_ERR, "config/nldap_winname_attr=%s is not a " 1241 "valid LDAP attribute name", pgcfg->nldap_winname_attr); 1242 return (-3); 1243 } 1244 if (pgcfg->ad_unixuser_attr == NULL && 1245 pgcfg->ad_unixgroup_attr == NULL && 1246 pgcfg->nldap_winname_attr == NULL) { 1247 idmapdlog(LOG_ERR, 1248 "If config/ds_name_mapping_enabled property is set to " 1249 "true then atleast one of the following name mapping " 1250 "attributes must be specified. (config/ad_unixuser_attr OR " 1251 "config/ad_unixgroup_attr OR config/nldap_winname_attr)"); 1252 return (-3); 1253 } 1254 1255 return (rc); 1256 1257 } 1258 1259 1260 /* 1261 * This is the half of idmap_cfg_load() that auto-discovers values of 1262 * discoverable properties that weren't already set via SMF properties. 1263 * 1264 * idmap_cfg_discover() is called *after* idmap_cfg_load_smf(), so it 1265 * needs to be careful not to overwrite any properties set in SMF. 1266 */ 1267 static 1268 void 1269 idmap_cfg_discover(idmap_cfg_handles_t *handles, idmap_pg_config_t *pgcfg) 1270 { 1271 ad_disc_t ad_ctx = handles->ad_ctx; 1272 ad_disc_t trusted_ctx; 1273 int i, j, k, l; 1274 char *forestname; 1275 int num_trusteddomains; 1276 boolean_t new_forest; 1277 int err; 1278 char *trusteddomain; 1279 idmap_ad_disc_ds_t *globalcatalog; 1280 idmap_trustedforest_t *trustedforests; 1281 ad_disc_domainsinforest_t *domainsinforest; 1282 1283 ad_disc_refresh(ad_ctx); 1284 1285 if (pgcfg->default_domain == NULL) 1286 pgcfg->default_domain = ad_disc_get_DomainName(ad_ctx, 1287 NULL); 1288 1289 if (pgcfg->domain_name == NULL) 1290 pgcfg->domain_name = ad_disc_get_DomainName(ad_ctx, 1291 &pgcfg->domain_name_auto_disc); 1292 1293 if (pgcfg->domain_controller == NULL) 1294 pgcfg->domain_controller = 1295 ad_disc_get_DomainController(ad_ctx, AD_DISC_PREFER_SITE, 1296 &pgcfg->domain_controller_auto_disc); 1297 1298 if (pgcfg->forest_name == NULL) 1299 pgcfg->forest_name = ad_disc_get_ForestName(ad_ctx, 1300 &pgcfg->forest_name_auto_disc); 1301 1302 if (pgcfg->site_name == NULL) 1303 pgcfg->site_name = ad_disc_get_SiteName(ad_ctx, 1304 &pgcfg->site_name_auto_disc); 1305 1306 if (pgcfg->global_catalog == NULL) 1307 pgcfg->global_catalog = 1308 ad_disc_get_GlobalCatalog(ad_ctx, AD_DISC_PREFER_SITE, 1309 &pgcfg->global_catalog_auto_disc); 1310 1311 pgcfg->domains_in_forest = 1312 ad_disc_get_DomainsInForest(ad_ctx, NULL); 1313 1314 pgcfg->trusted_domains = 1315 ad_disc_get_TrustedDomains(ad_ctx, NULL); 1316 1317 if (pgcfg->forest_name != NULL && pgcfg->trusted_domains != NULL && 1318 pgcfg->trusted_domains[0].domain[0] != '\0') { 1319 /* 1320 * We have trusted domains. We need to go through every 1321 * one and find its forest. If it is a new forest we then need 1322 * to find its Global Catalog and the domains in the forest 1323 */ 1324 for (i = 0; pgcfg->trusted_domains[i].domain[0] != '\0'; i++) 1325 continue; 1326 num_trusteddomains = i; 1327 1328 trustedforests = calloc(num_trusteddomains, 1329 sizeof (idmap_trustedforest_t)); 1330 j = 0; 1331 for (i = 0; pgcfg->trusted_domains[i].domain[0] != '\0'; i++) { 1332 trusteddomain = pgcfg->trusted_domains[i].domain; 1333 trusted_ctx = ad_disc_init(); 1334 ad_disc_set_DomainName(trusted_ctx, 1335 trusteddomain); 1336 forestname = 1337 ad_disc_get_ForestName(trusted_ctx, NULL); 1338 if (forestname == NULL) { 1339 idmapdlog(LOG_DEBUG, "unable to discover " 1340 "Forest Name for the trusted domain %s", 1341 trusteddomain); 1342 ad_disc_fini(trusted_ctx); 1343 continue; 1344 } 1345 1346 if (strcasecmp(forestname, pgcfg->forest_name) == 0) { 1347 /* 1348 * Ignore the domain as it is part of 1349 * the primary forest 1350 */ 1351 free(forestname); 1352 ad_disc_fini(trusted_ctx); 1353 continue; 1354 } 1355 1356 /* Is this a new forest? */ 1357 new_forest = B_TRUE; 1358 for (k = 0; k < j; k++) { 1359 if (strcasecmp(forestname, 1360 trustedforests[k].forest_name) == 0) { 1361 new_forest = B_FALSE; 1362 domainsinforest = 1363 trustedforests[k].domains_in_forest; 1364 break; 1365 } 1366 } 1367 if (!new_forest) { 1368 /* Mark the domain as trusted */ 1369 for (l = 0; 1370 domainsinforest[l].domain[0] != '\0'; l++) { 1371 if (u8_strcmp(trusteddomain, 1372 domainsinforest[l].domain, 0, 1373 U8_STRCMP_CI_LOWER, 1374 U8_UNICODE_LATEST, &err) == 0 && 1375 err == 0) { 1376 domainsinforest[l].trusted = 1377 TRUE; 1378 break; 1379 } 1380 } 1381 free(forestname); 1382 ad_disc_fini(trusted_ctx); 1383 continue; 1384 } 1385 1386 /* 1387 * Get the Global Catalog and the domains in 1388 * this new forest. 1389 */ 1390 globalcatalog = 1391 ad_disc_get_GlobalCatalog(trusted_ctx, 1392 AD_DISC_PREFER_SITE, NULL); 1393 if (globalcatalog == NULL) { 1394 idmapdlog(LOG_DEBUG, 1395 "unable to discover Global " 1396 "Catalog for the trusted domain %s", 1397 trusteddomain); 1398 free(forestname); 1399 ad_disc_fini(trusted_ctx); 1400 continue; 1401 } 1402 domainsinforest = 1403 ad_disc_get_DomainsInForest(trusted_ctx, 1404 NULL); 1405 if (domainsinforest == NULL) { 1406 idmapdlog(LOG_DEBUG, 1407 "unable to discover Domains in the Forest " 1408 "for the trusted domain %s", 1409 trusteddomain); 1410 free(globalcatalog); 1411 free(forestname); 1412 ad_disc_fini(trusted_ctx); 1413 continue; 1414 } 1415 1416 trustedforests[j].forest_name = forestname; 1417 trustedforests[j].global_catalog = globalcatalog; 1418 trustedforests[j].domains_in_forest = domainsinforest; 1419 j++; 1420 /* Mark the domain as trusted */ 1421 for (l = 0; domainsinforest[l].domain[0] != '\0'; 1422 l++) { 1423 if (u8_strcmp(trusteddomain, 1424 domainsinforest[l].domain, 0, 1425 U8_STRCMP_CI_LOWER, 1426 U8_UNICODE_LATEST, &err) == 0 && 1427 err == 0) { 1428 domainsinforest[l].trusted = TRUE; 1429 break; 1430 } 1431 } 1432 ad_disc_fini(trusted_ctx); 1433 } 1434 if (j > 0) { 1435 pgcfg->num_trusted_forests = j; 1436 pgcfg->trusted_forests = trustedforests; 1437 } 1438 } 1439 1440 if (pgcfg->domain_name == NULL) 1441 idmapdlog(LOG_DEBUG, "unable to discover Domain Name"); 1442 if (pgcfg->domain_controller == NULL) 1443 idmapdlog(LOG_DEBUG, "unable to discover Domain Controller"); 1444 if (pgcfg->forest_name == NULL) 1445 idmapdlog(LOG_DEBUG, "unable to discover Forest Name"); 1446 if (pgcfg->site_name == NULL) 1447 idmapdlog(LOG_DEBUG, "unable to discover Site Name"); 1448 if (pgcfg->global_catalog == NULL) 1449 idmapdlog(LOG_DEBUG, "unable to discover Global Catalog"); 1450 if (pgcfg->domains_in_forest == NULL) 1451 idmapdlog(LOG_DEBUG, 1452 "unable to discover Domains in the Forest"); 1453 if (pgcfg->trusted_domains == NULL) 1454 idmapdlog(LOG_DEBUG, "unable to discover Trusted Domains"); 1455 } 1456 1457 1458 /* 1459 * idmap_cfg_load() is called at startup, and periodically via the 1460 * update thread when the auto-discovery TTLs expire, as well as part of 1461 * the refresh method, to update the current configuration. It always 1462 * reads from SMF, but you still have to refresh the service after 1463 * changing the config pg in order for the changes to take effect. 1464 * 1465 * There are two flags: 1466 * 1467 * - CFG_DISCOVER 1468 * - CFG_LOG 1469 * 1470 * If CFG_DISCOVER is set then idmap_cfg_load() calls 1471 * idmap_cfg_discover() to discover, via DNS and LDAP lookups, property 1472 * values that weren't set in SMF. 1473 * 1474 * If CFG_LOG is set then idmap_cfg_load() will log (to LOG_NOTICE) 1475 * whether the configuration changed. This should be used only from the 1476 * refresh method. 1477 * 1478 * Return values: 0 -> success, -1 -> failure, -2 -> hard failures 1479 * reading from SMF. 1480 */ 1481 int 1482 idmap_cfg_load(idmap_cfg_t *cfg, int flags) 1483 { 1484 int rc = 0; 1485 int errors = 0; 1486 int changed = 0; 1487 int ad_reload_required = 0; 1488 idmap_pg_config_t new_pgcfg, *live_pgcfg; 1489 1490 live_pgcfg = &cfg->pgcfg; 1491 (void) memset(&new_pgcfg, 0, sizeof (new_pgcfg)); 1492 1493 pthread_mutex_lock(&cfg->handles.mutex); 1494 1495 if ((rc = idmap_cfg_load_smf(&cfg->handles, &new_pgcfg, &errors)) < -1) 1496 goto err; 1497 1498 if (flags & CFG_DISCOVER) 1499 idmap_cfg_discover(&cfg->handles, &new_pgcfg); 1500 1501 WRLOCK_CONFIG(); 1502 if (live_pgcfg->list_size_limit != new_pgcfg.list_size_limit) { 1503 idmapdlog(LOG_INFO, "change list_size=%d", 1504 new_pgcfg.list_size_limit); 1505 live_pgcfg->list_size_limit = new_pgcfg.list_size_limit; 1506 } 1507 1508 /* Non-discoverable props updated here */ 1509 changed += update_string(&live_pgcfg->machine_sid, 1510 &new_pgcfg.machine_sid, "machine_sid"); 1511 1512 changed += update_bool(&live_pgcfg->eph_map_unres_sids, 1513 &new_pgcfg.eph_map_unres_sids, "unresolvable_sid_mapping"); 1514 1515 changed += live_pgcfg->ds_name_mapping_enabled != 1516 new_pgcfg.ds_name_mapping_enabled; 1517 live_pgcfg->ds_name_mapping_enabled = 1518 new_pgcfg.ds_name_mapping_enabled; 1519 1520 changed += update_string(&live_pgcfg->ad_unixuser_attr, 1521 &new_pgcfg.ad_unixuser_attr, "ad_unixuser_attr"); 1522 1523 changed += update_string(&live_pgcfg->ad_unixgroup_attr, 1524 &new_pgcfg.ad_unixgroup_attr, "ad_unixgroup_attr"); 1525 1526 changed += update_string(&live_pgcfg->nldap_winname_attr, 1527 &new_pgcfg.nldap_winname_attr, "nldap_winname_attr"); 1528 1529 /* Props that can be discovered and set in SMF updated here */ 1530 if (!live_pgcfg->dflt_dom_set_in_smf) 1531 changed += update_string(&live_pgcfg->default_domain, 1532 &new_pgcfg.default_domain, "default_domain"); 1533 1534 changed += update_string(&live_pgcfg->domain_name, 1535 &new_pgcfg.domain_name, "domain_name"); 1536 live_pgcfg->domain_name_auto_disc = new_pgcfg.domain_name_auto_disc; 1537 1538 changed += update_dirs(&live_pgcfg->domain_controller, 1539 &new_pgcfg.domain_controller, "domain_controller"); 1540 live_pgcfg->domain_controller_auto_disc = 1541 new_pgcfg.domain_controller_auto_disc; 1542 1543 changed += update_string(&live_pgcfg->forest_name, 1544 &new_pgcfg.forest_name, "forest_name"); 1545 live_pgcfg->forest_name_auto_disc = new_pgcfg.forest_name_auto_disc; 1546 1547 changed += update_string(&live_pgcfg->site_name, 1548 &new_pgcfg.site_name, "site_name"); 1549 live_pgcfg->site_name_auto_disc = new_pgcfg.site_name_auto_disc; 1550 1551 if (update_dirs(&live_pgcfg->global_catalog, 1552 &new_pgcfg.global_catalog, "global_catalog")) { 1553 changed++; 1554 if (live_pgcfg->global_catalog != NULL && 1555 live_pgcfg->global_catalog[0].host[0] != '\0') 1556 ad_reload_required = TRUE; 1557 } 1558 live_pgcfg->global_catalog_auto_disc = 1559 new_pgcfg.global_catalog_auto_disc; 1560 1561 if (update_domains_in_forest(&live_pgcfg->domains_in_forest, 1562 &new_pgcfg.domains_in_forest, "domains_in_forest")) { 1563 changed++; 1564 ad_reload_required = TRUE; 1565 } 1566 1567 if (update_trusted_domains(&live_pgcfg->trusted_domains, 1568 &new_pgcfg.trusted_domains, "trusted_domains")) { 1569 changed++; 1570 if (live_pgcfg->trusted_domains != NULL && 1571 live_pgcfg->trusted_domains[0].domain[0] != '\0') 1572 ad_reload_required = TRUE; 1573 } 1574 1575 if (update_trusted_forest(&live_pgcfg->trusted_forests, 1576 &live_pgcfg->num_trusted_forests, &new_pgcfg.trusted_forests, 1577 &new_pgcfg.num_trusted_forests, "trusted_forest")) { 1578 changed++; 1579 if (live_pgcfg->trusted_forests != NULL) 1580 ad_reload_required = TRUE; 1581 } 1582 1583 if (ad_reload_required) 1584 reload_ad(); 1585 1586 idmap_cfg_unload(&new_pgcfg); 1587 1588 if (flags & CFG_LOG) { 1589 /* 1590 * If the config changes as a result of a refresh of the 1591 * service, then logging about it can provide useful 1592 * feedback to the sysadmin. 1593 */ 1594 idmapdlog(LOG_NOTICE, "Configuration %schanged", 1595 changed ? "" : "un"); 1596 } 1597 1598 UNLOCK_CONFIG(); 1599 1600 err: 1601 pthread_mutex_unlock(&cfg->handles.mutex); 1602 1603 if (rc < -1) 1604 return (rc); 1605 1606 return ((errors == 0) ? 0 : -1); 1607 } 1608 1609 /* 1610 * Initialize 'cfg'. 1611 */ 1612 idmap_cfg_t * 1613 idmap_cfg_init() 1614 { 1615 idmap_cfg_handles_t *handles; 1616 1617 /* First the smf repository handles: */ 1618 idmap_cfg_t *cfg = calloc(1, sizeof (idmap_cfg_t)); 1619 if (!cfg) { 1620 idmapdlog(LOG_ERR, "Out of memory"); 1621 return (NULL); 1622 } 1623 handles = &cfg->handles; 1624 1625 (void) pthread_mutex_init(&handles->mutex, NULL); 1626 1627 if (!(handles->main = scf_handle_create(SCF_VERSION))) { 1628 idmapdlog(LOG_ERR, "scf_handle_create() failed: %s", 1629 scf_strerror(scf_error())); 1630 goto error; 1631 } 1632 1633 if (scf_handle_bind(handles->main) < 0) { 1634 idmapdlog(LOG_ERR, "scf_handle_bind() failed: %s", 1635 scf_strerror(scf_error())); 1636 goto error; 1637 } 1638 1639 if (!(handles->service = scf_service_create(handles->main)) || 1640 !(handles->instance = scf_instance_create(handles->main)) || 1641 !(handles->config_pg = scf_pg_create(handles->main)) || 1642 !(handles->general_pg = scf_pg_create(handles->main))) { 1643 idmapdlog(LOG_ERR, "scf handle creation failed: %s", 1644 scf_strerror(scf_error())); 1645 goto error; 1646 } 1647 1648 if (scf_handle_decode_fmri(handles->main, 1649 FMRI_BASE "/:properties/" CONFIG_PG, 1650 NULL, /* scope */ 1651 handles->service, /* service */ 1652 handles->instance, /* instance */ 1653 handles->config_pg, /* pg */ 1654 NULL, /* prop */ 1655 SCF_DECODE_FMRI_EXACT) < 0) { 1656 idmapdlog(LOG_ERR, "scf_handle_decode_fmri() failed: %s", 1657 scf_strerror(scf_error())); 1658 goto error; 1659 } 1660 1661 if (scf_service_get_pg(handles->service, 1662 GENERAL_PG, handles->general_pg) < 0) { 1663 idmapdlog(LOG_ERR, "scf_service_get_pg() failed: %s", 1664 scf_strerror(scf_error())); 1665 goto error; 1666 } 1667 1668 /* Initialize AD Auto Discovery context */ 1669 handles->ad_ctx = ad_disc_init(); 1670 if (handles->ad_ctx == NULL) 1671 goto error; 1672 1673 return (cfg); 1674 1675 error: 1676 (void) idmap_cfg_fini(cfg); 1677 return (NULL); 1678 } 1679 1680 void 1681 idmap_cfg_unload(idmap_pg_config_t *pgcfg) 1682 { 1683 1684 if (pgcfg->default_domain) { 1685 free(pgcfg->default_domain); 1686 pgcfg->default_domain = NULL; 1687 } 1688 if (pgcfg->domain_name) { 1689 free(pgcfg->domain_name); 1690 pgcfg->domain_name = NULL; 1691 } 1692 if (pgcfg->machine_sid) { 1693 free(pgcfg->machine_sid); 1694 pgcfg->machine_sid = NULL; 1695 } 1696 if (pgcfg->domain_controller) { 1697 free(pgcfg->domain_controller); 1698 pgcfg->domain_controller = NULL; 1699 } 1700 if (pgcfg->forest_name) { 1701 free(pgcfg->forest_name); 1702 pgcfg->forest_name = NULL; 1703 } 1704 if (pgcfg->site_name) { 1705 free(pgcfg->site_name); 1706 pgcfg->site_name = NULL; 1707 } 1708 if (pgcfg->global_catalog) { 1709 free(pgcfg->global_catalog); 1710 pgcfg->global_catalog = NULL; 1711 } 1712 if (pgcfg->trusted_domains) { 1713 free(pgcfg->trusted_domains); 1714 pgcfg->trusted_domains = NULL; 1715 } 1716 if (pgcfg->trusted_forests) 1717 free_trusted_forests(&pgcfg->trusted_forests, 1718 &pgcfg->num_trusted_forests); 1719 1720 if (pgcfg->ad_unixuser_attr) { 1721 free(pgcfg->ad_unixuser_attr); 1722 pgcfg->ad_unixuser_attr = NULL; 1723 } 1724 if (pgcfg->ad_unixgroup_attr) { 1725 free(pgcfg->ad_unixgroup_attr); 1726 pgcfg->ad_unixgroup_attr = NULL; 1727 } 1728 if (pgcfg->nldap_winname_attr) { 1729 free(pgcfg->nldap_winname_attr); 1730 pgcfg->nldap_winname_attr = NULL; 1731 } 1732 } 1733 1734 int 1735 idmap_cfg_fini(idmap_cfg_t *cfg) 1736 { 1737 idmap_cfg_handles_t *handles = &cfg->handles; 1738 idmap_cfg_unload(&cfg->pgcfg); 1739 1740 (void) pthread_mutex_destroy(&handles->mutex); 1741 scf_pg_destroy(handles->config_pg); 1742 scf_pg_destroy(handles->general_pg); 1743 scf_instance_destroy(handles->instance); 1744 scf_service_destroy(handles->service); 1745 scf_handle_destroy(handles->main); 1746 if (handles->ad_ctx != NULL) 1747 ad_disc_fini(handles->ad_ctx); 1748 free(cfg); 1749 1750 return (0); 1751 } 1752 1753 void 1754 idmap_cfg_poke_updates(void) 1755 { 1756 if (idmapd_ev_port != -1) 1757 (void) port_send(idmapd_ev_port, POKE_AUTO_DISCOVERY, NULL); 1758 } 1759 1760 /*ARGSUSED*/ 1761 void 1762 idmap_cfg_hup_handler(int sig) 1763 { 1764 if (idmapd_ev_port >= 0) 1765 (void) port_send(idmapd_ev_port, RECONFIGURE, NULL); 1766 } 1767