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 /* 23 * Copyright (c) 2009, 2010, Oracle and/or its affiliates. All rights reserved. 24 * Copyright 2015 Nexenta Systems, Inc. All rights reserved. 25 */ 26 27 /* 28 * Local Security Authority RPC (LSAR) client-side interface. 29 */ 30 31 #include <sys/errno.h> 32 #include <stdio.h> 33 #include <stdlib.h> 34 #include <strings.h> 35 36 #include <smbsrv/libsmb.h> 37 #include <smbsrv/libmlsvc.h> 38 #include <smbsrv/smbinfo.h> 39 #include <smbsrv/ntaccess.h> 40 #include <smbsrv/ntlocale.h> 41 #include <smbsrv/string.h> 42 #include <lsalib.h> 43 44 /* 45 * The maximum number of bytes we are prepared to deal with in a 46 * response. 47 */ 48 #define MLSVC_MAX_RESPONSE_LEN 1024 49 50 /* 51 * This structure is used when looking up names. We only lookup one 52 * name at a time but the structure will allow for more. 53 */ 54 typedef struct lsa_names { 55 uint32_t n_entry; 56 mslsa_string_t name[8]; 57 } lsa_names_t; 58 59 typedef DWORD (*lsar_nameop_t)(mlsvc_handle_t *, lsa_names_t *, 60 smb_account_t *); 61 62 static uint32_t lsar_lookup_names1(mlsvc_handle_t *, lsa_names_t *, 63 smb_account_t *); 64 static uint32_t lsar_lookup_names2(mlsvc_handle_t *, lsa_names_t *, 65 smb_account_t *); 66 static uint32_t lsar_lookup_names3(mlsvc_handle_t *, lsa_names_t *, 67 smb_account_t *); 68 static uint32_t lsar_lookup_sids1(mlsvc_handle_t *, lsa_sid_t *, 69 smb_account_t *); 70 static uint32_t lsar_lookup_sids2(mlsvc_handle_t *, lsa_sid_t *, 71 smb_account_t *account); 72 73 static char *lsar_get_username(const char *); 74 static void smb_account_trace(const smb_account_t *); 75 76 static void lsar_set_trusted_domains_ex(struct mslsa_EnumTrustedDomainBufEx *, 77 smb_trusted_domains_t *); 78 static void lsar_set_trusted_domains(struct mslsa_EnumTrustedDomainBuf *, 79 smb_trusted_domains_t *); 80 81 /* 82 * lsar_open 83 * 84 * This is a wrapper round lsar_open_policy2 to ensure that we connect 85 * using the appropriate domain information. 86 * 87 * If username argument is NULL, an anonymous connection will be established. 88 * Otherwise, an authenticated connection will be established. 89 * 90 * Returns 0 or NT status (Raw, not LSA-ized) 91 */ 92 DWORD 93 lsar_open(char *server, char *domain, char *username, 94 mlsvc_handle_t *domain_handle) 95 { 96 DWORD status; 97 98 if (server == NULL || domain == NULL) 99 return (NT_STATUS_INTERNAL_ERROR); 100 101 if (username == NULL) 102 username = MLSVC_ANON_USER; 103 104 status = lsar_open_policy2(server, domain, username, domain_handle); 105 106 return (status); 107 } 108 109 /* 110 * lsar_open_policy2 111 * 112 * Obtain an LSA policy handle. A policy handle is required to access 113 * LSA resources on a remote server. The server name supplied here does 114 * not need the double backslash prefix; it is added here. Call this 115 * function via lsar_open to ensure that the appropriate connection is 116 * in place. 117 * 118 * Returns 0 or NT status (Raw, not LSA-ized) 119 */ 120 DWORD 121 lsar_open_policy2(char *server, char *domain, char *user, 122 mlsvc_handle_t *lsa_handle) 123 { 124 struct mslsa_OpenPolicy2 arg; 125 DWORD status; 126 int opnum; 127 int len; 128 129 status = ndr_rpc_bind(lsa_handle, server, domain, user, "LSARPC"); 130 if (status != 0) 131 return (status); 132 133 opnum = LSARPC_OPNUM_OpenPolicy2; 134 bzero(&arg, sizeof (struct mslsa_OpenPolicy2)); 135 136 len = strlen(server) + 4; 137 arg.servername = ndr_rpc_malloc(lsa_handle, len); 138 if (arg.servername == NULL) { 139 status = NT_STATUS_NO_MEMORY; 140 goto out; 141 } 142 143 (void) snprintf((char *)arg.servername, len, "\\\\%s", server); 144 arg.attributes.length = sizeof (struct mslsa_object_attributes); 145 arg.desiredAccess = MAXIMUM_ALLOWED; 146 147 if (ndr_rpc_call(lsa_handle, opnum, &arg) != 0) { 148 status = RPC_NT_CALL_FAILED; 149 goto out; 150 } 151 status = arg.status; 152 if (status == NT_STATUS_SUCCESS) { 153 (void) memcpy(&lsa_handle->handle, &arg.domain_handle, 154 sizeof (ndr_hdid_t)); 155 156 if (ndr_is_null_handle(lsa_handle)) 157 status = NT_STATUS_INVALID_PARAMETER; 158 } 159 160 ndr_rpc_release(lsa_handle); 161 162 out: 163 if (status != NT_STATUS_SUCCESS) 164 ndr_rpc_unbind(lsa_handle); 165 return (status); 166 } 167 168 /* 169 * lsar_open_account 170 * 171 * Obtain an LSA account handle. The lsa_handle must be a valid handle 172 * obtained via lsar_open_policy2. The main thing to remember here is 173 * to set up the context in the lsa_account_handle. I'm not sure what 174 * the requirements are for desired access. Some values require admin 175 * access. 176 * 177 * Returns 0 on success. Otherwise non-zero to indicate a failure. 178 */ 179 int 180 lsar_open_account(mlsvc_handle_t *lsa_handle, struct mslsa_sid *sid, 181 mlsvc_handle_t *lsa_account_handle) 182 { 183 struct mslsa_OpenAccount arg; 184 int opnum; 185 int rc; 186 187 if (ndr_is_null_handle(lsa_handle) || sid == NULL) 188 return (-1); 189 190 opnum = LSARPC_OPNUM_OpenAccount; 191 bzero(&arg, sizeof (struct mslsa_OpenAccount)); 192 193 (void) memcpy(&arg.handle, lsa_handle, sizeof (mslsa_handle_t)); 194 arg.sid = sid; 195 arg.access_mask = STANDARD_RIGHTS_REQUIRED 196 #if 0 197 | POLICY_VIEW_AUDIT_INFORMATION 198 | POLICY_GET_PRIVATE_INFORMATION 199 | POLICY_TRUST_ADMIN 200 #endif 201 | POLICY_VIEW_LOCAL_INFORMATION; 202 203 if ((rc = ndr_rpc_call(lsa_handle, opnum, &arg)) != 0) 204 return (-1); 205 206 if (arg.status != 0) { 207 rc = -1; 208 } else { 209 ndr_inherit_handle(lsa_account_handle, lsa_handle); 210 211 (void) memcpy(&lsa_account_handle->handle, 212 &arg.account_handle, sizeof (ndr_hdid_t)); 213 214 if (ndr_is_null_handle(lsa_account_handle)) 215 rc = -1; 216 } 217 218 ndr_rpc_release(lsa_handle); 219 return (rc); 220 } 221 222 /* 223 * lsar_close 224 * 225 * Close the LSA connection associated with the handle. The lsa_handle 226 * must be a valid handle obtained via a call to lsar_open_policy2 or 227 * lsar_open_account. On success the handle will be zeroed out to 228 * ensure that it is not used again. If this is the top level handle 229 * (i.e. the one obtained via lsar_open_policy2) the pipe is closed. 230 * 231 * Returns 0 on success. Otherwise non-zero to indicate a failure. 232 */ 233 int 234 lsar_close(mlsvc_handle_t *lsa_handle) 235 { 236 struct mslsa_CloseHandle arg; 237 int opnum; 238 239 if (ndr_is_null_handle(lsa_handle)) 240 return (-1); 241 242 opnum = LSARPC_OPNUM_CloseHandle; 243 bzero(&arg, sizeof (struct mslsa_CloseHandle)); 244 (void) memcpy(&arg.handle, lsa_handle, sizeof (mslsa_handle_t)); 245 246 (void) ndr_rpc_call(lsa_handle, opnum, &arg); 247 ndr_rpc_release(lsa_handle); 248 249 if (ndr_is_bind_handle(lsa_handle)) 250 ndr_rpc_unbind(lsa_handle); 251 252 bzero(lsa_handle, sizeof (mlsvc_handle_t)); 253 return (0); 254 } 255 256 /* 257 * lsar_query_security_desc 258 * 259 * Don't use this call yet. It is just a place holder for now. 260 */ 261 int 262 lsar_query_security_desc(mlsvc_handle_t *lsa_handle) 263 { 264 struct mslsa_QuerySecurityObject arg; 265 int rc; 266 int opnum; 267 268 opnum = LSARPC_OPNUM_QuerySecurityObject; 269 270 bzero(&arg, sizeof (struct mslsa_QuerySecurityObject)); 271 (void) memcpy(&arg.handle, lsa_handle, sizeof (mslsa_handle_t)); 272 273 rc = ndr_rpc_call(lsa_handle, opnum, &arg); 274 ndr_rpc_release(lsa_handle); 275 return (rc); 276 } 277 278 /* 279 * lsar_query_info_policy 280 * 281 * The general purpose of this function is to allow various pieces of 282 * information to be queried on the domain controller. The only 283 * information queries supported are MSLSA_POLICY_PRIMARY_DOMAIN_INFO 284 * and MSLSA_POLICY_ACCOUNT_DOMAIN_INFO. 285 * 286 * On success, the return code will be 0 and the user_info structure 287 * will be set up. The sid_name_use field will be set to SidTypeDomain 288 * indicating that the domain name and domain sid fields are vaild. If 289 * the infoClass returned from the server is not one of the supported 290 * values, the sid_name_use willbe set to SidTypeUnknown. If the RPC 291 * fails, a negative error code will be returned, in which case the 292 * user_info will not have been updated. 293 */ 294 DWORD 295 lsar_query_info_policy(mlsvc_handle_t *lsa_handle, WORD infoClass, 296 smb_domain_t *info) 297 { 298 struct mslsa_QueryInfoPolicy arg; 299 struct mslsa_PrimaryDomainInfo *pd_info; 300 struct mslsa_AccountDomainInfo *ad_info; 301 struct mslsa_DnsDomainInfo *dns_info; 302 char guid_str[UUID_PRINTABLE_STRING_LENGTH]; 303 char sidstr[SMB_SID_STRSZ]; 304 int opnum; 305 DWORD status; 306 307 if (lsa_handle == NULL || info == NULL) 308 return (NT_STATUS_INVALID_PARAMETER); 309 310 opnum = LSARPC_OPNUM_QueryInfoPolicy; 311 312 bzero(info, sizeof (smb_domain_t)); 313 bzero(&arg, sizeof (struct mslsa_QueryInfoPolicy)); 314 (void) memcpy(&arg.handle, lsa_handle, sizeof (mslsa_handle_t)); 315 316 arg.info_class = infoClass; 317 318 if (ndr_rpc_call(lsa_handle, opnum, &arg) != 0) { 319 status = NT_STATUS_INVALID_PARAMETER; 320 } else if (arg.status != 0) { 321 ndr_rpc_status(lsa_handle, opnum, arg.status); 322 status = NT_SC_VALUE(arg.status); 323 } else { 324 325 switch (infoClass) { 326 case MSLSA_POLICY_PRIMARY_DOMAIN_INFO: 327 pd_info = &arg.ru.pd_info; 328 329 smb_sid_tostr((smb_sid_t *)pd_info->sid, sidstr); 330 info->di_type = SMB_DOMAIN_PRIMARY; 331 smb_domain_set_basic_info(sidstr, 332 (char *)pd_info->name.str, "", info); 333 334 status = NT_STATUS_SUCCESS; 335 break; 336 337 case MSLSA_POLICY_ACCOUNT_DOMAIN_INFO: 338 ad_info = &arg.ru.ad_info; 339 340 smb_sid_tostr((smb_sid_t *)ad_info->sid, sidstr); 341 info->di_type = SMB_DOMAIN_ACCOUNT; 342 smb_domain_set_basic_info(sidstr, 343 (char *)ad_info->name.str, "", info); 344 345 status = NT_STATUS_SUCCESS; 346 break; 347 348 case MSLSA_POLICY_DNS_DOMAIN_INFO: 349 dns_info = &arg.ru.dns_info; 350 ndr_uuid_unparse((ndr_uuid_t *)&dns_info->guid, 351 guid_str); 352 smb_sid_tostr((smb_sid_t *)dns_info->sid, sidstr); 353 354 info->di_type = SMB_DOMAIN_PRIMARY; 355 smb_domain_set_dns_info(sidstr, 356 (char *)dns_info->nb_domain.str, 357 (char *)dns_info->dns_domain.str, 358 (char *)dns_info->forest.str, 359 guid_str, info); 360 status = NT_STATUS_SUCCESS; 361 break; 362 363 default: 364 status = NT_STATUS_INVALID_INFO_CLASS; 365 break; 366 } 367 } 368 369 ndr_rpc_release(lsa_handle); 370 return (status); 371 } 372 373 /* 374 * Lookup a name and obtain the sid/rid. 375 * This is a wrapper for the various lookup sid RPCs. 376 */ 377 uint32_t 378 lsar_lookup_names(mlsvc_handle_t *lsa_handle, char *name, smb_account_t *info) 379 { 380 static lsar_nameop_t ops[] = { 381 lsar_lookup_names3, 382 lsar_lookup_names2, 383 lsar_lookup_names1 384 }; 385 386 lsa_names_t names; 387 char *p; 388 uint32_t length; 389 uint32_t status = NT_STATUS_INVALID_PARAMETER; 390 int n_op = (sizeof (ops) / sizeof (ops[0])); 391 int i; 392 393 if (lsa_handle == NULL || name == NULL || info == NULL) 394 return (NT_STATUS_INVALID_PARAMETER); 395 396 bzero(info, sizeof (smb_account_t)); 397 398 /* 399 * Windows 2000 (or later) doesn't like an LSA lookup for 400 * DOMAIN\Administrator. 401 */ 402 if ((p = strchr(name, '\\')) != 0) { 403 ++p; 404 405 if (strcasecmp(p, "administrator") == 0) 406 name = p; 407 } 408 409 length = smb_wcequiv_strlen(name); 410 names.name[0].length = length; 411 names.name[0].allosize = length; 412 names.name[0].str = (unsigned char *)name; 413 names.n_entry = 1; 414 415 for (i = 0; i < n_op; ++i) { 416 ndr_rpc_set_nonull(lsa_handle); 417 status = (*ops[i])(lsa_handle, &names, info); 418 419 if (status != NT_STATUS_INVALID_PARAMETER) 420 break; 421 } 422 423 if (status == NT_STATUS_SUCCESS) { 424 info->a_name = lsar_get_username(name); 425 426 if (!smb_account_validate(info)) { 427 smb_account_free(info); 428 status = NT_STATUS_NO_MEMORY; 429 } else { 430 smb_account_trace(info); 431 } 432 } 433 434 return (status); 435 } 436 437 /* 438 * The name may be in one of the following forms: 439 * 440 * domain\username 441 * domain/username 442 * username 443 * username@domain 444 * 445 * Return a strdup'd copy of the username. The caller is responsible 446 * for freeing the allocated memory. 447 */ 448 static char * 449 lsar_get_username(const char *name) 450 { 451 char tmp[MAXNAMELEN]; 452 char *dp = NULL; 453 char *np = NULL; 454 455 (void) strlcpy(tmp, name, MAXNAMELEN); 456 smb_name_parse(tmp, &np, &dp); 457 458 if (dp != NULL && np != NULL) 459 return (strdup(np)); 460 else 461 return (strdup(name)); 462 } 463 464 /* 465 * lsar_lookup_names1 466 * 467 * Lookup a name and obtain the domain and user rid. 468 * 469 * Note: NT returns an error if the mapped_count is non-zero when the RPC 470 * is called. 471 * 472 * If the lookup fails, the status will typically be NT_STATUS_NONE_MAPPED. 473 */ 474 static uint32_t 475 lsar_lookup_names1(mlsvc_handle_t *lsa_handle, lsa_names_t *names, 476 smb_account_t *info) 477 { 478 struct mslsa_LookupNames arg; 479 struct mslsa_rid_entry *rid_entry; 480 struct mslsa_domain_entry *domain_entry; 481 uint32_t status = NT_STATUS_SUCCESS; 482 char *domname; 483 int opnum = LSARPC_OPNUM_LookupNames; 484 485 bzero(&arg, sizeof (struct mslsa_LookupNames)); 486 (void) memcpy(&arg.handle, lsa_handle, sizeof (mslsa_handle_t)); 487 arg.lookup_level = LSA_LOOKUP_WKSTA; 488 arg.name_table = (struct mslsa_lup_name_table *)names; 489 490 if (ndr_rpc_call(lsa_handle, opnum, &arg) != 0) { 491 ndr_rpc_release(lsa_handle); 492 return (NT_STATUS_INVALID_PARAMETER); 493 } 494 495 if (arg.status != NT_STATUS_SUCCESS) { 496 ndr_rpc_status(lsa_handle, opnum, arg.status); 497 ndr_rpc_release(lsa_handle); 498 return (NT_SC_VALUE(arg.status)); 499 } 500 501 if (arg.mapped_count == 0) { 502 ndr_rpc_release(lsa_handle); 503 return (NT_STATUS_NONE_MAPPED); 504 } 505 506 rid_entry = &arg.translated_sids.rids[0]; 507 if (rid_entry->domain_index != 0) { 508 ndr_rpc_release(lsa_handle); 509 return (NT_STATUS_NONE_MAPPED); 510 } 511 512 domain_entry = &arg.domain_table->entries[0]; 513 514 info->a_type = rid_entry->sid_name_use; 515 info->a_domsid = smb_sid_dup((smb_sid_t *)domain_entry->domain_sid); 516 if ((domname = (char *)domain_entry->domain_name.str) != NULL) 517 info->a_domain = strdup(domname); 518 info->a_rid = rid_entry->rid; 519 info->a_sid = smb_sid_splice(info->a_domsid, info->a_rid); 520 521 ndr_rpc_release(lsa_handle); 522 return (status); 523 } 524 525 /* 526 * lsar_lookup_names2 527 */ 528 static uint32_t 529 lsar_lookup_names2(mlsvc_handle_t *lsa_handle, lsa_names_t *names, 530 smb_account_t *info) 531 { 532 struct lsar_LookupNames2 arg; 533 struct lsar_rid_entry2 *rid_entry; 534 struct mslsa_domain_entry *domain_entry; 535 uint32_t status = NT_STATUS_SUCCESS; 536 char *domname; 537 int opnum = LSARPC_OPNUM_LookupNames2; 538 539 bzero(&arg, sizeof (struct lsar_LookupNames2)); 540 (void) memcpy(&arg.policy_handle, lsa_handle, sizeof (mslsa_handle_t)); 541 arg.lookup_level = LSA_LOOKUP_WKSTA; 542 arg.client_revision = LSA_CLIENT_REVISION_AD; 543 arg.name_table = (struct mslsa_lup_name_table *)names; 544 545 if (ndr_rpc_call(lsa_handle, opnum, &arg) != 0) { 546 ndr_rpc_release(lsa_handle); 547 return (NT_STATUS_INVALID_PARAMETER); 548 } 549 550 if (arg.status != NT_STATUS_SUCCESS) { 551 ndr_rpc_status(lsa_handle, opnum, arg.status); 552 ndr_rpc_release(lsa_handle); 553 return (NT_SC_VALUE(arg.status)); 554 } 555 556 if (arg.mapped_count == 0) { 557 ndr_rpc_release(lsa_handle); 558 return (NT_STATUS_NONE_MAPPED); 559 } 560 561 rid_entry = &arg.translated_sids.rids[0]; 562 if (rid_entry->domain_index != 0) { 563 ndr_rpc_release(lsa_handle); 564 return (NT_STATUS_NONE_MAPPED); 565 } 566 567 domain_entry = &arg.domain_table->entries[0]; 568 569 info->a_type = rid_entry->sid_name_use; 570 info->a_domsid = smb_sid_dup((smb_sid_t *)domain_entry->domain_sid); 571 if ((domname = (char *)domain_entry->domain_name.str) != NULL) 572 info->a_domain = strdup(domname); 573 info->a_rid = rid_entry->rid; 574 info->a_sid = smb_sid_splice(info->a_domsid, info->a_rid); 575 576 ndr_rpc_release(lsa_handle); 577 return (status); 578 } 579 580 /* 581 * lsar_lookup_names3 582 */ 583 static uint32_t 584 lsar_lookup_names3(mlsvc_handle_t *lsa_handle, lsa_names_t *names, 585 smb_account_t *info) 586 { 587 struct lsar_LookupNames3 arg; 588 lsar_translated_sid_ex2_t *sid_entry; 589 struct mslsa_domain_entry *domain_entry; 590 uint32_t status = NT_STATUS_SUCCESS; 591 char *domname; 592 int opnum = LSARPC_OPNUM_LookupNames3; 593 594 bzero(&arg, sizeof (struct lsar_LookupNames3)); 595 (void) memcpy(&arg.policy_handle, lsa_handle, sizeof (mslsa_handle_t)); 596 arg.lookup_level = LSA_LOOKUP_WKSTA; 597 arg.client_revision = LSA_CLIENT_REVISION_AD; 598 arg.name_table = (struct mslsa_lup_name_table *)names; 599 600 if (ndr_rpc_call(lsa_handle, opnum, &arg) != 0) { 601 ndr_rpc_release(lsa_handle); 602 return (NT_STATUS_INVALID_PARAMETER); 603 } 604 605 if (arg.status != NT_STATUS_SUCCESS) { 606 ndr_rpc_status(lsa_handle, opnum, arg.status); 607 ndr_rpc_release(lsa_handle); 608 return (NT_SC_VALUE(arg.status)); 609 } 610 611 if (arg.mapped_count == 0) { 612 ndr_rpc_release(lsa_handle); 613 return (NT_STATUS_NONE_MAPPED); 614 } 615 616 sid_entry = &arg.translated_sids.sids[0]; 617 if (sid_entry->domain_index != 0) { 618 ndr_rpc_release(lsa_handle); 619 return (NT_STATUS_NONE_MAPPED); 620 } 621 622 domain_entry = &arg.domain_table->entries[0]; 623 624 info->a_type = sid_entry->sid_name_use; 625 info->a_domsid = smb_sid_dup((smb_sid_t *)domain_entry->domain_sid); 626 if ((domname = (char *)domain_entry->domain_name.str) != NULL) 627 info->a_domain = strdup(domname); 628 info->a_sid = smb_sid_dup((smb_sid_t *)sid_entry->sid); 629 (void) smb_sid_getrid(info->a_sid, &info->a_rid); 630 631 ndr_rpc_release(lsa_handle); 632 return (status); 633 } 634 635 /* 636 * lsar_lookup_names4 637 * 638 * This function is only valid if the remote RPC server is a domain 639 * controller and requires the security extensions defined in MS-RPCE. 640 * 641 * Domain controllers will return RPC_NT_PROTSEQ_NOT_SUPPORTED here 642 * because we don't support the RPC_C_AUTHN_NETLOGON security provider. 643 * Non-domain controllers will return NT_STATUS_INVALID_SERVER_STATE. 644 */ 645 static uint32_t /*LINTED E_STATIC_UNUSED*/ 646 lsar_lookup_names4(mlsvc_handle_t *lsa_handle, lsa_names_t *names, 647 smb_account_t *info) 648 { 649 struct lsar_LookupNames4 arg; 650 lsar_translated_sid_ex2_t *sid_entry; 651 struct mslsa_domain_entry *domain_entry; 652 uint32_t status = NT_STATUS_SUCCESS; 653 char *domname; 654 int opnum = LSARPC_OPNUM_LookupNames4; 655 656 bzero(&arg, sizeof (struct lsar_LookupNames4)); 657 arg.lookup_level = LSA_LOOKUP_WKSTA; 658 arg.client_revision = LSA_CLIENT_REVISION_AD; 659 arg.name_table = (struct mslsa_lup_name_table *)names; 660 661 if (ndr_rpc_call(lsa_handle, opnum, &arg) != 0) { 662 ndr_rpc_release(lsa_handle); 663 return (NT_STATUS_INVALID_PARAMETER); 664 } 665 666 if (arg.status != NT_STATUS_SUCCESS) { 667 ndr_rpc_status(lsa_handle, opnum, arg.status); 668 ndr_rpc_release(lsa_handle); 669 if (arg.status == RPC_NT_PROTSEQ_NOT_SUPPORTED || 670 arg.status == NT_STATUS_INVALID_SERVER_STATE) 671 return (NT_STATUS_INVALID_PARAMETER); 672 return (NT_SC_VALUE(arg.status)); 673 } 674 675 if (arg.mapped_count == 0) { 676 ndr_rpc_release(lsa_handle); 677 return (NT_STATUS_NONE_MAPPED); 678 } 679 680 sid_entry = &arg.translated_sids.sids[0]; 681 if (sid_entry->domain_index != 0) { 682 ndr_rpc_release(lsa_handle); 683 return (NT_STATUS_NONE_MAPPED); 684 } 685 686 domain_entry = &arg.domain_table->entries[0]; 687 688 info->a_type = sid_entry->sid_name_use; 689 info->a_domsid = smb_sid_dup((smb_sid_t *)domain_entry->domain_sid); 690 if ((domname = (char *)domain_entry->domain_name.str) != NULL) 691 info->a_domain = strdup(domname); 692 info->a_sid = smb_sid_dup((smb_sid_t *)sid_entry->sid); 693 (void) smb_sid_getrid(info->a_sid, &info->a_rid); 694 695 ndr_rpc_release(lsa_handle); 696 return (status); 697 } 698 699 /* 700 * Lookup a sid and obtain the domain sid and account name. 701 * This is a wrapper for the various lookup sid RPCs. 702 */ 703 uint32_t 704 lsar_lookup_sids(mlsvc_handle_t *lsa_handle, smb_sid_t *sid, 705 smb_account_t *account) 706 { 707 char sidbuf[SMB_SID_STRSZ]; 708 uint32_t status; 709 710 if (lsa_handle == NULL || sid == NULL || account == NULL) 711 return (NT_STATUS_INVALID_PARAMETER); 712 713 bzero(account, sizeof (smb_account_t)); 714 bzero(sidbuf, SMB_SID_STRSZ); 715 smb_sid_tostr(sid, sidbuf); 716 smb_tracef("%s", sidbuf); 717 718 status = lsar_lookup_sids2(lsa_handle, (lsa_sid_t *)sid, account); 719 if (status == RPC_NT_PROCNUM_OUT_OF_RANGE) 720 status = lsar_lookup_sids1(lsa_handle, (lsa_sid_t *)sid, 721 account); 722 723 if (status == NT_STATUS_SUCCESS) { 724 if (!smb_account_validate(account)) { 725 smb_account_free(account); 726 status = NT_STATUS_NO_MEMORY; 727 } else { 728 smb_account_trace(account); 729 } 730 } 731 732 return (status); 733 } 734 735 /* 736 * lsar_lookup_sids1 737 */ 738 static uint32_t 739 lsar_lookup_sids1(mlsvc_handle_t *lsa_handle, lsa_sid_t *sid, 740 smb_account_t *account) 741 { 742 struct mslsa_LookupSids arg; 743 struct mslsa_lup_sid_entry sid_entry; 744 struct mslsa_name_entry *name_entry; 745 struct mslsa_domain_entry *domain_entry; 746 uint32_t status = NT_STATUS_SUCCESS; 747 char *name; 748 int opnum = LSARPC_OPNUM_LookupSids; 749 750 bzero(&arg, sizeof (struct mslsa_LookupSids)); 751 (void) memcpy(&arg.handle, lsa_handle, sizeof (mslsa_handle_t)); 752 arg.lookup_level = LSA_LOOKUP_WKSTA; 753 754 sid_entry.psid = sid; 755 arg.lup_sid_table.n_entry = 1; 756 arg.lup_sid_table.entries = &sid_entry; 757 758 if (ndr_rpc_call(lsa_handle, opnum, &arg) != 0) { 759 ndr_rpc_release(lsa_handle); 760 return (NT_STATUS_INVALID_PARAMETER); 761 } 762 763 if (arg.status != NT_STATUS_SUCCESS) { 764 ndr_rpc_status(lsa_handle, opnum, arg.status); 765 ndr_rpc_release(lsa_handle); 766 return (NT_SC_VALUE(arg.status)); 767 } 768 769 if (arg.mapped_count == 0) { 770 ndr_rpc_release(lsa_handle); 771 return (NT_STATUS_NONE_MAPPED); 772 } 773 774 name_entry = &arg.name_table.entries[0]; 775 if (name_entry->domain_ix != 0) { 776 ndr_rpc_release(lsa_handle); 777 return (NT_STATUS_NONE_MAPPED); 778 } 779 780 name = (char *)name_entry->name.str; 781 account->a_name = (name) ? strdup(name) : strdup(""); 782 account->a_type = name_entry->sid_name_use; 783 account->a_sid = smb_sid_dup((smb_sid_t *)sid); 784 (void) smb_sid_getrid(account->a_sid, &account->a_rid); 785 786 domain_entry = &arg.domain_table->entries[0]; 787 if ((name = (char *)domain_entry->domain_name.str) != NULL) 788 account->a_domain = strdup(name); 789 account->a_domsid = smb_sid_dup((smb_sid_t *)domain_entry->domain_sid); 790 791 ndr_rpc_release(lsa_handle); 792 return (status); 793 } 794 795 /* 796 * lsar_lookup_sids2 797 */ 798 static uint32_t 799 lsar_lookup_sids2(mlsvc_handle_t *lsa_handle, lsa_sid_t *sid, 800 smb_account_t *account) 801 { 802 struct lsar_lookup_sids2 arg; 803 struct lsar_name_entry2 *name_entry; 804 struct mslsa_lup_sid_entry sid_entry; 805 struct mslsa_domain_entry *domain_entry; 806 uint32_t status = NT_STATUS_SUCCESS; 807 char *name; 808 int opnum = LSARPC_OPNUM_LookupSids2; 809 810 bzero(&arg, sizeof (struct lsar_lookup_sids2)); 811 (void) memcpy(&arg.policy_handle, lsa_handle, sizeof (mslsa_handle_t)); 812 813 sid_entry.psid = sid; 814 arg.lup_sid_table.n_entry = 1; 815 arg.lup_sid_table.entries = &sid_entry; 816 arg.lookup_level = LSA_LOOKUP_WKSTA; 817 arg.client_revision = LSA_CLIENT_REVISION_AD; 818 819 if (ndr_rpc_call(lsa_handle, opnum, &arg) != 0) { 820 ndr_rpc_release(lsa_handle); 821 return (NT_STATUS_INVALID_PARAMETER); 822 } 823 824 if (arg.status != NT_STATUS_SUCCESS) { 825 ndr_rpc_status(lsa_handle, opnum, arg.status); 826 ndr_rpc_release(lsa_handle); 827 return (NT_SC_VALUE(arg.status)); 828 } 829 830 if (arg.mapped_count == 0) { 831 ndr_rpc_release(lsa_handle); 832 return (NT_STATUS_NONE_MAPPED); 833 } 834 835 name_entry = &arg.name_table.entries[0]; 836 if (name_entry->domain_ix != 0) { 837 ndr_rpc_release(lsa_handle); 838 return (NT_STATUS_NONE_MAPPED); 839 } 840 841 name = (char *)name_entry->name.str; 842 account->a_name = (name) ? strdup(name) : strdup(""); 843 account->a_type = name_entry->sid_name_use; 844 account->a_sid = smb_sid_dup((smb_sid_t *)sid); 845 (void) smb_sid_getrid(account->a_sid, &account->a_rid); 846 847 domain_entry = &arg.domain_table->entries[0]; 848 if ((name = (char *)domain_entry->domain_name.str) != NULL) 849 account->a_domain = strdup(name); 850 account->a_domsid = smb_sid_dup((smb_sid_t *)domain_entry->domain_sid); 851 852 ndr_rpc_release(lsa_handle); 853 return (status); 854 } 855 856 /* 857 * lsar_lookup_sids3 858 * 859 * This function is only valid if the remote RPC server is a domain 860 * controller and requires the security extensions defined in MS-RPCE. 861 * 862 * Domain controllers will return RPC_NT_PROTSEQ_NOT_SUPPORTED here 863 * because we don't support the RPC_C_AUTHN_NETLOGON security provider. 864 * Non-domain controllers will return NT_STATUS_INVALID_SERVER_STATE. 865 */ 866 static uint32_t /*LINTED E_STATIC_UNUSED*/ 867 lsar_lookup_sids3(mlsvc_handle_t *lsa_handle, lsa_sid_t *sid, 868 smb_account_t *account) 869 { 870 struct lsar_lookup_sids3 arg; 871 lsar_translated_name_ex_t *name_entry; 872 struct mslsa_lup_sid_entry sid_entry; 873 struct mslsa_domain_entry *domain_entry; 874 uint32_t status = NT_STATUS_SUCCESS; 875 char *name; 876 int opnum = LSARPC_OPNUM_LookupSids3; 877 878 bzero(&arg, sizeof (struct lsar_lookup_sids3)); 879 880 sid_entry.psid = sid; 881 arg.lup_sid_table.n_entry = 1; 882 arg.lup_sid_table.entries = &sid_entry; 883 arg.lookup_level = LSA_LOOKUP_WKSTA; 884 arg.client_revision = LSA_CLIENT_REVISION_AD; 885 886 if (ndr_rpc_call(lsa_handle, opnum, &arg) != 0) { 887 ndr_rpc_release(lsa_handle); 888 return (NT_STATUS_INVALID_PARAMETER); 889 } 890 891 if (arg.status != NT_STATUS_SUCCESS) { 892 ndr_rpc_status(lsa_handle, opnum, arg.status); 893 ndr_rpc_release(lsa_handle); 894 if (arg.status == RPC_NT_PROTSEQ_NOT_SUPPORTED || 895 arg.status == NT_STATUS_INVALID_SERVER_STATE) 896 return (NT_STATUS_INVALID_PARAMETER); 897 return (NT_SC_VALUE(arg.status)); 898 } 899 900 if (arg.mapped_count == 0) { 901 ndr_rpc_release(lsa_handle); 902 return (NT_STATUS_NONE_MAPPED); 903 } 904 905 name_entry = &arg.name_table.entries[0]; 906 if (name_entry->domain_ix != 0) { 907 ndr_rpc_release(lsa_handle); 908 return (NT_STATUS_NONE_MAPPED); 909 } 910 911 name = (char *)name_entry->name.str; 912 account->a_name = (name) ? strdup(name) : strdup(""); 913 account->a_type = name_entry->sid_name_use; 914 account->a_sid = smb_sid_dup((smb_sid_t *)sid); 915 (void) smb_sid_getrid(account->a_sid, &account->a_rid); 916 917 domain_entry = &arg.domain_table->entries[0]; 918 if ((name = (char *)domain_entry->domain_name.str) != NULL) 919 account->a_domain = strdup(name); 920 account->a_domsid = smb_sid_dup((smb_sid_t *)domain_entry->domain_sid); 921 922 ndr_rpc_release(lsa_handle); 923 return (status); 924 } 925 926 /* 927 * lsar_enum_accounts 928 * 929 * Enumerate the list of accounts (i.e. SIDs). Use the handle returned 930 * from lsa_open_policy2. The enum_context is used to support multiple 931 * calls to this enumeration function. It should be set to 0 on the 932 * first call. It will be updated by the domain controller and should 933 * simply be passed unchanged to subsequent calls until there are no 934 * more accounts. A warning status of 0x1A indicates that no more data 935 * is available. The list of accounts will be returned in accounts. 936 * This list is dynamically allocated using malloc, it should be freed 937 * by the caller when it is no longer required. 938 */ 939 DWORD 940 lsar_enum_accounts(mlsvc_handle_t *lsa_handle, DWORD *enum_context, 941 struct mslsa_EnumAccountBuf *accounts) 942 { 943 struct mslsa_EnumerateAccounts arg; 944 struct mslsa_AccountInfo *info; 945 int opnum; 946 int rc; 947 DWORD status; 948 DWORD n_entries; 949 DWORD i; 950 int nbytes; 951 952 if (lsa_handle == NULL || enum_context == NULL || accounts == NULL) 953 return (NT_STATUS_INTERNAL_ERROR); 954 955 accounts->entries_read = 0; 956 accounts->info = 0; 957 958 opnum = LSARPC_OPNUM_EnumerateAccounts; 959 960 bzero(&arg, sizeof (struct mslsa_EnumerateAccounts)); 961 (void) memcpy(&arg.handle, lsa_handle, sizeof (mslsa_handle_t)); 962 arg.enum_context = *enum_context; 963 arg.max_length = MLSVC_MAX_RESPONSE_LEN; 964 965 rc = ndr_rpc_call(lsa_handle, opnum, &arg); 966 if (rc == 0) { 967 status = arg.status; 968 if (arg.status != 0) { 969 if (arg.status == NT_STATUS_NO_MORE_ENTRIES) { 970 *enum_context = arg.enum_context; 971 } else { 972 ndr_rpc_status(lsa_handle, opnum, arg.status); 973 } 974 } else if (arg.enum_buf->entries_read != 0) { 975 n_entries = arg.enum_buf->entries_read; 976 nbytes = n_entries * sizeof (struct mslsa_AccountInfo); 977 978 if ((info = malloc(nbytes)) == NULL) { 979 ndr_rpc_release(lsa_handle); 980 return (NT_STATUS_NO_MEMORY); 981 } 982 983 for (i = 0; i < n_entries; ++i) 984 info[i].sid = (lsa_sid_t *)smb_sid_dup( 985 (smb_sid_t *)arg.enum_buf->info[i].sid); 986 987 accounts->entries_read = n_entries; 988 accounts->info = info; 989 *enum_context = arg.enum_context; 990 } 991 } else { 992 status = NT_STATUS_INVALID_PARAMETER; 993 } 994 995 ndr_rpc_release(lsa_handle); 996 return (status); 997 } 998 999 /* 1000 * lsar_enum_trusted_domains 1001 * 1002 * Enumerate the list of trusted domains. Use the handle returned from 1003 * lsa_open_policy2. The enum_context is used to support multiple calls 1004 * to this enumeration function. It should be set to 0 on the first 1005 * call. It will be updated by the domain controller and should simply 1006 * be passed unchanged to subsequent calls until there are no more 1007 * domains. 1008 * 1009 * The trusted domains aren't actually returned here. They are added 1010 * to the NT domain database. After all of the trusted domains have 1011 * been discovered, the database can be interrogated to find all of 1012 * the trusted domains. 1013 */ 1014 DWORD 1015 lsar_enum_trusted_domains(mlsvc_handle_t *lsa_handle, DWORD *enum_context, 1016 smb_trusted_domains_t *list) 1017 { 1018 struct mslsa_EnumTrustedDomain arg; 1019 int opnum; 1020 DWORD status; 1021 1022 if (list == NULL) 1023 return (NT_STATUS_INVALID_PARAMETER); 1024 1025 opnum = LSARPC_OPNUM_EnumTrustedDomain; 1026 1027 bzero(list, sizeof (smb_trusted_domains_t)); 1028 bzero(&arg, sizeof (struct mslsa_EnumTrustedDomain)); 1029 (void) memcpy(&arg.handle, lsa_handle, sizeof (mslsa_handle_t)); 1030 arg.enum_context = *enum_context; 1031 arg.max_length = MLSVC_MAX_RESPONSE_LEN; 1032 1033 if (ndr_rpc_call(lsa_handle, opnum, &arg) != 0) { 1034 status = NT_STATUS_INVALID_PARAMETER; 1035 } else if (arg.status != 0) { 1036 *enum_context = arg.enum_context; 1037 status = NT_SC_VALUE(arg.status); 1038 1039 /* 1040 * STATUS_NO_MORE_ENTRIES provides call 1041 * status but does not indicate an error. 1042 */ 1043 if (status != NT_STATUS_NO_MORE_ENTRIES) 1044 ndr_rpc_status(lsa_handle, opnum, arg.status); 1045 } else if (arg.enum_buf->entries_read == 0) { 1046 *enum_context = arg.enum_context; 1047 status = 0; 1048 } else { 1049 lsar_set_trusted_domains(arg.enum_buf, list); 1050 *enum_context = arg.enum_context; 1051 status = 0; 1052 } 1053 1054 ndr_rpc_release(lsa_handle); 1055 return (status); 1056 } 1057 1058 DWORD 1059 lsar_enum_trusted_domains_ex(mlsvc_handle_t *lsa_handle, DWORD *enum_context, 1060 smb_trusted_domains_t *list) 1061 { 1062 struct mslsa_EnumTrustedDomainEx arg; 1063 int opnum; 1064 DWORD status; 1065 1066 if (list == NULL) 1067 return (NT_STATUS_INVALID_PARAMETER); 1068 1069 opnum = LSARPC_OPNUM_EnumTrustedDomainsEx; 1070 1071 bzero(list, sizeof (smb_trusted_domains_t)); 1072 bzero(&arg, sizeof (struct mslsa_EnumTrustedDomainEx)); 1073 (void) memcpy(&arg.handle, lsa_handle, sizeof (mslsa_handle_t)); 1074 arg.enum_context = *enum_context; 1075 arg.max_length = MLSVC_MAX_RESPONSE_LEN; 1076 1077 if (ndr_rpc_call(lsa_handle, opnum, &arg) != 0) { 1078 status = NT_STATUS_INVALID_PARAMETER; 1079 } else if (arg.status != 0) { 1080 *enum_context = arg.enum_context; 1081 status = NT_SC_VALUE(arg.status); 1082 1083 /* 1084 * STATUS_NO_MORE_ENTRIES provides call 1085 * status but does not indicate an error. 1086 */ 1087 if (status != NT_STATUS_NO_MORE_ENTRIES) 1088 ndr_rpc_status(lsa_handle, opnum, arg.status); 1089 } else if (arg.enum_buf->entries_read == 0) { 1090 *enum_context = arg.enum_context; 1091 status = 0; 1092 } else { 1093 lsar_set_trusted_domains_ex(arg.enum_buf, list); 1094 *enum_context = arg.enum_context; 1095 status = 0; 1096 } 1097 1098 ndr_rpc_release(lsa_handle); 1099 return (status); 1100 } 1101 1102 /* 1103 * lsar_enum_privs_account 1104 * 1105 * Privileges enum? Need an account handle. 1106 */ 1107 /*ARGSUSED*/ 1108 int 1109 lsar_enum_privs_account(mlsvc_handle_t *account_handle, smb_account_t *account) 1110 { 1111 struct mslsa_EnumPrivsAccount arg; 1112 int opnum; 1113 int rc; 1114 1115 opnum = LSARPC_OPNUM_EnumPrivsAccount; 1116 1117 bzero(&arg, sizeof (struct mslsa_EnumPrivsAccount)); 1118 (void) memcpy(&arg.account_handle, &account_handle->handle, 1119 sizeof (mslsa_handle_t)); 1120 1121 rc = ndr_rpc_call(account_handle, opnum, &arg); 1122 if ((rc == 0) && (arg.status != 0)) { 1123 ndr_rpc_status(account_handle, opnum, arg.status); 1124 rc = -1; 1125 } 1126 ndr_rpc_release(account_handle); 1127 return (rc); 1128 } 1129 1130 /* 1131 * lsar_lookup_priv_value 1132 * 1133 * Map a privilege name to a local unique id (LUID). Privilege names 1134 * are consistent across the network. LUIDs are machine specific. 1135 * This function provides the means to map a privilege name to the 1136 * LUID used by a remote server to represent it. The handle here is 1137 * a policy handle. 1138 */ 1139 int 1140 lsar_lookup_priv_value(mlsvc_handle_t *lsa_handle, char *name, 1141 struct ms_luid *luid) 1142 { 1143 struct mslsa_LookupPrivValue arg; 1144 int opnum; 1145 int rc; 1146 size_t length; 1147 1148 if (lsa_handle == NULL || name == NULL || luid == NULL) 1149 return (-1); 1150 1151 opnum = LSARPC_OPNUM_LookupPrivValue; 1152 1153 bzero(&arg, sizeof (struct mslsa_LookupPrivValue)); 1154 (void) memcpy(&arg.handle, lsa_handle, sizeof (mslsa_handle_t)); 1155 1156 length = smb_wcequiv_strlen(name); 1157 length += sizeof (smb_wchar_t); 1158 1159 arg.name.length = length; 1160 arg.name.allosize = length; 1161 arg.name.str = (unsigned char *)name; 1162 1163 rc = ndr_rpc_call(lsa_handle, opnum, &arg); 1164 if (rc == 0) { 1165 if (arg.status != 0) 1166 rc = -1; 1167 else 1168 (void) memcpy(luid, &arg.luid, sizeof (struct ms_luid)); 1169 } 1170 1171 ndr_rpc_release(lsa_handle); 1172 return (rc); 1173 } 1174 1175 /* 1176 * lsar_lookup_priv_name 1177 * 1178 * Map a local unique id (LUID) to a privilege name. Privilege names 1179 * are consistent across the network. LUIDs are machine specific. 1180 * This function the means to map the LUID used by a remote server to 1181 * the appropriate privilege name. The handle here is a policy handle. 1182 */ 1183 int 1184 lsar_lookup_priv_name(mlsvc_handle_t *lsa_handle, struct ms_luid *luid, 1185 char *name, int namelen) 1186 { 1187 struct mslsa_LookupPrivName arg; 1188 int opnum; 1189 int rc; 1190 1191 if (lsa_handle == NULL || luid == NULL || name == NULL) 1192 return (-1); 1193 1194 opnum = LSARPC_OPNUM_LookupPrivName; 1195 1196 bzero(&arg, sizeof (struct mslsa_LookupPrivName)); 1197 (void) memcpy(&arg.handle, lsa_handle, sizeof (mslsa_handle_t)); 1198 (void) memcpy(&arg.luid, luid, sizeof (struct ms_luid)); 1199 1200 rc = ndr_rpc_call(lsa_handle, opnum, &arg); 1201 if (rc == 0) { 1202 if (arg.status != 0) 1203 rc = -1; 1204 else 1205 (void) strlcpy(name, (char const *)arg.name->str, 1206 namelen); 1207 } 1208 1209 ndr_rpc_release(lsa_handle); 1210 return (rc); 1211 } 1212 1213 /* 1214 * lsar_lookup_priv_display_name 1215 * 1216 * Map a privilege name to a privilege display name. The input handle 1217 * should be an LSA policy handle and the name would normally be one 1218 * of the privileges defined in smb_privilege.h 1219 * 1220 * There's something peculiar about the return status from NT servers, 1221 * it's not always present. So for now, I'm ignoring the status in the 1222 * RPC response. 1223 * 1224 * Returns NT status codes. 1225 */ 1226 DWORD 1227 lsar_lookup_priv_display_name(mlsvc_handle_t *lsa_handle, char *name, 1228 char *display_name, int display_len) 1229 { 1230 struct mslsa_LookupPrivDisplayName arg; 1231 int opnum; 1232 size_t length; 1233 DWORD status; 1234 1235 if (lsa_handle == NULL || name == NULL || display_name == NULL) 1236 return (NT_STATUS_INVALID_PARAMETER); 1237 1238 opnum = LSARPC_OPNUM_LookupPrivDisplayName; 1239 1240 bzero(&arg, sizeof (struct mslsa_LookupPrivDisplayName)); 1241 (void) memcpy(&arg.handle, lsa_handle, sizeof (mslsa_handle_t)); 1242 1243 length = smb_wcequiv_strlen(name); 1244 arg.name.length = length; 1245 arg.name.allosize = length; 1246 arg.name.str = (unsigned char *)name; 1247 1248 arg.client_language = MAKELANGID(LANG_ENGLISH, SUBLANG_ENGLISH_US); 1249 arg.default_language = MAKELANGID(LANG_ENGLISH, SUBLANG_NEUTRAL); 1250 1251 if (ndr_rpc_call(lsa_handle, opnum, &arg) != 0) 1252 status = NT_STATUS_INVALID_PARAMETER; 1253 #if 0 1254 else if (arg.status != 0) 1255 status = NT_SC_VALUE(arg.status); 1256 #endif 1257 else { 1258 (void) strlcpy(display_name, 1259 (char const *)arg.display_name->str, display_len); 1260 status = NT_STATUS_SUCCESS; 1261 } 1262 1263 ndr_rpc_release(lsa_handle); 1264 return (status); 1265 } 1266 1267 static void 1268 lsar_set_trusted_domains_ex(struct mslsa_EnumTrustedDomainBufEx *enum_buf, 1269 smb_trusted_domains_t *list) 1270 { 1271 char sidstr[SMB_SID_STRSZ]; 1272 int i; 1273 1274 if (list == NULL || enum_buf == NULL || enum_buf->entries_read == 0) 1275 return; 1276 1277 list->td_num = 0; 1278 list->td_domains = calloc(enum_buf->entries_read, 1279 sizeof (smb_domain_t)); 1280 1281 if (list->td_domains == NULL) 1282 return; 1283 1284 list->td_num = enum_buf->entries_read; 1285 for (i = 0; i < list->td_num; i++) { 1286 smb_sid_tostr((smb_sid_t *)enum_buf->info[i].sid, sidstr); 1287 smb_domain_set_trust_info( 1288 sidstr, 1289 (char *)enum_buf->info[i].nb_name.str, 1290 (char *)enum_buf->info[i].dns_name.str, 1291 enum_buf->info[i].trust_direction, 1292 enum_buf->info[i].trust_type, 1293 enum_buf->info[i].trust_attrs, 1294 &list->td_domains[i]); 1295 } 1296 } 1297 1298 static void 1299 lsar_set_trusted_domains(struct mslsa_EnumTrustedDomainBuf *enum_buf, 1300 smb_trusted_domains_t *list) 1301 { 1302 char sidstr[SMB_SID_STRSZ]; 1303 int i; 1304 1305 if (list == NULL || enum_buf == NULL || enum_buf->entries_read == 0) 1306 return; 1307 1308 list->td_num = 0; 1309 list->td_domains = calloc(enum_buf->entries_read, 1310 sizeof (smb_domain_t)); 1311 1312 if (list->td_domains == NULL) 1313 return; 1314 1315 list->td_num = enum_buf->entries_read; 1316 for (i = 0; i < list->td_num; i++) { 1317 smb_sid_tostr((smb_sid_t *)enum_buf->info[i].sid, sidstr); 1318 smb_domain_set_trust_info( 1319 sidstr, (char *)enum_buf->info[i].name.str, 1320 "", 0, 0, 0, &list->td_domains[i]); 1321 } 1322 } 1323 1324 static void 1325 smb_account_trace(const smb_account_t *info) 1326 { 1327 char sidbuf[SMB_SID_STRSZ]; 1328 1329 bzero(sidbuf, SMB_SID_STRSZ); 1330 smb_sid_tostr(info->a_sid, sidbuf); 1331 1332 smb_tracef("%s %s %s %lu %s", info->a_domain, info->a_name, 1333 sidbuf, info->a_rid, smb_sid_type2str(info->a_type)); 1334 }