2 * Unix SMB/CIFS implementation.
3 * RPC Pipe client / server routines
4 * Copyright (C) Andrew Tridgell 1992-1997,
5 * Copyright (C) Luke Kenneth Casson Leighton 1996-1997,
6 * Copyright (C) Jeremy Allison 2001.
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 3 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, see <http://www.gnu.org/licenses/>.
23 #include "system/passwd.h" /* uid_wrapper */
24 #include "../librpc/gen_ndr/ndr_lsa.h"
25 #include "../librpc/gen_ndr/ndr_samr.h"
27 #include "rpc_server/rpc_pipes.h"
28 #include "../libcli/security/security.h"
29 #include "lib/tsocket/tsocket.h"
30 #include "librpc/ndr/ndr_table.h"
33 #define DBGC_CLASS DBGC_RPC_SRV
35 static struct pipes_struct *InternalPipes;
38 * the following prototypes are declared here to avoid
39 * code being moved about too much for a patch to be
40 * disrupted / less obvious.
42 * these functions, and associated functions that they
43 * call, should be moved behind a .so module-loading
44 * system _anyway_. so that's the next step...
47 int make_base_pipes_struct(TALLOC_CTX *mem_ctx,
48 struct messaging_context *msg_ctx,
49 const char *pipe_name,
50 enum dcerpc_transport_t transport,
52 const struct tsocket_address *remote_address,
53 const struct tsocket_address *local_address,
54 struct pipes_struct **_p)
56 struct pipes_struct *p;
58 p = talloc_zero(mem_ctx, struct pipes_struct);
63 p->mem_ctx = talloc_named(p, 0, "pipe %s %p", pipe_name, p);
70 p->transport = transport;
73 p->remote_address = tsocket_address_copy(remote_address, p);
74 if (p->remote_address == NULL) {
80 p->local_address = tsocket_address_copy(local_address, p);
81 if (p->local_address == NULL) {
87 DLIST_ADD(InternalPipes, p);
88 talloc_set_destructor(p, close_internal_rpc_pipe_hnd);
95 bool check_open_pipes(void)
97 struct pipes_struct *p;
99 for (p = InternalPipes; p != NULL; p = p->next) {
100 if (num_pipe_handles(p) != 0) {
107 /****************************************************************************
109 ****************************************************************************/
111 int close_internal_rpc_pipe_hnd(struct pipes_struct *p)
114 DEBUG(0,("Invalid pipe in close_internal_rpc_pipe_hnd\n"));
118 /* Free the handles database. */
119 close_policy_by_pipe(p);
121 DLIST_REMOVE(InternalPipes, p);
127 * Handle database - stored per pipe.
130 struct dcesrv_handle_old {
131 struct dcesrv_handle_old *prev, *next;
132 struct policy_handle wire_handle;
133 uint32_t access_granted;
138 struct dcesrv_handle_old *handles; /* List of pipe handles. */
139 size_t count; /* Current number of handles. */
140 size_t pipe_ref_count; /* Number of pipe handles referring
144 /* This is the max handles across all instances of a pipe name. */
145 #ifndef MAX_OPEN_POLS
146 #define MAX_OPEN_POLS 2048
149 /****************************************************************************
150 Hack as handles need to be persistent over lsa pipe closes so long as a samr
152 ****************************************************************************/
154 static bool is_samr_lsa_pipe(const struct ndr_syntax_id *syntax)
156 return (ndr_syntax_id_equal(syntax, &ndr_table_samr.syntax_id)
157 || ndr_syntax_id_equal(syntax, &ndr_table_lsarpc.syntax_id));
160 size_t num_pipe_handles(struct pipes_struct *p)
162 if (p->pipe_handles == NULL) {
165 return p->pipe_handles->count;
168 /****************************************************************************
169 Initialise a policy handle list on a pipe. Handle list is shared between all
170 pipes of the same name.
171 ****************************************************************************/
173 bool init_pipe_handles(struct pipes_struct *p, const struct ndr_syntax_id *syntax)
175 struct pipes_struct *plist;
176 struct handle_list *hl;
178 for (plist = InternalPipes; plist; plist = plist->next) {
179 struct pipe_rpc_fns *p_ctx;
182 for (p_ctx = plist->contexts;
184 p_ctx = p_ctx->next) {
185 if (ndr_syntax_id_equal(syntax, &p_ctx->syntax)) {
189 if (is_samr_lsa_pipe(&p_ctx->syntax)
190 && is_samr_lsa_pipe(syntax)) {
192 * samr and lsa share a handle space (same process
206 hl = plist->pipe_handles;
212 * First open, we have to create the handle list
214 hl = talloc_zero(NULL, struct handle_list);
219 DEBUG(10,("init_pipe_handle_list: created handle list for "
221 ndr_interface_name(&syntax->uuid,
222 syntax->if_version)));
226 * One more pipe is using this list.
229 hl->pipe_ref_count++;
232 * Point this pipe at this list.
235 p->pipe_handles = hl;
237 DEBUG(10,("init_pipe_handle_list: pipe_handles ref count = %lu for "
238 "pipe %s\n", (unsigned long)p->pipe_handles->pipe_ref_count,
239 ndr_interface_name(&syntax->uuid, syntax->if_version)));
244 /****************************************************************************
245 find first available policy slot. creates a policy handle for you.
247 If "data_ptr" is given, this must be a talloc'ed object, create_policy_hnd
248 talloc_moves this into the handle. If the policy_hnd is closed,
249 data_ptr is TALLOC_FREE()'ed
250 ****************************************************************************/
252 static struct dcesrv_handle_old *create_rpc_handle_internal(
253 struct pipes_struct *p,
254 struct policy_handle *hnd,
258 struct dcesrv_handle_old *rpc_hnd = NULL;
259 static uint32_t pol_hnd_low = 0;
260 static uint32_t pol_hnd_high = 0;
261 time_t t = time(NULL);
263 if (p->pipe_handles->count > MAX_OPEN_POLS) {
264 DEBUG(0,("create_policy_hnd: ERROR: too many handles (%d) on this pipe.\n",
265 (int)p->pipe_handles->count));
269 rpc_hnd = talloc_zero(p->pipe_handles, struct dcesrv_handle_old);
271 DEBUG(0,("create_policy_hnd: ERROR: out of memory!\n"));
275 if (data_ptr != NULL) {
276 rpc_hnd->data = talloc_move(rpc_hnd, &data_ptr);
280 if (pol_hnd_low == 0) {
284 SIVAL(&rpc_hnd->wire_handle.handle_type, 0 , handle_type);
286 /* second bit is incrementing */
287 SIVAL(&rpc_hnd->wire_handle.uuid.time_low, 0 , pol_hnd_low);
288 SSVAL(&rpc_hnd->wire_handle.uuid.time_mid, 0 , pol_hnd_high);
289 SSVAL(&rpc_hnd->wire_handle.uuid.time_hi_and_version, 0, (pol_hnd_high >> 16));
291 /* split the current time into two 16 bit values */
293 /* something random */
294 SSVAL(rpc_hnd->wire_handle.uuid.clock_seq, 0, (t >> 16));
295 /* something random */
296 SSVAL(rpc_hnd->wire_handle.uuid.node, 0, t);
297 /* something more random */
298 SIVAL(rpc_hnd->wire_handle.uuid.node, 2, getpid());
300 DLIST_ADD(p->pipe_handles->handles, rpc_hnd);
301 p->pipe_handles->count++;
303 *hnd = rpc_hnd->wire_handle;
305 DEBUG(6, ("Opened policy hnd[%d] ", (int)p->pipe_handles->count));
306 dump_data(6, (uint8_t *)hnd, sizeof(*hnd));
311 bool create_policy_hnd(struct pipes_struct *p,
312 struct policy_handle *hnd,
316 struct dcesrv_handle_old *rpc_hnd = NULL;
318 rpc_hnd = create_rpc_handle_internal(p, hnd, handle_type, data_ptr);
319 if (rpc_hnd == NULL) {
325 /****************************************************************************
326 find policy by handle - internal version.
327 ****************************************************************************/
329 static struct dcesrv_handle_old *find_policy_by_hnd_internal(struct pipes_struct *p,
330 const struct policy_handle *hnd, void **data_p)
332 struct dcesrv_handle_old *h = NULL;
340 for (h = p->pipe_handles->handles; h != NULL; h = h->next) {
341 if (memcmp(&h->wire_handle, hnd, sizeof(*hnd)) == 0) {
342 DEBUG(6,("Found policy hnd[%u] ", count));
343 dump_data(6, (const uint8_t *)hnd, sizeof(*hnd));
352 DEBUG(4,("Policy not found: "));
353 dump_data(4, (const uint8_t *)hnd, sizeof(*hnd));
355 p->fault_state = DCERPC_FAULT_CONTEXT_MISMATCH;
360 /****************************************************************************
361 find policy by handle
362 ****************************************************************************/
364 bool find_policy_by_hnd(struct pipes_struct *p, const struct policy_handle *hnd,
367 struct dcesrv_handle_old *rpc_hnd = NULL;
369 rpc_hnd = find_policy_by_hnd_internal(p, hnd, data_p);
370 if (rpc_hnd == NULL) {
376 /****************************************************************************
378 ****************************************************************************/
380 bool close_policy_hnd(struct pipes_struct *p, struct policy_handle *hnd)
382 struct dcesrv_handle_old *rpc_hnd = NULL;
384 rpc_hnd = find_policy_by_hnd_internal(p, hnd, NULL);
386 if (rpc_hnd == NULL) {
387 DEBUG(3, ("Error closing policy (policy not found)\n"));
391 DEBUG(6,("Closed policy\n"));
393 p->pipe_handles->count--;
395 DLIST_REMOVE(p->pipe_handles->handles, rpc_hnd);
396 TALLOC_FREE(rpc_hnd);
401 /****************************************************************************
402 Close a pipe - free the handle set if it was the last pipe reference.
403 ****************************************************************************/
405 void close_policy_by_pipe(struct pipes_struct *p)
407 if (p->pipe_handles == NULL) {
411 p->pipe_handles->pipe_ref_count--;
413 if (p->pipe_handles->pipe_ref_count == 0) {
415 * Last pipe open on this list - free the list.
417 TALLOC_FREE(p->pipe_handles);
419 DBG_DEBUG("Deleted handle list for RPC connection %s\n",
421 ndr_interface_name(&p->contexts->syntax.uuid,
422 p->contexts->syntax.if_version) :
427 /*******************************************************************
428 Shall we allow access to this rpc? Currently this function
429 implements the 'restrict anonymous' setting by denying access to
430 anonymous users if the restrict anonymous level is > 0. Further work
431 will be checking a security descriptor to determine whether a user
432 token has enough access to access the pipe.
433 ********************************************************************/
435 bool pipe_access_check(struct pipes_struct *p)
437 /* Don't let anonymous users access this RPC if restrict
440 if (lp_restrict_anonymous() > 0) {
442 /* schannel, so we must be ok */
444 (p->auth.auth_type == DCERPC_AUTH_TYPE_SCHANNEL)) {
448 if (security_session_user_level(p->session_info, NULL) < SECURITY_USER) {
456 void *_policy_handle_create(struct pipes_struct *p,
457 struct policy_handle *hnd,
459 uint32_t access_granted,
464 struct dcesrv_handle_old *rpc_hnd = NULL;
467 if (p->pipe_handles->count > MAX_OPEN_POLS) {
468 DEBUG(0, ("ERROR: Too many handles (%d) for RPC connection %s\n",
469 (int) p->pipe_handles->count,
470 ndr_interface_name(&p->contexts->syntax.uuid,
471 p->contexts->syntax.if_version)));
473 *pstatus = NT_STATUS_INSUFFICIENT_RESOURCES;
477 data = talloc_size(talloc_tos(), data_size);
479 *pstatus = NT_STATUS_NO_MEMORY;
482 talloc_set_name_const(data, type);
484 rpc_hnd = create_rpc_handle_internal(p, hnd, handle_type, data);
485 if (rpc_hnd == NULL) {
487 *pstatus = NT_STATUS_NO_MEMORY;
490 rpc_hnd->access_granted = access_granted;
491 *pstatus = NT_STATUS_OK;
495 void *_policy_handle_find(struct pipes_struct *p,
496 const struct policy_handle *hnd,
497 uint32_t access_required,
498 uint32_t *paccess_granted,
499 const char *name, const char *location,
502 struct dcesrv_handle_old *rpc_hnd = NULL;
505 rpc_hnd = find_policy_by_hnd_internal(p, hnd, &data);
506 if (rpc_hnd == NULL) {
507 *pstatus = NT_STATUS_INVALID_HANDLE;
510 if (strcmp(name, talloc_get_name(data)) != 0) {
511 DEBUG(10, ("expected %s, got %s\n", name,
512 talloc_get_name(data)));
513 *pstatus = NT_STATUS_INVALID_HANDLE;
516 if ((access_required & rpc_hnd->access_granted) != access_required) {
518 DEBUG(4, ("%s: ACCESS should be DENIED (granted: "
519 "%#010x; required: %#010x)\n", location,
520 rpc_hnd->access_granted, access_required));
521 DEBUGADD(4,("but overwritten by euid == 0\n"));
524 DEBUG(2,("%s: ACCESS DENIED (granted: %#010x; required: "
525 "%#010x)\n", location, rpc_hnd->access_granted,
527 *pstatus = NT_STATUS_ACCESS_DENIED;
532 DEBUG(10, ("found handle of type %s\n", talloc_get_name(data)));
533 if (paccess_granted != NULL) {
534 *paccess_granted = rpc_hnd->access_granted;
536 *pstatus = NT_STATUS_OK;