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 2014 Nexenta Systems, Inc.  All rights reserved.
  25  */
  26 
  27 /*
  28  * Security Accounts Manager RPC (SAMR) client-side interface.
  29  *
  30  * The SAM is a hierarchical database:
  31  * - If you want to talk to the SAM you need a SAM handle.
  32  * - If you want to work with a domain, use the SAM handle.
  33  *   to obtain a domain handle.
  34  * - Use domain handles to obtain user handles etc.
  35  *
  36  * Be careful about returning null handles to the application.  Use of a
  37  * null handle may crash the domain controller if you attempt to use it.
  38  */
  39 
  40 #include <stdio.h>
  41 #include <strings.h>
  42 #include <stdlib.h>
  43 #include <unistd.h>
  44 #include <netdb.h>
  45 #include <sys/param.h>
  46 
  47 #include <smbsrv/libsmb.h>
  48 #include <smbsrv/libmlrpc.h>
  49 #include <smbsrv/libmlsvc.h>
  50 #include <smbsrv/smbinfo.h>
  51 #include <smbsrv/ntaccess.h>
  52 #include <smbsrv/smb_sid.h>
  53 #include <samlib.h>
  54 
  55 static DWORD samr_connect2(char *, char *, char *, DWORD, mlsvc_handle_t *);
  56 static DWORD samr_connect4(char *, char *, char *, DWORD, mlsvc_handle_t *);
  57 static DWORD samr_connect5(char *, char *, char *, DWORD, mlsvc_handle_t *);
  58 
  59 typedef DWORD (*samr_connop_t)(char *, char *, char *, DWORD,
  60     mlsvc_handle_t *);
  61 
  62 static int samr_setup_user_info(WORD, struct samr_QueryUserInfo *,
  63     union samr_user_info *);
  64 
  65 /*
  66  * samr_open
  67  *
  68  * Wrapper round samr_connect to ensure that we connect using the server
  69  * and domain.  We default to the resource domain if the caller doesn't
  70  * supply a server name and a domain name.
  71  *
  72  * If username argument is NULL, an anonymous connection will be established.
  73  * Otherwise, an authenticated connection will be established.
  74  *
  75  * On success 0 is returned. Otherwise a -ve error code.
  76  */
  77 DWORD
  78 samr_open(char *server, char *domain, char *username, DWORD access_mask,
  79     mlsvc_handle_t *samr_handle)
  80 {
  81         smb_domainex_t di;
  82         DWORD status;
  83 
  84         if (server == NULL || domain == NULL) {
  85                 if (!smb_domain_getinfo(&di))
  86                         return (NT_STATUS_INTERNAL_ERROR);
  87                 server = di.d_dci.dc_name;
  88                 domain = di.d_primary.di_nbname;
  89         }
  90 
  91         if (username == NULL)
  92                 username = MLSVC_ANON_USER;
  93 
  94         status = samr_connect(server, domain, username, access_mask,
  95             samr_handle);
  96 
  97         return (status);
  98 }
  99 
 100 
 101 /*
 102  * samr_connect
 103  *
 104  * Connect to the SAMR service on the specified server (domain controller).
 105  * New SAM connect calls have been added to Windows over time:
 106  *
 107  *      Windows NT3.x:  SamrConnect
 108  *      Windows NT4.0:  SamrConnect2
 109  *      Windows 2000:   SamrConnect4
 110  *      Windows XP:     SamrConnect5
 111  *
 112  * Try the calls from most recent to oldest until the server responds with
 113  * something other than an RPC protocol error.  We don't use the original
 114  * connect call because all supported servers should support SamrConnect2.
 115  */
 116 DWORD
 117 samr_connect(char *server, char *domain, char *username, DWORD access_mask,
 118     mlsvc_handle_t *samr_handle)
 119 {
 120         static samr_connop_t samr_connop[] = {
 121                 samr_connect5,
 122                 samr_connect4,
 123                 samr_connect2
 124         };
 125 
 126         int     n_op = (sizeof (samr_connop) / sizeof (samr_connop[0]));
 127         DWORD   status;
 128         int     i;
 129 
 130         status = ndr_rpc_bind(samr_handle, server, domain, username, "SAMR");
 131         if (status)
 132                 return (status);
 133 
 134         for (i = 0; i < n_op; ++i) {
 135                 status = (*samr_connop[i])(server, domain, username,
 136                     access_mask, samr_handle);
 137 
 138                 if (status == NT_STATUS_SUCCESS)
 139                         return (status);
 140         }
 141 
 142         ndr_rpc_unbind(samr_handle);
 143         return (status);
 144 }
 145 
 146 /*
 147  * samr_connect2
 148  *
 149  * Connect to the SAM on a Windows NT 4.0 server (domain controller).
 150  * We need the domain controller name and, if everything works, we
 151  * return a handle.  This function adds the double backslash prefx to
 152  * make it easy for applications.
 153  *
 154  * Returns 0 on success. Otherwise returns a -ve error code.
 155  */
 156 /*ARGSUSED*/
 157 static DWORD
 158 samr_connect2(char *server, char *domain, char *username, DWORD access_mask,
 159     mlsvc_handle_t *samr_handle)
 160 {
 161         struct samr_Connect2 arg;
 162         int opnum;
 163         DWORD status;
 164         int len;
 165 
 166         bzero(&arg, sizeof (struct samr_Connect2));
 167         opnum = SAMR_OPNUM_Connect2;
 168         status = NT_STATUS_SUCCESS;
 169 
 170         len = strlen(server) + 4;
 171         arg.servername = ndr_rpc_malloc(samr_handle, len);
 172         (void) snprintf((char *)arg.servername, len, "\\\\%s", server);
 173         arg.access_mask = access_mask;
 174 
 175         if (ndr_rpc_call(samr_handle, opnum, &arg) != 0) {
 176                 status = NT_STATUS_UNSUCCESSFUL;
 177         } else if (arg.status != 0) {
 178                 status = NT_SC_VALUE(arg.status);
 179         } else {
 180                 (void) memcpy(&samr_handle->handle, &arg.handle,
 181                     sizeof (ndr_hdid_t));
 182 
 183                 if (ndr_is_null_handle(samr_handle))
 184                         status = NT_STATUS_INVALID_HANDLE;
 185         }
 186 
 187         ndr_rpc_release(samr_handle);
 188         return (status);
 189 }
 190 
 191 /*
 192  * samr_connect4
 193  *
 194  * Connect to the SAM on a Windows 2000 domain controller.
 195  */
 196 /*ARGSUSED*/
 197 static DWORD
 198 samr_connect4(char *server, char *domain, char *username, DWORD access_mask,
 199     mlsvc_handle_t *samr_handle)
 200 {
 201         struct samr_Connect4 arg;
 202         int opnum;
 203         DWORD status;
 204         int len;
 205 
 206         bzero(&arg, sizeof (struct samr_Connect4));
 207         opnum = SAMR_OPNUM_Connect4;
 208         status = NT_STATUS_SUCCESS;
 209 
 210         len = strlen(server) + 4;
 211         arg.servername = ndr_rpc_malloc(samr_handle, len);
 212         (void) snprintf((char *)arg.servername, len, "\\\\%s", server);
 213         arg.revision = SAMR_REVISION_2;
 214         arg.access_mask = access_mask;
 215 
 216         if (ndr_rpc_call(samr_handle, opnum, &arg) != 0) {
 217                 status = NT_STATUS_UNSUCCESSFUL;
 218         } else if (arg.status != 0) {
 219                 status = NT_SC_VALUE(arg.status);
 220         } else {
 221                 (void) memcpy(&samr_handle->handle, &arg.handle,
 222                     sizeof (ndr_hdid_t));
 223 
 224                 if (ndr_is_null_handle(samr_handle))
 225                         status = NT_STATUS_INVALID_HANDLE;
 226         }
 227 
 228         ndr_rpc_release(samr_handle);
 229         return (status);
 230 }
 231 
 232 /*
 233  * samr_connect5
 234  *
 235  * Connect to the SAM on a Windows XP domain controller.  On Windows
 236  * XP, the server should be the fully qualified DNS domain name with
 237  * a double backslash prefix.  At this point, it is assumed that we
 238  * need to add the prefix and the DNS domain name here.
 239  *
 240  * If this call succeeds, a SAMR handle is placed in samr_handle and
 241  * zero is returned. Otherwise, a -ve error code is returned.
 242  */
 243 /*ARGSUSED*/
 244 static DWORD
 245 samr_connect5(char *server, char *domain, char *username, DWORD access_mask,
 246     mlsvc_handle_t *samr_handle)
 247 {
 248         struct samr_Connect5 arg;
 249         int len;
 250         int opnum;
 251         DWORD status;
 252 
 253         bzero(&arg, sizeof (struct samr_Connect5));
 254         opnum = SAMR_OPNUM_Connect5;
 255         status = NT_STATUS_SUCCESS;
 256 
 257         len = strlen(server) + 4;
 258         arg.servername = ndr_rpc_malloc(samr_handle, len);
 259         (void) snprintf((char *)arg.servername, len, "\\\\%s", server);
 260 
 261         arg.access_mask = SAM_ENUM_LOCAL_DOMAIN;
 262         arg.unknown2_00000001 = 0x00000001;
 263         arg.unknown3_00000001 = 0x00000001;
 264         arg.unknown4_00000003 = 0x00000003;
 265         arg.unknown5_00000000 = 0x00000000;
 266 
 267         if (ndr_rpc_call(samr_handle, opnum, &arg) != 0) {
 268                 status = NT_STATUS_UNSUCCESSFUL;
 269         } else if (arg.status != 0) {
 270                 status = NT_SC_VALUE(arg.status);
 271         } else {
 272 
 273                 (void) memcpy(&samr_handle->handle, &arg.handle,
 274                     sizeof (ndr_hdid_t));
 275 
 276                 if (ndr_is_null_handle(samr_handle))
 277                         status = NT_STATUS_INVALID_HANDLE;
 278         }
 279 
 280         ndr_rpc_release(samr_handle);
 281         return (status);
 282 }
 283 
 284 
 285 /*
 286  * samr_close_handle
 287  *
 288  * This is function closes any valid handle, i.e. sam, domain, user etc.
 289  * If the handle being closed is the top level connect handle, we unbind.
 290  * Then we zero out the handle to invalidate it.
 291  */
 292 void
 293 samr_close_handle(mlsvc_handle_t *samr_handle)
 294 {
 295         struct samr_CloseHandle arg;
 296         int opnum;
 297 
 298         if (ndr_is_null_handle(samr_handle))
 299                 return;
 300 
 301         opnum = SAMR_OPNUM_CloseHandle;
 302         bzero(&arg, sizeof (struct samr_CloseHandle));
 303         (void) memcpy(&arg.handle, &samr_handle->handle, sizeof (ndr_hdid_t));
 304 
 305         (void) ndr_rpc_call(samr_handle, opnum, &arg);
 306         ndr_rpc_release(samr_handle);
 307 
 308         if (ndr_is_bind_handle(samr_handle))
 309                 ndr_rpc_unbind(samr_handle);
 310 
 311         bzero(samr_handle, sizeof (mlsvc_handle_t));
 312 }
 313 
 314 /*
 315  * samr_open_domain
 316  *
 317  * We use a SAM handle to obtain a handle for a domain, specified by
 318  * the SID. The SID can be obtain via the LSA interface. A handle for
 319  * the domain is returned in domain_handle.
 320  */
 321 DWORD
 322 samr_open_domain(mlsvc_handle_t *samr_handle, DWORD access_mask,
 323     struct samr_sid *sid, mlsvc_handle_t *domain_handle)
 324 {
 325         struct samr_OpenDomain arg;
 326         int opnum;
 327         DWORD status;
 328 
 329         if (ndr_is_null_handle(samr_handle) ||
 330             sid == NULL || domain_handle == NULL) {
 331                 return (NT_STATUS_INVALID_PARAMETER);
 332         }
 333 
 334         opnum = SAMR_OPNUM_OpenDomain;
 335         bzero(&arg, sizeof (struct samr_OpenDomain));
 336         (void) memcpy(&arg.handle, &samr_handle->handle, sizeof (ndr_hdid_t));
 337 
 338         arg.access_mask = access_mask;
 339         arg.sid = sid;
 340 
 341         if (ndr_rpc_call(samr_handle, opnum, &arg) != 0) {
 342                 status = NT_STATUS_UNSUCCESSFUL;
 343         } else if (arg.status != 0) {
 344                 status = arg.status;
 345         } else {
 346                 status = NT_STATUS_SUCCESS;
 347                 ndr_inherit_handle(domain_handle, samr_handle);
 348 
 349                 (void) memcpy(&domain_handle->handle, &arg.domain_handle,
 350                     sizeof (ndr_hdid_t));
 351 
 352                 if (ndr_is_null_handle(domain_handle))
 353                         status = NT_STATUS_INVALID_HANDLE;
 354         }
 355 
 356         if (status != NT_STATUS_SUCCESS)
 357                 ndr_rpc_status(samr_handle, opnum, status);
 358 
 359         ndr_rpc_release(samr_handle);
 360         return (status);
 361 }
 362 
 363 /*
 364  * samr_open_user
 365  *
 366  * Use a domain handle to obtain a handle for a user, specified by the
 367  * user RID. A user RID (effectively a uid) can be obtained via the
 368  * LSA interface. A handle for the user is returned in user_handle.
 369  * Once you have a user handle it should be possible to query the SAM
 370  * for information on that user.
 371  */
 372 DWORD
 373 samr_open_user(mlsvc_handle_t *domain_handle, DWORD access_mask, DWORD rid,
 374     mlsvc_handle_t *user_handle)
 375 {
 376         struct samr_OpenUser arg;
 377         int opnum;
 378         DWORD status = NT_STATUS_SUCCESS;
 379 
 380         if (ndr_is_null_handle(domain_handle) || user_handle == NULL)
 381                 return (NT_STATUS_INVALID_PARAMETER);
 382 
 383         opnum = SAMR_OPNUM_OpenUser;
 384         bzero(&arg, sizeof (struct samr_OpenUser));
 385         (void) memcpy(&arg.handle, &domain_handle->handle,
 386             sizeof (ndr_hdid_t));
 387         arg.access_mask = access_mask;
 388         arg.rid = rid;
 389 
 390         if (ndr_rpc_call(domain_handle, opnum, &arg) != 0) {
 391                 status = NT_STATUS_UNSUCCESSFUL;
 392         } else if (arg.status != 0) {
 393                 ndr_rpc_status(domain_handle, opnum, arg.status);
 394                 status = NT_SC_VALUE(arg.status);
 395         } else {
 396                 ndr_inherit_handle(user_handle, domain_handle);
 397 
 398                 (void) memcpy(&user_handle->handle, &arg.user_handle,
 399                     sizeof (ndr_hdid_t));
 400 
 401                 if (ndr_is_null_handle(user_handle))
 402                         status = NT_STATUS_INVALID_HANDLE;
 403         }
 404 
 405         ndr_rpc_release(domain_handle);
 406         return (status);
 407 }
 408 
 409 /*
 410  * samr_delete_user
 411  *
 412  * Delete the user specified by the user_handle.
 413  */
 414 DWORD
 415 samr_delete_user(mlsvc_handle_t *user_handle)
 416 {
 417         struct samr_DeleteUser arg;
 418         int opnum;
 419         DWORD status;
 420 
 421         if (ndr_is_null_handle(user_handle))
 422                 return (NT_STATUS_INVALID_PARAMETER);
 423 
 424         opnum = SAMR_OPNUM_DeleteUser;
 425         bzero(&arg, sizeof (struct samr_DeleteUser));
 426         (void) memcpy(&arg.user_handle, &user_handle->handle,
 427             sizeof (ndr_hdid_t));
 428 
 429         if (ndr_rpc_call(user_handle, opnum, &arg) != 0) {
 430                 status = NT_STATUS_INVALID_PARAMETER;
 431         } else if (arg.status != 0) {
 432                 ndr_rpc_status(user_handle, opnum, arg.status);
 433                 status = NT_SC_VALUE(arg.status);
 434         } else {
 435                 status = 0;
 436         }
 437 
 438         ndr_rpc_release(user_handle);
 439         return (status);
 440 }
 441 
 442 /*
 443  * samr_open_group
 444  *
 445  * Use a domain handle to obtain a handle for a group, specified by the
 446  * group RID. A group RID (effectively a gid) can be obtained via the
 447  * LSA interface. A handle for the group is returned in group_handle.
 448  * Once you have a group handle it should be possible to query the SAM
 449  * for information on that group.
 450  */
 451 int
 452 samr_open_group(
 453         mlsvc_handle_t *domain_handle,
 454         DWORD rid,
 455         mlsvc_handle_t *group_handle)
 456 {
 457         struct samr_OpenGroup arg;
 458         int opnum;
 459         int rc;
 460 
 461         if (ndr_is_null_handle(domain_handle) || group_handle == NULL)
 462                 return (-1);
 463 
 464         opnum = SAMR_OPNUM_OpenGroup;
 465         bzero(&arg, sizeof (struct samr_OpenUser));
 466         (void) memcpy(&arg.handle, &domain_handle->handle,
 467             sizeof (ndr_hdid_t));
 468         arg.access_mask = SAM_LOOKUP_INFORMATION | SAM_ACCESS_USER_READ;
 469         arg.rid = rid;
 470 
 471         if ((rc = ndr_rpc_call(domain_handle, opnum, &arg)) != 0)
 472                 return (-1);
 473 
 474         if (arg.status != 0) {
 475                 ndr_rpc_status(domain_handle, opnum, arg.status);
 476                 rc = -1;
 477         } else {
 478                 ndr_inherit_handle(group_handle, domain_handle);
 479 
 480                 (void) memcpy(&group_handle->handle, &arg.group_handle,
 481                     sizeof (ndr_hdid_t));
 482 
 483                 if (ndr_is_null_handle(group_handle))
 484                         rc = -1;
 485         }
 486 
 487         ndr_rpc_release(domain_handle);
 488         return (rc);
 489 }
 490 
 491 /*
 492  * samr_create_user
 493  *
 494  * Create a user in the domain specified by the domain handle. If this
 495  * call is successful, the server will return the RID for the user and
 496  * a user handle, which may be used to set or query the SAM.
 497  *
 498  * Observed status codes:
 499  *      NT_STATUS_INVALID_PARAMETER
 500  *      NT_STATUS_INVALID_ACCOUNT_NAME
 501  *      NT_STATUS_ACCESS_DENIED
 502  *      NT_STATUS_USER_EXISTS
 503  *
 504  * Returns 0 on success. Otherwise returns an NT status code.
 505  */
 506 DWORD
 507 samr_create_user(mlsvc_handle_t *domain_handle, char *username,
 508     DWORD account_flags, DWORD *rid, mlsvc_handle_t *user_handle)
 509 {
 510         struct samr_CreateUser arg;
 511         ndr_heap_t *heap;
 512         int opnum;
 513         int rc;
 514         DWORD status = 0;
 515 
 516         if (ndr_is_null_handle(domain_handle) ||
 517             username == NULL || rid == NULL) {
 518                 return (NT_STATUS_INVALID_PARAMETER);
 519         }
 520 
 521         opnum = SAMR_OPNUM_CreateUser;
 522 
 523         bzero(&arg, sizeof (struct samr_CreateUser));
 524         (void) memcpy(&arg.handle, &domain_handle->handle,
 525             sizeof (ndr_hdid_t));
 526 
 527         heap = ndr_rpc_get_heap(domain_handle);
 528         ndr_heap_mkvcs(heap, username, (ndr_vcstr_t *)&arg.username);
 529 
 530         arg.account_flags = account_flags;
 531         arg.desired_access = 0xE00500B0;
 532 
 533         rc = ndr_rpc_call(domain_handle, opnum, &arg);
 534         if (rc != 0) {
 535                 status = NT_STATUS_INVALID_PARAMETER;
 536         } else if (arg.status != 0) {
 537                 status = NT_SC_VALUE(arg.status);
 538 
 539                 if (status != NT_STATUS_USER_EXISTS) {
 540                         smb_tracef("SamrCreateUser[%s]: %s",
 541                             username, xlate_nt_status(status));
 542                 }
 543         } else {
 544                 ndr_inherit_handle(user_handle, domain_handle);
 545 
 546                 (void) memcpy(&user_handle->handle, &arg.user_handle,
 547                     sizeof (ndr_hdid_t));
 548 
 549                 *rid = arg.rid;
 550 
 551                 if (ndr_is_null_handle(user_handle))
 552                         status = NT_STATUS_INVALID_HANDLE;
 553                 else
 554                         status = 0;
 555         }
 556 
 557         ndr_rpc_release(domain_handle);
 558         return (status);
 559 }
 560 
 561 /*
 562  * samr_lookup_domain
 563  *
 564  * Lookup up the domain SID for the specified domain name. The handle
 565  * should be one returned from samr_connect. The allocated memory for
 566  * the returned SID must be freed by caller.
 567  */
 568 smb_sid_t *
 569 samr_lookup_domain(mlsvc_handle_t *samr_handle, char *domain_name)
 570 {
 571         struct samr_LookupDomain        arg;
 572         smb_sid_t       *domsid = NULL;
 573         int             opnum;
 574         size_t          length;
 575 
 576         if (ndr_is_null_handle(samr_handle) || domain_name == NULL)
 577                 return (NULL);
 578 
 579         opnum = SAMR_OPNUM_LookupDomain;
 580         bzero(&arg, sizeof (struct samr_LookupDomain));
 581 
 582         (void) memcpy(&arg.handle, &samr_handle->handle,
 583             sizeof (samr_handle_t));
 584 
 585         length = smb_wcequiv_strlen(domain_name);
 586         length += sizeof (smb_wchar_t);
 587 
 588         arg.domain_name.length = length;
 589         arg.domain_name.allosize = length;
 590         arg.domain_name.str = (unsigned char *)domain_name;
 591 
 592         if (ndr_rpc_call(samr_handle, opnum, &arg) == 0)
 593                 domsid = smb_sid_dup((smb_sid_t *)arg.sid);
 594 
 595         ndr_rpc_release(samr_handle);
 596         return (domsid);
 597 }
 598 
 599 /*
 600  * samr_enum_local_domains
 601  *
 602  * Get the list of local domains supported by a server.
 603  *
 604  * Returns NT status codes.
 605  */
 606 DWORD
 607 samr_enum_local_domains(mlsvc_handle_t *samr_handle)
 608 {
 609         struct samr_EnumLocalDomain     arg;
 610         int     opnum;
 611         DWORD   status;
 612 
 613         if (ndr_is_null_handle(samr_handle))
 614                 return (NT_STATUS_INVALID_PARAMETER);
 615 
 616         opnum = SAMR_OPNUM_EnumLocalDomains;
 617         bzero(&arg, sizeof (struct samr_EnumLocalDomain));
 618 
 619         (void) memcpy(&arg.handle, &samr_handle->handle,
 620             sizeof (samr_handle_t));
 621         arg.enum_context = 0;
 622         arg.max_length = 0x00002000;    /* Value used by NT */
 623 
 624         if (ndr_rpc_call(samr_handle, opnum, &arg) != 0) {
 625                 status = NT_STATUS_INVALID_PARAMETER;
 626         } else {
 627                 status = NT_SC_VALUE(arg.status);
 628 
 629                 /*
 630                  * Handle none-mapped status quietly.
 631                  */
 632                 if (status != NT_STATUS_NONE_MAPPED)
 633                         ndr_rpc_status(samr_handle, opnum, arg.status);
 634         }
 635 
 636         ndr_rpc_release(samr_handle);
 637         return (status);
 638 }
 639 
 640 /*
 641  * samr_lookup_domain_names
 642  *
 643  * Lookup up the given name in the domain specified by domain_handle.
 644  * Upon a successful lookup the information is returned in the account
 645  * arg and caller must free allocated memories by calling smb_account_free().
 646  *
 647  * Returns NT status codes.
 648  */
 649 uint32_t
 650 samr_lookup_domain_names(mlsvc_handle_t *domain_handle, char *name,
 651     smb_account_t *account)
 652 {
 653         struct samr_LookupNames arg;
 654         int                     opnum;
 655         uint32_t                status;
 656         size_t                  length;
 657 
 658         if (ndr_is_null_handle(domain_handle) ||
 659             name == NULL || account == NULL) {
 660                 return (NT_STATUS_INVALID_PARAMETER);
 661         }
 662 
 663         bzero(account, sizeof (smb_account_t));
 664         opnum = SAMR_OPNUM_LookupNames;
 665         bzero(&arg, sizeof (struct samr_LookupNames));
 666 
 667         (void) memcpy(&arg.handle, &domain_handle->handle,
 668             sizeof (samr_handle_t));
 669         arg.n_entry = 1;
 670         arg.max_n_entry = 1000;
 671         arg.index = 0;
 672         arg.total = 1;
 673 
 674         length = smb_wcequiv_strlen(name);
 675         length += sizeof (smb_wchar_t);
 676 
 677         arg.name.length = length;
 678         arg.name.allosize = length;
 679         arg.name.str = (unsigned char *)name;
 680 
 681         if (ndr_rpc_call(domain_handle, opnum, &arg) != 0) {
 682                 status = NT_STATUS_INVALID_PARAMETER;
 683         } else if (arg.status != NT_STATUS_SUCCESS) {
 684                 status = NT_SC_VALUE(arg.status);
 685 
 686                 /*
 687                  * Handle none-mapped status quietly.
 688                  */
 689                 if (status != NT_STATUS_NONE_MAPPED)
 690                         ndr_rpc_status(domain_handle, opnum, arg.status);
 691         } else {
 692                 account->a_type = arg.rid_types.rid_type[0];
 693                 account->a_rid = arg.rids.rid[0];
 694                 status = NT_STATUS_SUCCESS;
 695         }
 696 
 697         ndr_rpc_release(domain_handle);
 698         return (status);
 699 }
 700 
 701 /*
 702  * samr_query_user_info
 703  *
 704  * Query information on a specific user. The handle must be a valid
 705  * user handle obtained via samr_open_user.
 706  *
 707  * Returns 0 on success, otherwise returns NT status code.
 708  */
 709 DWORD
 710 samr_query_user_info(mlsvc_handle_t *user_handle, WORD switch_value,
 711     union samr_user_info *user_info)
 712 {
 713         struct samr_QueryUserInfo       arg;
 714         int     opnum;
 715 
 716         if (ndr_is_null_handle(user_handle) || user_info == 0)
 717                 return (NT_STATUS_INTERNAL_ERROR);
 718 
 719         opnum = SAMR_OPNUM_QueryUserInfo;
 720         bzero(&arg, sizeof (struct samr_QueryUserInfo));
 721 
 722         (void) memcpy(&arg.user_handle, &user_handle->handle,
 723             sizeof (samr_handle_t));
 724         arg.switch_value = switch_value;
 725 
 726         if (ndr_rpc_call(user_handle, opnum, &arg) != 0)
 727                 arg.status = RPC_NT_CALL_FAILED;
 728 
 729         if (arg.status == 0)
 730                 (void) samr_setup_user_info(switch_value, &arg, user_info);
 731 
 732         return (arg.status);
 733 }
 734 
 735 /*
 736  * samr_setup_user_info
 737  *
 738  * Private function to set up the samr_user_info data. Dependent on
 739  * the switch value this function may use strdup which will malloc
 740  * memory. The caller is responsible for deallocating this memory.
 741  *
 742  * Returns 0 on success, otherwise returns -1.
 743  */
 744 static int
 745 samr_setup_user_info(WORD switch_value,
 746     struct samr_QueryUserInfo *arg, union samr_user_info *user_info)
 747 {
 748         struct samr_QueryUserInfo1      *info1;
 749         struct samr_QueryUserInfo6      *info6;
 750 
 751         switch (switch_value) {
 752         case 1:
 753                 info1 = &arg->ru.info1;
 754                 user_info->info1.username = strdup(
 755                     (char const *)info1->username.str);
 756                 user_info->info1.fullname = strdup(
 757                     (char const *)info1->fullname.str);
 758                 user_info->info1.description = strdup(
 759                     (char const *)info1->description.str);
 760                 user_info->info1.unknown = 0;
 761                 user_info->info1.group_rid = info1->group_rid;
 762                 return (0);
 763 
 764         case 6:
 765                 info6 = &arg->ru.info6;
 766                 user_info->info6.username = strdup(
 767                     (char const *)info6->username.str);
 768                 user_info->info6.fullname = strdup(
 769                     (char const *)info6->fullname.str);
 770                 return (0);
 771 
 772         case 7:
 773                 user_info->info7.username = strdup(
 774                     (char const *)arg->ru.info7.username.str);
 775                 return (0);
 776 
 777         case 8:
 778                 user_info->info8.fullname = strdup(
 779                     (char const *)arg->ru.info8.fullname.str);
 780                 return (0);
 781 
 782         case 9:
 783                 user_info->info9.group_rid = arg->ru.info9.group_rid;
 784                 return (0);
 785 
 786         case 16:
 787                 user_info->info16.acct_ctrl =
 788                     arg->ru.info16.UserAccountControl;
 789                 return (0);
 790 
 791         default:
 792                 break;
 793         };
 794 
 795         return (-1);
 796 }
 797 
 798 /*
 799  * samr_query_user_groups
 800  *
 801  * Query the groups for a specific user. The handle must be a valid
 802  * user handle obtained via samr_open_user. The list of groups is
 803  * returned in group_info. Note that group_info->groups is allocated
 804  * using malloc. The caller is responsible for deallocating this
 805  * memory when it is no longer required. If group_info->n_entry is 0
 806  * then no memory was allocated.
 807  *
 808  * Returns 0 on success, otherwise returns -1.
 809  */
 810 int
 811 samr_query_user_groups(mlsvc_handle_t *user_handle, int *n_groups,
 812     struct samr_UserGroups **groups)
 813 {
 814         struct samr_QueryUserGroups arg;
 815         int     opnum;
 816         int     rc;
 817         int     nbytes;
 818 
 819         if (ndr_is_null_handle(user_handle))
 820                 return (-1);
 821 
 822         opnum = SAMR_OPNUM_QueryUserGroups;
 823         bzero(&arg, sizeof (struct samr_QueryUserGroups));
 824 
 825         (void) memcpy(&arg.user_handle, &user_handle->handle,
 826             sizeof (samr_handle_t));
 827 
 828         rc = ndr_rpc_call(user_handle, opnum, &arg);
 829         if (rc == 0) {
 830                 if (arg.info == 0) {
 831                         rc = -1;
 832                 } else {
 833                         nbytes = arg.info->n_entry *
 834                             sizeof (struct samr_UserGroups);
 835 
 836                         if ((*groups = malloc(nbytes)) == NULL) {
 837                                 *n_groups = 0;
 838                                 rc = -1;
 839                         } else {
 840                                 *n_groups = arg.info->n_entry;
 841                                 bcopy(arg.info->groups, *groups, nbytes);
 842                         }
 843                 }
 844         }
 845 
 846         ndr_rpc_release(user_handle);
 847         return (rc);
 848 }
 849 
 850 /*
 851  * samr_get_user_pwinfo
 852  *
 853  * Get some user password info. I'm not sure what this is yet but it is
 854  * part of the create user sequence. The handle must be a valid user
 855  * handle. Since I don't know what this is returning, I haven't provided
 856  * any return data yet.
 857  *
 858  * Returns 0 on success. Otherwise returns an NT status code.
 859  */
 860 DWORD
 861 samr_get_user_pwinfo(mlsvc_handle_t *user_handle)
 862 {
 863         struct samr_GetUserPwInfo arg;
 864         int     opnum;
 865         DWORD   status;
 866 
 867         if (ndr_is_null_handle(user_handle))
 868                 return (NT_STATUS_INVALID_PARAMETER);
 869 
 870         opnum = SAMR_OPNUM_GetUserPwInfo;
 871         bzero(&arg, sizeof (struct samr_GetUserPwInfo));
 872         (void) memcpy(&arg.user_handle, &user_handle->handle,
 873             sizeof (samr_handle_t));
 874 
 875         if (ndr_rpc_call(user_handle, opnum, &arg) != 0) {
 876                 status = NT_STATUS_INVALID_PARAMETER;
 877         } else if (arg.status != 0) {
 878                 ndr_rpc_status(user_handle, opnum, arg.status);
 879                 status = NT_SC_VALUE(arg.status);
 880         } else {
 881                 status = 0;
 882         }
 883 
 884         ndr_rpc_release(user_handle);
 885         return (status);
 886 }
 887 
 888 DECL_FIXUP_STRUCT(samr_SetUserInfo_u);
 889 DECL_FIXUP_STRUCT(samr_SetUserInfo_s);
 890 DECL_FIXUP_STRUCT(samr_SetUserInfo);
 891 
 892 /*
 893  * samr_set_user_info
 894  *
 895  * Returns 0 on success. Otherwise returns an NT status code.
 896  * NT status codes observed so far:
 897  *      NT_STATUS_WRONG_PASSWORD
 898  */
 899 DWORD
 900 samr_set_user_info(
 901         mlsvc_handle_t *user_handle,
 902         int info_level,
 903         void *info_buf)
 904 {
 905         struct samr_SetUserInfo arg;
 906         uint16_t usize, tsize;
 907         int opnum;
 908 
 909         if (ndr_is_null_handle(user_handle))
 910                 return (NT_STATUS_INTERNAL_ERROR);
 911 
 912         /*
 913          * Only support a few levels
 914          * MS-SAMR: UserInternal4Information
 915          */
 916         switch (info_level) {
 917         case 16: /* samr_SetUserInfo16 */
 918                 usize = sizeof (struct samr_SetUserInfo16);
 919                 break;
 920         case 21: /* samr_SetUserInfo21 */
 921                 usize = sizeof (struct samr_SetUserInfo21);
 922                 break;
 923         case 23: /* samr_SetUserInfo23 */
 924                 usize = sizeof (struct samr_SetUserInfo23);
 925                 break;
 926         case 24: /* samr_SetUserInfo24 */
 927                 usize = sizeof (struct samr_SetUserInfo24);
 928                 break;
 929         default:
 930                 return (NT_STATUS_INVALID_LEVEL);
 931         }
 932 
 933         /*
 934          * OK, now this gets really ugly, because
 935          * ndrgen doesn't do unions correctly.
 936          */
 937         FIXUP_PDU_SIZE(samr_SetUserInfo_u, usize);
 938         tsize = usize + (2 * sizeof (WORD));
 939         FIXUP_PDU_SIZE(samr_SetUserInfo_s, tsize);
 940         tsize += sizeof (ndr_request_hdr_t) + sizeof (DWORD);
 941         FIXUP_PDU_SIZE(samr_SetUserInfo, tsize);
 942 
 943         opnum = SAMR_OPNUM_SetUserInfo;
 944         bzero(&arg, sizeof (arg));
 945         (void) memcpy(&arg.user_handle, &user_handle->handle,
 946             sizeof (samr_handle_t));
 947         arg.info.info_level = info_level;
 948         arg.info.switch_value = info_level;
 949         (void) memcpy(&arg.info.ru, info_buf, usize);
 950 
 951         if (ndr_rpc_call(user_handle, opnum, &arg) != 0)
 952                 arg.status = RPC_NT_CALL_FAILED;
 953         else if (arg.status != 0)
 954                 ndr_rpc_status(user_handle, opnum, arg.status);
 955 
 956         ndr_rpc_release(user_handle);
 957         return (arg.status);
 958 }
 959 
 960 /*
 961  * Client side wrapper for SamrUnicodeChangePasswordUser2
 962  * [MS-SAMR 3.1.5.10.3]
 963  */
 964 
 965 DWORD
 966 samr_change_password(
 967         mlsvc_handle_t *handle,
 968         char *server,
 969         char *account,
 970         struct samr_encr_passwd *newpw,
 971         struct samr_encr_hash *oldpw)
 972 {
 973         static struct samr_encr_passwd zero_newpw;
 974         static struct samr_encr_hash zero_oldpw;
 975         struct samr_ChangePasswordUser2 arg;
 976         int opnum = SAMR_OPNUM_ChangePasswordUser2;
 977         char *slashserver;
 978         int len;
 979 
 980         (void) memset(&arg, 0, sizeof (arg));
 981 
 982         /* Need server name with slashes */
 983         len = 2 + strlen(server) + 1;
 984         slashserver = ndr_rpc_malloc(handle, len);
 985         if (slashserver == NULL)
 986                 return (NT_STATUS_NO_MEMORY);
 987         (void) snprintf(slashserver, len, "\\\\%s", server);
 988 
 989         arg.servername = ndr_rpc_malloc(handle, sizeof (samr_string_t));
 990         if (arg.servername == NULL)
 991                 return (NT_STATUS_NO_MEMORY);
 992         len = smb_wcequiv_strlen(slashserver);
 993         if (len < 1)
 994                 return (NT_STATUS_INVALID_PARAMETER);
 995         len += 2;       /* the WC null */
 996         arg.servername->length = len;
 997         arg.servername->allosize = len;
 998         arg.servername->str = (uint8_t *)slashserver;
 999 
1000         arg.username = ndr_rpc_malloc(handle, sizeof (samr_string_t));
1001         if (arg.username == NULL)
1002                 return (NT_STATUS_NO_MEMORY);
1003         len = smb_wcequiv_strlen(account);
1004         if (len < 1)
1005                 return (NT_STATUS_INVALID_PARAMETER);
1006         len += 2;       /* the WC null */
1007         arg.username->length = len;
1008         arg.username->allosize = len;
1009         arg.username->str = (uint8_t *)account;
1010 
1011         arg.nt_newpw = newpw;
1012         arg.nt_oldpw = oldpw;
1013 
1014         arg.lm_newpw = &zero_newpw;
1015         arg.lm_oldpw = &zero_oldpw;
1016 
1017         if (ndr_rpc_call(handle, opnum, &arg) != 0)
1018                 arg.status = RPC_NT_CALL_FAILED;
1019         else if (arg.status != 0)
1020                 ndr_rpc_status(handle, opnum, arg.status);
1021 
1022         ndr_rpc_release(handle);
1023         return (arg.status);
1024 }