2 Unix SMB/Netbios implementation.
4 NT Domain Authentication SMB / MSRPC client
5 Copyright (C) Andrew Tridgell 1994-1997
6 Copyright (C) Luke Kenneth Casson Leighton 1996-1997
8 This program is free software; you can redistribute it and/or modify
9 it under the terms of the GNU General Public License as published by
10 the Free Software Foundation; either version 2 of the License, or
11 (at your option) any later version.
13 This program is distributed in the hope that it will be useful,
14 but WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 GNU General Public License for more details.
18 You should have received a copy of the GNU General Public License
19 along with this program; if not, write to the Free Software
20 Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
32 extern int DEBUGLEVEL;
36 /****************************************************************************
37 do a SAMR query user groups
38 ****************************************************************************/
39 BOOL get_samr_query_usergroups(struct cli_state *cli,
40 POLICY_HND *pol_open_domain, uint32 user_rid,
41 uint32 *num_groups, DOM_GID *gid)
43 POLICY_HND pol_open_user;
44 if (pol_open_domain == NULL || num_groups == NULL || gid == NULL) return False;
46 /* send open domain (on user sid) */
47 if (!do_samr_open_user(cli,
55 /* send user groups query */
56 if (!do_samr_query_usergroups(cli,
60 DEBUG(5,("do_samr_query_usergroups: error in query user groups\n"));
63 return do_samr_close(cli, &pol_open_user);
66 /****************************************************************************
67 do a SAMR query user info
68 ****************************************************************************/
69 BOOL get_samr_query_userinfo(struct cli_state *cli,
70 POLICY_HND *pol_open_domain,
72 uint32 user_rid, SAM_USER_INFO_21 *usr)
74 POLICY_HND pol_open_user;
75 if (pol_open_domain == NULL || usr == NULL) return False;
77 bzero(usr, sizeof(*usr));
79 /* send open domain (on user sid) */
80 if (!do_samr_open_user(cli,
88 /* send user info query */
89 if (!do_samr_query_userinfo(cli,
91 info_level, (void*)usr))
93 DEBUG(5,("do_samr_query_userinfo: error in query user info, level 0x%x\n",
97 return do_samr_close(cli, &pol_open_user);
100 /****************************************************************************
101 do a SAMR change user password command
102 ****************************************************************************/
103 BOOL do_samr_chgpasswd_user(struct cli_state *cli,
104 char *srv_name, char *user_name,
105 char nt_newpass[516], uchar nt_oldhash[16],
106 char lm_newpass[516], uchar lm_oldhash[16])
111 SAMR_Q_CHGPASSWD_USER q_e;
112 BOOL valid_pwc = False;
114 /* create and send a MSRPC command with api SAMR_CHGPASSWD_USER */
116 prs_init(&data , 1024, 4, SAFETY_MARGIN, False);
117 prs_init(&rdata, 0 , 4, SAFETY_MARGIN, True );
119 DEBUG(4,("SAMR Change User Password. server:%s username:%s\n",
120 srv_name, user_name));
122 make_samr_q_chgpasswd_user(&q_e, srv_name, user_name,
123 nt_newpass, nt_oldhash,
124 lm_newpass, lm_oldhash);
126 /* turn parameters into data stream */
127 samr_io_q_chgpasswd_user("", &q_e, &data, 0);
129 /* send the data on \PIPE\ */
130 if (rpc_api_pipe_req(cli, SAMR_CHGPASSWD_USER, &data, &rdata))
132 SAMR_R_CHGPASSWD_USER r_e;
135 samr_io_r_chgpasswd_user("", &r_e, &rdata, 0);
137 p = rdata.offset != 0;
138 if (p && r_e.status != 0)
140 /* report error code */
141 DEBUG(0,("SAMR_R_CHGPASSWD_USER: %s\n", get_nt_error_msg(r_e.status)));
151 prs_mem_free(&data );
152 prs_mem_free(&rdata );
157 /****************************************************************************
158 do a SAMR unknown 0x38 command
159 ****************************************************************************/
160 BOOL do_samr_unknown_38(struct cli_state *cli, char *srv_name)
165 SAMR_Q_UNKNOWN_38 q_e;
166 BOOL valid_un8 = False;
168 /* create and send a MSRPC command with api SAMR_ENUM_DOM_USERS */
170 prs_init(&data , 1024, 4, SAFETY_MARGIN, False);
171 prs_init(&rdata, 0 , 4, SAFETY_MARGIN, True );
173 DEBUG(4,("SAMR Unknown 38 server:%s\n", srv_name));
175 make_samr_q_unknown_38(&q_e, srv_name);
177 /* turn parameters into data stream */
178 samr_io_q_unknown_38("", &q_e, &data, 0);
180 /* send the data on \PIPE\ */
181 if (rpc_api_pipe_req(cli, SAMR_UNKNOWN_38, &data, &rdata))
183 SAMR_R_UNKNOWN_38 r_e;
186 samr_io_r_unknown_38("", &r_e, &rdata, 0);
188 p = rdata.offset != 0;
190 if (p && r_e.status != 0)
192 /* report error code */
193 DEBUG(0,("SAMR_R_UNKNOWN_38: %s\n", get_nt_error_msg(r_e.status)));
203 prs_mem_free(&data );
204 prs_mem_free(&rdata );
209 /****************************************************************************
210 do a SAMR unknown 0x8 command
211 ****************************************************************************/
212 BOOL do_samr_query_dom_info(struct cli_state *cli,
213 POLICY_HND *domain_pol, uint16 switch_value)
218 SAMR_Q_QUERY_DOMAIN_INFO q_e;
219 BOOL valid_un8 = False;
221 /* create and send a MSRPC command with api SAMR_ENUM_DOM_USERS */
223 prs_init(&data , 1024, 4, SAFETY_MARGIN, False);
224 prs_init(&rdata, 0 , 4, SAFETY_MARGIN, True );
226 DEBUG(4,("SAMR Unknown 8 switch:%d\n", switch_value));
228 if (domain_pol == NULL) return False;
230 /* store the parameters */
231 make_samr_q_query_dom_info(&q_e, domain_pol, switch_value);
233 /* turn parameters into data stream */
234 samr_io_q_query_dom_info("", &q_e, &data, 0);
236 /* send the data on \PIPE\ */
237 if (rpc_api_pipe_req(cli, SAMR_QUERY_DOMAIN_INFO, &data, &rdata))
240 SAMR_R_QUERY_DOMAIN_INFO r_e;
243 samr_io_r_query_dom_info("", &r_e, &rdata, 0);
245 p = rdata.offset != 0;
246 if (p && r_e.status != 0)
248 /* report error code */
249 DEBUG(0,("SAMR_R_QUERY_DOMAIN_INFO: %s\n", get_nt_error_msg(r_e.status)));
260 prs_mem_free(&data );
261 prs_mem_free(&rdata );
266 /****************************************************************************
267 do a SAMR enumerate users
268 ****************************************************************************/
269 BOOL do_samr_enum_dom_users(struct cli_state *cli,
270 POLICY_HND *pol, uint16 num_entries, uint16 unk_0,
271 uint16 acb_mask, uint16 unk_1, uint32 size,
272 struct acct_info **sam,
278 SAMR_Q_ENUM_DOM_USERS q_e;
279 BOOL valid_pol = False;
281 /* create and send a MSRPC command with api SAMR_ENUM_DOM_USERS */
283 prs_init(&data , 1024, 4, SAFETY_MARGIN, False);
284 prs_init(&rdata, 0 , 4, SAFETY_MARGIN, True );
286 DEBUG(4,("SAMR Enum SAM DB max size:%x\n", size));
288 if (pol == NULL || num_sam_users == NULL) return False;
290 /* store the parameters */
291 make_samr_q_enum_dom_users(&q_e, pol,
293 acb_mask, unk_1, size);
295 /* turn parameters into data stream */
296 samr_io_q_enum_dom_users("", &q_e, &data, 0);
298 /* send the data on \PIPE\ */
299 if (rpc_api_pipe_req(cli, SAMR_ENUM_DOM_USERS, &data, &rdata))
301 SAMR_R_ENUM_DOM_USERS r_e;
304 samr_io_r_enum_dom_users("", &r_e, &rdata, 0);
306 p = rdata.offset != 0;
307 if (p && r_e.status != 0)
309 /* report error code */
310 DEBUG(0,("SAMR_R_ENUM_DOM_USERS: %s\n", get_nt_error_msg(r_e.status)));
319 *num_sam_users = r_e.num_entries2;
320 if (*num_sam_users > MAX_SAM_ENTRIES)
322 *num_sam_users = MAX_SAM_ENTRIES;
323 DEBUG(2,("do_samr_enum_dom_users: sam user entries limited to %d\n",
327 *sam = (struct acct_info*) malloc(sizeof(struct acct_info) * (*num_sam_users));
334 for (i = 0; i < *num_sam_users; i++)
337 (*sam)[i].user_rid = r_e.sam[i].rid;
338 if (r_e.sam[i].hdr_name.buffer)
340 char *acct_name = unistrn2(r_e.uni_acct_name[name_idx].buffer,
341 r_e.uni_acct_name[name_idx].uni_str_len);
342 fstrcpy((*sam)[i].acct_name, acct_name);
347 bzero((*sam)[i].acct_name, sizeof((*sam)[i].acct_name));
349 DEBUG(5,("do_samr_enum_dom_users: idx: %4d rid: %8x acct: %s\n",
350 i, (*sam)[i].user_rid, (*sam)[i].acct_name));
356 prs_mem_free(&data );
357 prs_mem_free(&rdata );
362 /****************************************************************************
364 ****************************************************************************/
365 BOOL do_samr_connect(struct cli_state *cli,
366 char *srv_name, uint32 unknown_0,
367 POLICY_HND *connect_pol)
373 BOOL valid_pol = False;
375 /* create and send a MSRPC command with api SAMR_CONNECT */
377 prs_init(&data , 1024, 4, SAFETY_MARGIN, False);
378 prs_init(&rdata, 0 , 4, SAFETY_MARGIN, True );
380 DEBUG(4,("SAMR Open Policy server:%s undoc value:%x\n",
381 srv_name, unknown_0));
383 if (srv_name == NULL || connect_pol == NULL) return False;
385 /* store the parameters */
386 make_samr_q_connect(&q_o, srv_name, unknown_0);
388 /* turn parameters into data stream */
389 samr_io_q_connect("", &q_o, &data, 0);
391 /* send the data on \PIPE\ */
392 if (rpc_api_pipe_req(cli, SAMR_CONNECT, &data, &rdata))
397 samr_io_r_connect("", &r_o, &rdata, 0);
398 p = rdata.offset != 0;
400 if (p && r_o.status != 0)
402 /* report error code */
403 DEBUG(0,("SAMR_R_CONNECT: %s\n", get_nt_error_msg(r_o.status)));
409 memcpy(connect_pol, &r_o.connect_pol, sizeof(r_o.connect_pol));
414 prs_mem_free(&data );
415 prs_mem_free(&rdata );
420 /****************************************************************************
422 ****************************************************************************/
423 BOOL do_samr_open_user(struct cli_state *cli,
424 POLICY_HND *pol, uint32 unk_0, uint32 rid,
425 POLICY_HND *user_pol)
430 SAMR_Q_OPEN_USER q_o;
431 BOOL valid_pol = False;
433 /* create and send a MSRPC command with api SAMR_OPEN_USER */
435 prs_init(&data , 1024, 4, SAFETY_MARGIN, False);
436 prs_init(&rdata, 0 , 4, SAFETY_MARGIN, True );
438 DEBUG(4,("SAMR Open User. unk_0: %08x RID:%x\n",
441 if (pol == NULL || user_pol == NULL) return False;
443 /* store the parameters */
444 make_samr_q_open_user(&q_o, pol, unk_0, rid);
446 /* turn parameters into data stream */
447 samr_io_q_open_user("", &q_o, &data, 0);
449 /* send the data on \PIPE\ */
450 if (rpc_api_pipe_req(cli, SAMR_OPEN_USER, &data, &rdata))
452 SAMR_R_OPEN_USER r_o;
455 samr_io_r_open_user("", &r_o, &rdata, 0);
456 p = rdata.offset != 0;
458 if (p && r_o.status != 0)
460 /* report error code */
461 DEBUG(0,("SAMR_R_OPEN_USER: %s\n", get_nt_error_msg(r_o.status)));
467 memcpy(user_pol, &r_o.user_pol, sizeof(r_o.user_pol));
472 prs_mem_free(&data );
473 prs_mem_free(&rdata );
478 /****************************************************************************
479 do a SAMR Open Domain
480 ****************************************************************************/
481 BOOL do_samr_open_domain(struct cli_state *cli,
482 POLICY_HND *connect_pol, uint32 rid, DOM_SID *sid,
483 POLICY_HND *domain_pol)
489 SAMR_Q_OPEN_DOMAIN q_o;
490 BOOL valid_pol = False;
492 /* create and send a MSRPC command with api SAMR_OPEN_DOMAIN */
494 prs_init(&data , 1024, 4, SAFETY_MARGIN, False);
495 prs_init(&rdata, 0 , 4, SAFETY_MARGIN, True );
497 sid_to_string(sid_str, sid);
498 DEBUG(4,("SAMR Open Domain. SID:%s RID:%x\n", sid_str, rid));
500 if (connect_pol == NULL || sid == NULL || domain_pol == NULL) return False;
502 /* store the parameters */
503 make_samr_q_open_domain(&q_o, connect_pol, rid, sid);
505 /* turn parameters into data stream */
506 samr_io_q_open_domain("", &q_o, &data, 0);
508 /* send the data on \PIPE\ */
509 if (rpc_api_pipe_req(cli, SAMR_OPEN_DOMAIN, &data, &rdata))
511 SAMR_R_OPEN_DOMAIN r_o;
514 samr_io_r_open_domain("", &r_o, &rdata, 0);
515 p = rdata.offset != 0;
517 if (p && r_o.status != 0)
519 /* report error code */
520 DEBUG(0,("SAMR_R_OPEN_DOMAIN: %s\n", get_nt_error_msg(r_o.status)));
526 memcpy(domain_pol, &r_o.domain_pol, sizeof(r_o.domain_pol));
531 prs_mem_free(&data );
532 prs_mem_free(&rdata );
537 /****************************************************************************
538 do a SAMR Query Unknown 12
539 ****************************************************************************/
540 BOOL do_samr_query_unknown_12(struct cli_state *cli,
541 POLICY_HND *pol, uint32 rid, uint32 num_gids, uint32 *gids,
543 fstring als_names [MAX_LOOKUP_SIDS],
544 uint32 num_als_users[MAX_LOOKUP_SIDS])
549 SAMR_Q_UNKNOWN_12 q_o;
550 BOOL valid_query = False;
552 /* create and send a MSRPC command with api SAMR_UNKNOWN_12 */
554 prs_init(&data , 1024, 4, SAFETY_MARGIN, False);
555 prs_init(&rdata, 0 , 4, SAFETY_MARGIN, True );
557 DEBUG(4,("SAMR Query Unknown 12.\n"));
559 if (pol == NULL || rid == 0 || num_gids == 0 || gids == NULL ||
560 num_aliases == NULL || als_names == NULL || num_als_users == NULL ) return False;
562 /* store the parameters */
563 make_samr_q_unknown_12(&q_o, pol, rid, num_gids, gids);
565 /* turn parameters into data stream */
566 samr_io_q_unknown_12("", &q_o, &data, 0);
568 /* send the data on \PIPE\ */
569 if (rpc_api_pipe_req(cli, SAMR_UNKNOWN_12, &data, &rdata))
571 SAMR_R_UNKNOWN_12 r_o;
574 samr_io_r_unknown_12("", &r_o, &rdata, 0);
575 p = rdata.offset != 0;
577 if (p && r_o.status != 0)
579 /* report error code */
580 DEBUG(0,("SAMR_R_UNKNOWN_12: %s\n", get_nt_error_msg(r_o.status)));
586 if (r_o.ptr_aliases != 0 && r_o.ptr_als_usrs != 0 &&
587 r_o.num_als_usrs1 == r_o.num_aliases1)
592 *num_aliases = r_o.num_aliases1;
594 for (i = 0; i < r_o.num_aliases1; i++)
596 fstrcpy(als_names[i], unistrn2(r_o.uni_als_name[i].buffer, r_o.uni_als_name[i].uni_str_len));
598 for (i = 0; i < r_o.num_als_usrs1; i++)
600 num_als_users[i] = r_o.num_als_usrs[i];
603 else if (r_o.ptr_aliases == 0 && r_o.ptr_als_usrs == 0)
615 prs_mem_free(&data );
616 prs_mem_free(&rdata );
621 /****************************************************************************
622 do a SAMR Query User Aliases
623 ****************************************************************************/
624 BOOL do_samr_query_useraliases(struct cli_state *cli,
625 POLICY_HND *pol, DOM_SID *sid,
626 uint32 *num_aliases, uint32 *rid)
631 SAMR_Q_QUERY_USERALIASES q_o;
632 BOOL valid_query = False;
634 /* create and send a MSRPC command with api SAMR_QUERY_USERALIASES */
636 prs_init(&data , 1024, 4, SAFETY_MARGIN, False);
637 prs_init(&rdata, 0 , 4, SAFETY_MARGIN, True );
639 DEBUG(4,("SAMR Query User Aliases.\n"));
641 if (pol == NULL || sid == NULL || rid == NULL || num_aliases == 0) return False;
643 /* store the parameters */
644 make_samr_q_query_useraliases(&q_o, pol, sid);
646 /* turn parameters into data stream */
647 samr_io_q_query_useraliases("", &q_o, &data, 0);
649 /* send the data on \PIPE\ */
650 if (rpc_api_pipe_req(cli, SAMR_QUERY_USERALIASES, &data, &rdata))
652 SAMR_R_QUERY_USERALIASES r_o;
658 samr_io_r_query_useraliases("", &r_o, &rdata, 0);
659 p = rdata.offset != 0;
661 if (p && r_o.status != 0)
663 /* report error code */
664 DEBUG(0,("SAMR_R_QUERY_USERALIASES: %s\n", get_nt_error_msg(r_o.status)));
668 if (p && r_o.ptr != 0)
671 *num_aliases = r_o.num_entries;
676 prs_mem_free(&data );
677 prs_mem_free(&rdata );
682 /****************************************************************************
683 do a SAMR Query User Groups
684 ****************************************************************************/
685 BOOL do_samr_query_usergroups(struct cli_state *cli,
686 POLICY_HND *pol, uint32 *num_groups, DOM_GID *gid)
691 SAMR_Q_QUERY_USERGROUPS q_o;
692 BOOL valid_query = False;
694 /* create and send a MSRPC command with api SAMR_QUERY_USERGROUPS */
696 prs_init(&data , 1024, 4, SAFETY_MARGIN, False);
697 prs_init(&rdata, 0 , 4, SAFETY_MARGIN, True );
699 DEBUG(4,("SAMR Query User Groups.\n"));
701 if (pol == NULL || gid == NULL || num_groups == 0) return False;
703 /* store the parameters */
704 make_samr_q_query_usergroups(&q_o, pol);
706 /* turn parameters into data stream */
707 samr_io_q_query_usergroups("", &q_o, &data, 0);
709 /* send the data on \PIPE\ */
710 if (rpc_api_pipe_req(cli, SAMR_QUERY_USERGROUPS, &data, &rdata))
712 SAMR_R_QUERY_USERGROUPS r_o;
718 samr_io_r_query_usergroups("", &r_o, &rdata, 0);
719 p = rdata.offset != 0;
721 if (p && r_o.status != 0)
723 /* report error code */
724 DEBUG(0,("SAMR_R_QUERY_USERGROUPS: %s\n", get_nt_error_msg(r_o.status)));
728 if (p && r_o.ptr_0 != 0)
731 *num_groups = r_o.num_entries;
736 prs_mem_free(&data );
737 prs_mem_free(&rdata );
742 /****************************************************************************
743 do a SAMR Query User Info
744 ****************************************************************************/
745 BOOL do_samr_query_userinfo(struct cli_state *cli,
746 POLICY_HND *pol, uint16 switch_value, void* usr)
751 SAMR_Q_QUERY_USERINFO q_o;
752 BOOL valid_query = False;
754 /* create and send a MSRPC command with api SAMR_QUERY_USERINFO */
756 prs_init(&data , 1024, 4, SAFETY_MARGIN, False);
757 prs_init(&rdata, 0 , 4, SAFETY_MARGIN, True );
759 DEBUG(4,("SAMR Query User Info. level: %d\n", switch_value));
761 if (pol == NULL || usr == NULL || switch_value == 0) return False;
763 /* store the parameters */
764 make_samr_q_query_userinfo(&q_o, pol, switch_value);
766 /* turn parameters into data stream */
767 samr_io_q_query_userinfo("", &q_o, &data, 0);
769 /* send the data on \PIPE\ */
770 if (rpc_api_pipe_req(cli, SAMR_QUERY_USERINFO, &data, &rdata))
772 SAMR_R_QUERY_USERINFO r_o;
778 samr_io_r_query_userinfo("", &r_o, &rdata, 0);
779 p = rdata.offset != 0;
781 if (p && r_o.status != 0)
783 /* report error code */
784 DEBUG(0,("SAMR_R_QUERY_USERINFO: %s\n", get_nt_error_msg(r_o.status)));
788 if (p && r_o.switch_value != switch_value)
790 DEBUG(0,("SAMR_R_QUERY_USERINFO: received incorrect level %d\n",
794 if (p && r_o.ptr != 0)
800 prs_mem_free(&data );
801 prs_mem_free(&rdata );
806 /****************************************************************************
808 ****************************************************************************/
809 BOOL do_samr_close(struct cli_state *cli, POLICY_HND *hnd)
814 SAMR_Q_CLOSE_HND q_c;
815 BOOL valid_close = False;
817 prs_init(&data , 1024, 4, SAFETY_MARGIN, False);
818 prs_init(&rdata, 0 , 4, SAFETY_MARGIN, True );
820 if (hnd == NULL) return False;
822 /* create and send a MSRPC command with api SAMR_CLOSE_HND */
824 DEBUG(4,("SAMR Close\n"));
826 /* store the parameters */
827 make_samr_q_close_hnd(&q_c, hnd);
829 /* turn parameters into data stream */
830 samr_io_q_close_hnd("", &q_c, &data, 0);
832 /* send the data on \PIPE\ */
833 if (rpc_api_pipe_req(cli, SAMR_CLOSE_HND, &data, &rdata))
835 SAMR_R_CLOSE_HND r_c;
838 samr_io_r_close_hnd("", &r_c, &rdata, 0);
839 p = rdata.offset != 0;
841 if (p && r_c.status != 0)
843 /* report error code */
844 DEBUG(0,("SAMR_CLOSE_HND: %s\n", get_nt_error_msg(r_c.status)));
850 /* check that the returned policy handle is all zeros */
854 for (i = 0; i < sizeof(r_c.pol.data); i++)
856 if (r_c.pol.data[i] != 0)
864 DEBUG(0,("SAMR_CLOSE_HND: non-zero handle returned\n"));
869 prs_mem_free(&data );
870 prs_mem_free(&rdata );