Line data Source code
1 : /*
2 : Unix SMB/CIFS implementation.
3 :
4 : dcerpc utility functions
5 :
6 : Copyright (C) Andrew Tridgell 2003
7 : Copyright (C) Jelmer Vernooij 2004
8 : Copyright (C) Andrew Bartlett <abartlet@samba.org> 2005
9 : Copyright (C) Rafal Szczesniak 2006
10 : Copyright (C) Stefan Metzmacher 2014
11 :
12 : This program is free software; you can redistribute it and/or modify
13 : it under the terms of the GNU General Public License as published by
14 : the Free Software Foundation; either version 3 of the License, or
15 : (at your option) any later version.
16 :
17 : This program is distributed in the hope that it will be useful,
18 : but WITHOUT ANY WARRANTY; without even the implied warranty of
19 : MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
20 : GNU General Public License for more details.
21 :
22 : You should have received a copy of the GNU General Public License
23 : along with this program. If not, see <http://www.gnu.org/licenses/>.
24 : */
25 :
26 : #include "includes.h"
27 : #include "../../lib/util/util_net.h"
28 : #include "librpc/gen_ndr/ndr_epmapper.h"
29 : #include "librpc/gen_ndr/ndr_misc.h"
30 : #include "librpc/rpc/dcerpc.h"
31 : #include "rpc_common.h"
32 :
33 : #undef strcasecmp
34 : #undef strncasecmp
35 :
36 : #define MAX_PROTSEQ 10
37 :
38 : struct dcerpc_binding {
39 : enum dcerpc_transport_t transport;
40 : struct GUID object;
41 : const char *object_string;
42 : const char *host;
43 : const char *target_hostname;
44 : const char *target_principal;
45 : const char *endpoint;
46 : const char **options;
47 : uint32_t flags;
48 : uint32_t assoc_group_id;
49 : char assoc_group_string[11]; /* 0x3456789a + '\0' */
50 : };
51 :
52 : static const struct {
53 : const char *name;
54 : enum dcerpc_transport_t transport;
55 : int num_protocols;
56 : enum epm_protocol protseq[MAX_PROTSEQ];
57 : } transports[] = {
58 : { "ncacn_np", NCACN_NP, 3,
59 : { EPM_PROTOCOL_NCACN, EPM_PROTOCOL_SMB, EPM_PROTOCOL_NETBIOS }},
60 : { "ncacn_ip_tcp", NCACN_IP_TCP, 3,
61 : { EPM_PROTOCOL_NCACN, EPM_PROTOCOL_TCP, EPM_PROTOCOL_IP } },
62 : { "ncacn_http", NCACN_HTTP, 3,
63 : { EPM_PROTOCOL_NCACN, EPM_PROTOCOL_HTTP, EPM_PROTOCOL_IP } },
64 : { "ncadg_ip_udp", NCACN_IP_UDP, 3,
65 : { EPM_PROTOCOL_NCADG, EPM_PROTOCOL_UDP, EPM_PROTOCOL_IP } },
66 : { "ncalrpc", NCALRPC, 2,
67 : { EPM_PROTOCOL_NCALRPC, EPM_PROTOCOL_NAMED_PIPE } },
68 : { "ncacn_unix_stream", NCACN_UNIX_STREAM, 2,
69 : { EPM_PROTOCOL_NCACN, EPM_PROTOCOL_UNIX_DS } },
70 : { "ncadg_unix_dgram", NCADG_UNIX_DGRAM, 2,
71 : { EPM_PROTOCOL_NCADG, EPM_PROTOCOL_UNIX_DS } },
72 : { "ncacn_at_dsp", NCACN_AT_DSP, 3,
73 : { EPM_PROTOCOL_NCACN, EPM_PROTOCOL_APPLETALK, EPM_PROTOCOL_DSP } },
74 : { "ncadg_at_ddp", NCADG_AT_DDP, 3,
75 : { EPM_PROTOCOL_NCADG, EPM_PROTOCOL_APPLETALK, EPM_PROTOCOL_DDP } },
76 : { "ncacn_vns_ssp", NCACN_VNS_SPP, 3,
77 : { EPM_PROTOCOL_NCACN, EPM_PROTOCOL_STREETTALK, EPM_PROTOCOL_VINES_SPP } },
78 : { "ncacn_vns_ipc", NCACN_VNS_IPC, 3,
79 : { EPM_PROTOCOL_NCACN, EPM_PROTOCOL_STREETTALK, EPM_PROTOCOL_VINES_IPC }, },
80 : { "ncadg_ipx", NCADG_IPX, 2,
81 : { EPM_PROTOCOL_NCADG, EPM_PROTOCOL_IPX },
82 : },
83 : { "ncacn_spx", NCACN_SPX, 3,
84 : /* I guess some MS programmer confused the identifier for
85 : * EPM_PROTOCOL_UUID (0x0D or 13) with the one for
86 : * EPM_PROTOCOL_SPX (0x13) here. -- jelmer*/
87 : { EPM_PROTOCOL_NCACN, EPM_PROTOCOL_NCALRPC, EPM_PROTOCOL_UUID },
88 : },
89 : };
90 :
91 : static const struct ncacn_option {
92 : const char *name;
93 : uint32_t flag;
94 : } ncacn_options[] = {
95 : {"sign", DCERPC_SIGN},
96 : {"seal", DCERPC_SEAL},
97 : {"connect", DCERPC_CONNECT},
98 : {"spnego", DCERPC_AUTH_SPNEGO},
99 : {"ntlm", DCERPC_AUTH_NTLM},
100 : {"krb5", DCERPC_AUTH_KRB5},
101 : {"schannel", DCERPC_SCHANNEL | DCERPC_SCHANNEL_AUTO},
102 : {"validate", DCERPC_DEBUG_VALIDATE_BOTH},
103 : {"print", DCERPC_DEBUG_PRINT_BOTH},
104 : {"padcheck", DCERPC_DEBUG_PAD_CHECK},
105 : {"bigendian", DCERPC_PUSH_BIGENDIAN},
106 : {"smb1", DCERPC_SMB1},
107 : {"smb2", DCERPC_SMB2},
108 : {"ndr64", DCERPC_NDR64},
109 : {"packet", DCERPC_PACKET},
110 : };
111 :
112 1074984 : static const struct ncacn_option *ncacn_option_by_name(const char *name)
113 : {
114 17389 : size_t i;
115 :
116 16972231 : for (i=0; i<ARRAY_SIZE(ncacn_options); i++) {
117 255673 : int ret;
118 :
119 15917973 : ret = strcasecmp(ncacn_options[i].name, name);
120 15917973 : if (ret != 0) {
121 15897247 : continue;
122 : }
123 :
124 20726 : return &ncacn_options[i];
125 : }
126 :
127 1037591 : return NULL;
128 : }
129 :
130 4392 : const char *epm_floor_string(TALLOC_CTX *mem_ctx, struct epm_floor *epm_floor)
131 : {
132 0 : struct ndr_syntax_id syntax;
133 0 : NTSTATUS status;
134 :
135 4392 : switch(epm_floor->lhs.protocol) {
136 1780 : case EPM_PROTOCOL_UUID:
137 1780 : status = dcerpc_floor_get_uuid_full(epm_floor, &syntax);
138 1780 : if (NT_STATUS_IS_OK(status)) {
139 : /* lhs is used: UUID */
140 0 : struct GUID_txt_buf buf;
141 :
142 1780 : if (GUID_equal(&syntax.uuid, &ndr_transfer_syntax_ndr.uuid)) {
143 893 : return "NDR";
144 : }
145 :
146 887 : if (GUID_equal(&syntax.uuid, &ndr_transfer_syntax_ndr64.uuid)) {
147 0 : return "NDR64";
148 : }
149 :
150 887 : return talloc_asprintf(
151 : mem_ctx,
152 : " uuid %s/0x%02x",
153 : GUID_buf_string(&syntax.uuid, &buf),
154 : syntax.if_version);
155 : } else { /* IPX */
156 0 : return talloc_asprintf(mem_ctx, "IPX:%s",
157 0 : data_blob_hex_string_upper(mem_ctx, &epm_floor->rhs.uuid.unknown));
158 : }
159 :
160 832 : case EPM_PROTOCOL_NCACN:
161 832 : return "RPC-C";
162 :
163 0 : case EPM_PROTOCOL_NCADG:
164 0 : return "RPC";
165 :
166 58 : case EPM_PROTOCOL_NCALRPC:
167 58 : return "NCALRPC";
168 :
169 0 : case EPM_PROTOCOL_DNET_NSP:
170 0 : return "DNET/NSP";
171 :
172 385 : case EPM_PROTOCOL_IP:
173 385 : return talloc_asprintf(mem_ctx, "IP:%s", epm_floor->rhs.ip.ipaddr);
174 :
175 58 : case EPM_PROTOCOL_NAMED_PIPE:
176 58 : return talloc_asprintf(mem_ctx, "NAMED-PIPE:%s", epm_floor->rhs.named_pipe.path);
177 :
178 447 : case EPM_PROTOCOL_SMB:
179 447 : return talloc_asprintf(mem_ctx, "SMB:%s", epm_floor->rhs.smb.unc);
180 :
181 0 : case EPM_PROTOCOL_UNIX_DS:
182 0 : return talloc_asprintf(mem_ctx, "Unix:%s", epm_floor->rhs.unix_ds.path);
183 :
184 447 : case EPM_PROTOCOL_NETBIOS:
185 447 : return talloc_asprintf(mem_ctx, "NetBIOS:%s", epm_floor->rhs.netbios.name);
186 :
187 0 : case EPM_PROTOCOL_NETBEUI:
188 0 : return "NETBeui";
189 :
190 0 : case EPM_PROTOCOL_SPX:
191 0 : return "SPX";
192 :
193 0 : case EPM_PROTOCOL_NB_IPX:
194 0 : return "NB_IPX";
195 :
196 76 : case EPM_PROTOCOL_HTTP:
197 76 : return talloc_asprintf(mem_ctx, "HTTP:%"PRIu16, epm_floor->rhs.http.port);
198 :
199 309 : case EPM_PROTOCOL_TCP:
200 309 : return talloc_asprintf(mem_ctx, "TCP:%"PRIu16, epm_floor->rhs.tcp.port);
201 :
202 0 : case EPM_PROTOCOL_UDP:
203 0 : return talloc_asprintf(mem_ctx, "UDP:%"PRIu16, epm_floor->rhs.udp.port);
204 :
205 0 : default:
206 0 : return talloc_asprintf(mem_ctx, "UNK(%02x):", epm_floor->lhs.protocol);
207 : }
208 : }
209 :
210 :
211 : /*
212 : form a binding string from a binding structure
213 : */
214 47066 : _PUBLIC_ char *dcerpc_binding_string(TALLOC_CTX *mem_ctx, const struct dcerpc_binding *b)
215 : {
216 47066 : char *s = NULL;
217 298 : size_t i;
218 47066 : const char *t_name = NULL;
219 47066 : bool option_section = false;
220 47066 : const char *target_hostname = NULL;
221 :
222 47066 : if (b->transport != NCA_UNKNOWN) {
223 47044 : t_name = derpc_transport_string_by_transport(b->transport);
224 47044 : if (!t_name) {
225 0 : return NULL;
226 : }
227 : }
228 :
229 47066 : s = talloc_strdup(mem_ctx, "");
230 :
231 47066 : if (!GUID_all_zero(&b->object)) {
232 6 : struct GUID_txt_buf buf;
233 6 : talloc_asprintf_addbuf(
234 : &s, "%s@", GUID_buf_string(&b->object, &buf));
235 : }
236 :
237 47066 : if (t_name != NULL) {
238 47044 : talloc_asprintf_addbuf(&s, "%s:", t_name);
239 : }
240 :
241 47066 : if (b->host) {
242 1244 : talloc_asprintf_addbuf(&s, "%s", b->host);
243 : }
244 :
245 47066 : target_hostname = b->target_hostname;
246 47066 : if (target_hostname != NULL && b->host != NULL) {
247 1244 : if (strcmp(target_hostname, b->host) == 0) {
248 1211 : target_hostname = NULL;
249 : }
250 : }
251 :
252 47364 : option_section =
253 4492 : (b->endpoint != NULL) ||
254 4473 : (target_hostname != NULL) ||
255 4473 : (b->target_principal != NULL) ||
256 4473 : (b->assoc_group_id != 0) ||
257 55864 : (b->options != NULL) ||
258 4470 : (b->flags != 0);
259 :
260 47066 : if (!option_section) {
261 4108 : return s;
262 : }
263 :
264 42958 : talloc_asprintf_addbuf(&s, "[");
265 :
266 42958 : if (b->endpoint) {
267 42574 : talloc_asprintf_addbuf(&s, "%s", b->endpoint);
268 : }
269 :
270 687328 : for (i=0;i<ARRAY_SIZE(ncacn_options);i++) {
271 644370 : if (!(b->flags & ncacn_options[i].flag)) {
272 643693 : continue;
273 : }
274 :
275 677 : talloc_asprintf_addbuf(&s, ",%s", ncacn_options[i].name);
276 : }
277 :
278 42958 : if (target_hostname) {
279 33 : talloc_asprintf_addbuf(
280 33 : &s, ",target_hostname=%s", b->target_hostname);
281 : }
282 :
283 42958 : if (b->target_principal) {
284 9 : talloc_asprintf_addbuf(
285 0 : &s, ",target_principal=%s", b->target_principal);
286 : }
287 :
288 42958 : if (b->assoc_group_id != 0) {
289 87 : talloc_asprintf_addbuf(
290 60 : &s, ",assoc_group_id=0x%08x", b->assoc_group_id);
291 : }
292 :
293 44007 : for (i=0;b->options && b->options[i];i++) {
294 1049 : talloc_asprintf_addbuf(&s, ",%s", b->options[i]);
295 : }
296 :
297 42958 : talloc_asprintf_addbuf(&s, "]");
298 :
299 42958 : return s;
300 : }
301 :
302 : /*
303 : parse a binding string into a dcerpc_binding structure
304 : */
305 181992 : _PUBLIC_ NTSTATUS dcerpc_parse_binding(TALLOC_CTX *mem_ctx, const char *_s, struct dcerpc_binding **b_out)
306 : {
307 1873 : char *_t;
308 1873 : struct dcerpc_binding *b;
309 1873 : char *s;
310 181992 : char *options = NULL;
311 1873 : char *p;
312 1873 : size_t i;
313 1873 : NTSTATUS status;
314 :
315 181992 : b = talloc_zero(mem_ctx, struct dcerpc_binding);
316 181992 : if (!b) {
317 0 : return NT_STATUS_NO_MEMORY;
318 : }
319 :
320 181992 : _t = talloc_strdup(b, _s);
321 181992 : if (_t == NULL) {
322 0 : talloc_free(b);
323 0 : return NT_STATUS_NO_MEMORY;
324 : }
325 :
326 181992 : s = _t;
327 :
328 181992 : p = strchr(s, '[');
329 181992 : if (p) {
330 150706 : char *q = p + strlen(p) - 1;
331 150706 : if (*q != ']') {
332 0 : talloc_free(b);
333 0 : return NT_STATUS_INVALID_PARAMETER_MIX;
334 : }
335 150706 : *p = '\0';
336 150706 : *q = '\0';
337 150706 : options = p + 1;
338 : }
339 :
340 181992 : p = strchr(s, '@');
341 :
342 181992 : if (p && PTR_DIFF(p, s) == 36) { /* 36 is the length of a UUID */
343 24 : *p = '\0';
344 :
345 24 : status = dcerpc_binding_set_string_option(b, "object", s);
346 24 : if (!NT_STATUS_IS_OK(status)) {
347 0 : talloc_free(b);
348 0 : return status;
349 : }
350 :
351 24 : s = p + 1;
352 : }
353 :
354 181992 : p = strchr(s, ':');
355 :
356 181992 : if (p == NULL) {
357 1892 : b->transport = NCA_UNKNOWN;
358 180100 : } else if (is_ipaddress_v6(s)) {
359 14 : b->transport = NCA_UNKNOWN;
360 : } else {
361 180086 : *p = '\0';
362 :
363 180086 : status = dcerpc_binding_set_string_option(b, "transport", s);
364 180086 : if (!NT_STATUS_IS_OK(status)) {
365 0 : talloc_free(b);
366 0 : return status;
367 : }
368 :
369 180086 : s = p + 1;
370 : }
371 :
372 181992 : if (strlen(s) > 0) {
373 13253 : status = dcerpc_binding_set_string_option(b, "host", s);
374 13253 : if (!NT_STATUS_IS_OK(status)) {
375 0 : talloc_free(b);
376 0 : return status;
377 : }
378 :
379 13253 : b->target_hostname = talloc_strdup(b, b->host);
380 13253 : if (b->target_hostname == NULL) {
381 0 : talloc_free(b);
382 0 : return NT_STATUS_NO_MEMORY;
383 : }
384 : }
385 :
386 336621 : for (i=0; options != NULL; i++) {
387 154629 : const char *name = options;
388 154629 : const char *value = NULL;
389 :
390 154629 : p = strchr(options, ',');
391 154629 : if (p != NULL) {
392 3923 : *p = '\0';
393 3923 : options = p+1;
394 : } else {
395 149302 : options = NULL;
396 : }
397 :
398 154629 : p = strchr(name, '=');
399 154629 : if (p != NULL) {
400 1596 : *p = '\0';
401 1596 : value = p + 1;
402 : }
403 :
404 153083 : if (value == NULL) {
405 : /*
406 : * If it's not a key=value pair
407 : * it might be a ncacn_option
408 : * or if it's the first option
409 : * it's the endpoint.
410 : */
411 153033 : const struct ncacn_option *no = NULL;
412 :
413 153033 : value = name;
414 :
415 153033 : no = ncacn_option_by_name(name);
416 153033 : if (no == NULL) {
417 142684 : if (i > 0) {
418 : /*
419 : * we don't allow unknown options
420 : */
421 0 : return NT_STATUS_INVALID_PARAMETER_MIX;
422 : }
423 :
424 : /*
425 : * This is the endpoint
426 : */
427 142684 : name = "endpoint";
428 142684 : if (strlen(value) == 0) {
429 2770 : value = NULL;
430 : }
431 : }
432 : }
433 :
434 154629 : status = dcerpc_binding_set_string_option(b, name, value);
435 154629 : if (!NT_STATUS_IS_OK(status)) {
436 0 : talloc_free(b);
437 0 : return status;
438 : }
439 : }
440 :
441 181992 : talloc_free(_t);
442 181992 : *b_out = b;
443 181992 : return NT_STATUS_OK;
444 : }
445 :
446 51929 : _PUBLIC_ struct GUID dcerpc_binding_get_object(const struct dcerpc_binding *b)
447 : {
448 51929 : return b->object;
449 : }
450 :
451 12451 : _PUBLIC_ NTSTATUS dcerpc_binding_set_object(struct dcerpc_binding *b,
452 : struct GUID object)
453 : {
454 12451 : char *tmp = discard_const_p(char, b->object_string);
455 :
456 12451 : if (GUID_all_zero(&object)) {
457 12425 : talloc_free(tmp);
458 12425 : b->object_string = NULL;
459 12425 : ZERO_STRUCT(b->object);
460 12425 : return NT_STATUS_OK;
461 : }
462 :
463 26 : b->object_string = GUID_string(b, &object);
464 26 : if (b->object_string == NULL) {
465 0 : b->object_string = tmp;
466 0 : return NT_STATUS_NO_MEMORY;
467 : }
468 26 : talloc_free(tmp);
469 :
470 26 : b->object = object;
471 26 : return NT_STATUS_OK;
472 : }
473 :
474 1848970 : _PUBLIC_ enum dcerpc_transport_t dcerpc_binding_get_transport(const struct dcerpc_binding *b)
475 : {
476 1848970 : return b->transport;
477 : }
478 :
479 184060 : _PUBLIC_ NTSTATUS dcerpc_binding_set_transport(struct dcerpc_binding *b,
480 : enum dcerpc_transport_t transport)
481 : {
482 2177 : NTSTATUS status;
483 :
484 : /*
485 : * TODO: we may want to check the transport value is
486 : * wellknown.
487 : */
488 184060 : if (b->transport == transport) {
489 923 : return NT_STATUS_OK;
490 : }
491 :
492 : /*
493 : * This implicitly resets the endpoint
494 : * as the endpoint is transport specific.
495 : *
496 : * It also resets the assoc group as it's
497 : * also endpoint specific.
498 : *
499 : * TODO: in future we may reset more options
500 : * here.
501 : */
502 183137 : status = dcerpc_binding_set_string_option(b, "endpoint", NULL);
503 183137 : if (!NT_STATUS_IS_OK(status)) {
504 0 : return status;
505 : }
506 :
507 183137 : b->assoc_group_id = 0;
508 :
509 183137 : b->transport = transport;
510 183137 : return NT_STATUS_OK;
511 : }
512 :
513 0 : _PUBLIC_ void dcerpc_binding_get_auth_info(const struct dcerpc_binding *b,
514 : enum dcerpc_AuthType *_auth_type,
515 : enum dcerpc_AuthLevel *_auth_level)
516 : {
517 0 : enum dcerpc_AuthType auth_type;
518 0 : enum dcerpc_AuthLevel auth_level;
519 :
520 0 : if (b->flags & DCERPC_AUTH_SPNEGO) {
521 0 : auth_type = DCERPC_AUTH_TYPE_SPNEGO;
522 0 : } else if (b->flags & DCERPC_AUTH_KRB5) {
523 0 : auth_type = DCERPC_AUTH_TYPE_KRB5;
524 0 : } else if (b->flags & DCERPC_SCHANNEL) {
525 0 : auth_type = DCERPC_AUTH_TYPE_SCHANNEL;
526 0 : } else if (b->flags & DCERPC_AUTH_NTLM) {
527 0 : auth_type = DCERPC_AUTH_TYPE_NTLMSSP;
528 : } else {
529 0 : auth_type = DCERPC_AUTH_TYPE_NONE;
530 : }
531 :
532 0 : if (b->flags & DCERPC_SEAL) {
533 0 : auth_level = DCERPC_AUTH_LEVEL_PRIVACY;
534 0 : } else if (b->flags & DCERPC_SIGN) {
535 0 : auth_level = DCERPC_AUTH_LEVEL_INTEGRITY;
536 0 : } else if (b->flags & DCERPC_CONNECT) {
537 0 : auth_level = DCERPC_AUTH_LEVEL_CONNECT;
538 0 : } else if (b->flags & DCERPC_PACKET) {
539 0 : auth_level = DCERPC_AUTH_LEVEL_PACKET;
540 0 : } else if (auth_type != DCERPC_AUTH_TYPE_NONE) {
541 0 : auth_level = DCERPC_AUTH_LEVEL_INTEGRITY;
542 : } else {
543 0 : auth_level = DCERPC_AUTH_LEVEL_NONE;
544 : }
545 :
546 0 : if (_auth_type != NULL) {
547 0 : *_auth_type = auth_type;
548 : }
549 :
550 0 : if (_auth_level != NULL) {
551 0 : *_auth_level = auth_level;
552 : }
553 0 : }
554 :
555 25942 : _PUBLIC_ uint32_t dcerpc_binding_get_assoc_group_id(const struct dcerpc_binding *b)
556 : {
557 25942 : return b->assoc_group_id;
558 : }
559 :
560 33000 : _PUBLIC_ NTSTATUS dcerpc_binding_set_assoc_group_id(struct dcerpc_binding *b,
561 : uint32_t assoc_group_id)
562 : {
563 33000 : b->assoc_group_id = assoc_group_id;
564 33000 : return NT_STATUS_OK;
565 : }
566 :
567 426925 : _PUBLIC_ struct ndr_syntax_id dcerpc_binding_get_abstract_syntax(const struct dcerpc_binding *b)
568 : {
569 426925 : const char *s = dcerpc_binding_get_string_option(b, "abstract_syntax");
570 7161 : bool ok;
571 7161 : struct ndr_syntax_id id;
572 :
573 426925 : if (s == NULL) {
574 38 : return ndr_syntax_id_null;
575 : }
576 :
577 426887 : ok = ndr_syntax_id_from_string(s, &id);
578 426887 : if (!ok) {
579 0 : return ndr_syntax_id_null;
580 : }
581 :
582 426887 : return id;
583 : }
584 :
585 453151 : _PUBLIC_ NTSTATUS dcerpc_binding_set_abstract_syntax(struct dcerpc_binding *b,
586 : const struct ndr_syntax_id *syntax)
587 : {
588 7812 : NTSTATUS status;
589 7812 : struct ndr_syntax_id_buf buf;
590 :
591 453151 : if (syntax == NULL) {
592 0 : status = dcerpc_binding_set_string_option(b, "abstract_syntax", NULL);
593 0 : return status;
594 : }
595 :
596 453151 : if (ndr_syntax_id_equal(&ndr_syntax_id_null, syntax)) {
597 42 : status = dcerpc_binding_set_string_option(b, "abstract_syntax", NULL);
598 42 : return status;
599 : }
600 :
601 453109 : status = dcerpc_binding_set_string_option(
602 453109 : b, "abstract_syntax", ndr_syntax_id_buf_string(syntax, &buf));
603 453109 : return status;
604 : }
605 :
606 1168922 : _PUBLIC_ const char *dcerpc_binding_get_string_option(const struct dcerpc_binding *b,
607 : const char *name)
608 : {
609 16332 : struct {
610 : const char *name;
611 : const char *value;
612 : #define _SPECIAL(x) { .name = #x, .value = b->x, }
613 1168922 : } specials[] = {
614 1168922 : { .name = "object", .value = b->object_string, },
615 1168922 : _SPECIAL(host),
616 1168922 : _SPECIAL(endpoint),
617 1168922 : _SPECIAL(target_hostname),
618 1168922 : _SPECIAL(target_principal),
619 : #undef _SPECIAL
620 : };
621 1168922 : const struct ncacn_option *no = NULL;
622 1168922 : size_t name_len = strlen(name);
623 16332 : size_t i;
624 16332 : int ret;
625 :
626 1168922 : ret = strcmp(name, "transport");
627 1168922 : if (ret == 0) {
628 0 : return derpc_transport_string_by_transport(b->transport);
629 : }
630 :
631 1168922 : ret = strcmp(name, "assoc_group_id");
632 1168922 : if (ret == 0) {
633 0 : char *tmp = discard_const_p(char, b->assoc_group_string);
634 :
635 0 : if (b->assoc_group_id == 0) {
636 0 : return NULL;
637 : }
638 :
639 0 : snprintf(tmp, sizeof(b->assoc_group_string),
640 0 : "0x%08x", b->assoc_group_id);
641 0 : return (const char *)b->assoc_group_string;
642 : }
643 :
644 4808579 : for (i=0; i < ARRAY_SIZE(specials); i++) {
645 4362808 : ret = strcmp(specials[i].name, name);
646 4362808 : if (ret != 0) {
647 3639657 : continue;
648 : }
649 :
650 723151 : return specials[i].value;
651 : }
652 :
653 445771 : no = ncacn_option_by_name(name);
654 445771 : if (no != NULL) {
655 0 : if (b->flags & no->flag) {
656 0 : return no->name;
657 : }
658 :
659 0 : return NULL;
660 : }
661 :
662 445771 : if (b->options == NULL) {
663 23 : return NULL;
664 : }
665 :
666 465210 : for (i=0; b->options[i]; i++) {
667 447477 : const char *o = b->options[i];
668 447477 : const char *vs = NULL;
669 :
670 447477 : ret = strncmp(name, o, name_len);
671 447477 : if (ret != 0) {
672 19495 : continue;
673 : }
674 :
675 427982 : if (o[name_len] != '=') {
676 0 : continue;
677 : }
678 :
679 427982 : vs = &o[name_len + 1];
680 :
681 427982 : return vs;
682 : }
683 :
684 17488 : return NULL;
685 : }
686 :
687 0 : _PUBLIC_ char *dcerpc_binding_copy_string_option(TALLOC_CTX *mem_ctx,
688 : const struct dcerpc_binding *b,
689 : const char *name)
690 : {
691 0 : const char *c = dcerpc_binding_get_string_option(b, name);
692 0 : char *v;
693 :
694 0 : if (c == NULL) {
695 0 : errno = ENOENT;
696 0 : return NULL;
697 : }
698 :
699 0 : v = talloc_strdup(mem_ctx, c);
700 0 : if (v == NULL) {
701 0 : errno = ENOMEM;
702 0 : return NULL;
703 : }
704 :
705 0 : return v;
706 : }
707 :
708 1048282 : _PUBLIC_ NTSTATUS dcerpc_binding_set_string_option(struct dcerpc_binding *b,
709 : const char *name,
710 : const char *value)
711 : {
712 15263 : struct {
713 : const char *name;
714 : const char **ptr;
715 : #define _SPECIAL(x) { .name = #x, .ptr = &b->x, }
716 1048282 : } specials[] = {
717 1048282 : _SPECIAL(host),
718 1048282 : _SPECIAL(endpoint),
719 1048282 : _SPECIAL(target_hostname),
720 1048282 : _SPECIAL(target_principal),
721 : #undef _SPECIAL
722 : };
723 1048282 : const struct ncacn_option *no = NULL;
724 1048282 : size_t name_len = strlen(name);
725 1048282 : const char *opt = NULL;
726 15263 : char *tmp;
727 15263 : size_t i;
728 15263 : int ret;
729 :
730 : /*
731 : * Note: value == NULL, means delete it.
732 : * value != NULL means add or reset.
733 : */
734 :
735 1048282 : ret = strcmp(name, "transport");
736 1048282 : if (ret == 0) {
737 180086 : enum dcerpc_transport_t t = dcerpc_transport_by_name(value);
738 :
739 180086 : if (t == NCA_UNKNOWN && value != NULL) {
740 0 : return NT_STATUS_INVALID_PARAMETER_MIX;
741 : }
742 :
743 180086 : return dcerpc_binding_set_transport(b, t);
744 : }
745 :
746 868196 : ret = strcmp(name, "object");
747 868196 : if (ret == 0) {
748 8 : NTSTATUS status;
749 24 : struct GUID uuid = GUID_zero();
750 :
751 24 : if (value != NULL) {
752 8 : DATA_BLOB blob;
753 24 : blob = data_blob_string_const(value);
754 24 : if (blob.length != 36) {
755 0 : return NT_STATUS_INVALID_PARAMETER_MIX;
756 : }
757 :
758 24 : status = GUID_from_data_blob(&blob, &uuid);
759 24 : if (!NT_STATUS_IS_OK(status)) {
760 0 : return status;
761 : }
762 : }
763 :
764 24 : return dcerpc_binding_set_object(b, uuid);
765 : }
766 :
767 868172 : ret = strcmp(name, "assoc_group_id");
768 868172 : if (ret == 0) {
769 4 : uint32_t assoc_group_id = 0;
770 :
771 4 : if (value != NULL) {
772 4 : char c;
773 :
774 4 : ret = sscanf(value, "0x%08x%c", &assoc_group_id, &c);
775 4 : if (ret != 1) {
776 0 : return NT_STATUS_INVALID_PARAMETER_MIX;
777 : }
778 : }
779 :
780 4 : return dcerpc_binding_set_assoc_group_id(b, assoc_group_id);
781 : }
782 :
783 3142321 : for (i=0; i < ARRAY_SIZE(specials); i++) {
784 2666141 : ret = strcmp(specials[i].name, name);
785 2666141 : if (ret != 0) {
786 2274153 : continue;
787 : }
788 :
789 391988 : tmp = discard_const_p(char, *specials[i].ptr);
790 :
791 391988 : if (value == NULL) {
792 199837 : talloc_free(tmp);
793 199837 : *specials[i].ptr = NULL;
794 199837 : return NT_STATUS_OK;
795 : }
796 :
797 192151 : if (value[0] == '\0') {
798 0 : return NT_STATUS_INVALID_PARAMETER_MIX;
799 : }
800 :
801 192151 : *specials[i].ptr = talloc_strdup(b, value);
802 192151 : if (*specials[i].ptr == NULL) {
803 0 : *specials[i].ptr = tmp;
804 0 : return NT_STATUS_NO_MEMORY;
805 : }
806 192151 : talloc_free(tmp);
807 :
808 192151 : return NT_STATUS_OK;
809 : }
810 :
811 476180 : no = ncacn_option_by_name(name);
812 476180 : if (no != NULL) {
813 10377 : if (value == NULL) {
814 0 : b->flags &= ~no->flag;
815 0 : return NT_STATUS_OK;
816 : }
817 :
818 10377 : ret = strcasecmp(no->name, value);
819 10377 : if (ret != 0) {
820 0 : return NT_STATUS_INVALID_PARAMETER_MIX;
821 : }
822 :
823 10377 : b->flags |= no->flag;
824 10377 : return NT_STATUS_OK;
825 : }
826 :
827 478773 : for (i=0; b->options && b->options[i]; i++) {
828 28514 : const char *o = b->options[i];
829 :
830 28514 : ret = strncmp(name, o, name_len);
831 28514 : if (ret != 0) {
832 12970 : continue;
833 : }
834 :
835 15544 : if (o[name_len] != '=') {
836 0 : continue;
837 : }
838 :
839 15120 : opt = o;
840 15120 : break;
841 : }
842 :
843 465803 : if (opt == NULL) {
844 7646 : const char **n;
845 :
846 450259 : if (value == NULL) {
847 35 : return NT_STATUS_OK;
848 : }
849 :
850 450224 : n = talloc_realloc(b, b->options, const char *, i + 2);
851 450224 : if (n == NULL) {
852 0 : return NT_STATUS_NO_MEMORY;
853 : }
854 450224 : n[i] = NULL;
855 450224 : n[i + 1] = NULL;
856 450224 : b->options = n;
857 : }
858 :
859 465768 : tmp = discard_const_p(char, opt);
860 :
861 465768 : if (value == NULL) {
862 14 : for (;b->options[i];i++) {
863 7 : b->options[i] = b->options[i+1];
864 : }
865 7 : talloc_free(tmp);
866 7 : return NT_STATUS_OK;
867 : }
868 :
869 465761 : b->options[i] = talloc_asprintf(b->options, "%s=%s",
870 : name, value);
871 465761 : if (b->options[i] == NULL) {
872 0 : b->options[i] = tmp;
873 0 : return NT_STATUS_NO_MEMORY;
874 : }
875 :
876 465761 : return NT_STATUS_OK;
877 : }
878 :
879 72200 : _PUBLIC_ uint32_t dcerpc_binding_get_flags(const struct dcerpc_binding *b)
880 : {
881 72200 : return b->flags;
882 : }
883 :
884 37740 : _PUBLIC_ NTSTATUS dcerpc_binding_set_flags(struct dcerpc_binding *b,
885 : uint32_t additional,
886 : uint32_t clear)
887 : {
888 : /*
889 : * TODO: in future we may want to reject invalid combinations
890 : */
891 37740 : b->flags &= ~clear;
892 37740 : b->flags |= additional;
893 :
894 37740 : return NT_STATUS_OK;
895 : }
896 :
897 72976 : _PUBLIC_ NTSTATUS dcerpc_floor_get_uuid_full(const struct epm_floor *epm_floor,
898 : struct ndr_syntax_id *syntax)
899 : {
900 72976 : TALLOC_CTX *mem_ctx = talloc_init("floor_get_lhs_data");
901 1562 : struct ndr_pull *ndr;
902 1562 : enum ndr_err_code ndr_err;
903 72976 : uint16_t if_version=0;
904 :
905 72976 : *syntax = (struct ndr_syntax_id) { .if_version = 0, };
906 :
907 72976 : if (epm_floor->lhs.protocol != EPM_PROTOCOL_UUID) {
908 0 : talloc_free(mem_ctx);
909 0 : return NT_STATUS_INVALID_PARAMETER;
910 : }
911 :
912 72976 : ndr = ndr_pull_init_blob(&epm_floor->lhs.lhs_data, mem_ctx);
913 72976 : if (ndr == NULL) {
914 0 : talloc_free(mem_ctx);
915 0 : return NT_STATUS_NO_MEMORY;
916 : }
917 72976 : ndr->flags |= LIBNDR_FLAG_NOALIGN;
918 :
919 72976 : ndr_err = ndr_pull_GUID(ndr, NDR_SCALARS | NDR_BUFFERS, &syntax->uuid);
920 72976 : if (!NDR_ERR_CODE_IS_SUCCESS(ndr_err)) {
921 0 : talloc_free(mem_ctx);
922 0 : return ndr_map_error2ntstatus(ndr_err);
923 : }
924 :
925 72976 : ndr_err = ndr_pull_uint16(ndr, NDR_SCALARS, &if_version);
926 72976 : if (!NDR_ERR_CODE_IS_SUCCESS(ndr_err)) {
927 0 : talloc_free(mem_ctx);
928 0 : return ndr_map_error2ntstatus(ndr_err);
929 : }
930 :
931 72976 : syntax->if_version = if_version;
932 :
933 72976 : TALLOC_FREE(ndr);
934 :
935 72976 : ndr = ndr_pull_init_blob(&epm_floor->rhs.uuid.unknown, mem_ctx);
936 72976 : if (ndr == NULL) {
937 0 : talloc_free(mem_ctx);
938 0 : return NT_STATUS_NO_MEMORY;
939 : }
940 72976 : ndr->flags |= LIBNDR_FLAG_NOALIGN;
941 :
942 72976 : ndr_err = ndr_pull_uint16(ndr, NDR_SCALARS, &if_version);
943 72976 : if (!NDR_ERR_CODE_IS_SUCCESS(ndr_err)) {
944 0 : talloc_free(mem_ctx);
945 0 : return ndr_map_error2ntstatus(ndr_err);
946 : }
947 :
948 72976 : syntax->if_version |= (((uint32_t)if_version) << 16) & 0xffff0000;
949 :
950 72976 : talloc_free(mem_ctx);
951 :
952 72976 : return NT_STATUS_OK;
953 : }
954 :
955 853848 : static DATA_BLOB dcerpc_floor_pack_lhs_data(TALLOC_CTX *mem_ctx, const struct ndr_syntax_id *syntax)
956 : {
957 14320 : DATA_BLOB blob;
958 14320 : enum ndr_err_code ndr_err;
959 14320 : struct ndr_push *ndr;
960 :
961 853848 : ndr = ndr_push_init_ctx(mem_ctx);
962 853848 : if (ndr == NULL) {
963 0 : return data_blob_null;
964 : }
965 :
966 853848 : ndr->flags |= LIBNDR_FLAG_NOALIGN;
967 :
968 853848 : ndr_err = ndr_push_GUID(ndr, NDR_SCALARS | NDR_BUFFERS, &syntax->uuid);
969 853848 : if (!NDR_ERR_CODE_IS_SUCCESS(ndr_err)) {
970 0 : return data_blob_null;
971 : }
972 853848 : ndr_err = ndr_push_uint16(ndr, NDR_SCALARS, syntax->if_version);
973 853848 : if (!NDR_ERR_CODE_IS_SUCCESS(ndr_err)) {
974 0 : return data_blob_null;
975 : }
976 :
977 853848 : blob = ndr_push_blob(ndr);
978 853848 : talloc_steal(mem_ctx, blob.data);
979 853848 : talloc_free(ndr);
980 853848 : return blob;
981 : }
982 :
983 853848 : static bool dcerpc_floor_pack_rhs_if_version_data(
984 : TALLOC_CTX *mem_ctx, const struct ndr_syntax_id *syntax,
985 : DATA_BLOB *pblob)
986 : {
987 14320 : DATA_BLOB blob;
988 853848 : struct ndr_push *ndr = ndr_push_init_ctx(mem_ctx);
989 14320 : enum ndr_err_code ndr_err;
990 :
991 853848 : if (ndr == NULL) {
992 0 : return false;
993 : }
994 :
995 853848 : ndr->flags |= LIBNDR_FLAG_NOALIGN;
996 :
997 853848 : ndr_err = ndr_push_uint16(ndr, NDR_SCALARS, syntax->if_version >> 16);
998 853848 : if (!NDR_ERR_CODE_IS_SUCCESS(ndr_err)) {
999 0 : return false;
1000 : }
1001 :
1002 853848 : blob = ndr_push_blob(ndr);
1003 853848 : talloc_steal(mem_ctx, blob.data);
1004 853848 : talloc_free(ndr);
1005 853848 : *pblob = blob;
1006 853848 : return true;
1007 : }
1008 :
1009 853848 : static NTSTATUS dcerpc_floor_pack_uuid_full(TALLOC_CTX *mem_ctx,
1010 : struct epm_floor *floor,
1011 : const struct ndr_syntax_id *syntax)
1012 : {
1013 14320 : bool ok;
1014 :
1015 853848 : floor->lhs.protocol = EPM_PROTOCOL_UUID;
1016 :
1017 853848 : floor->lhs.lhs_data = dcerpc_floor_pack_lhs_data(mem_ctx, syntax);
1018 853848 : if (floor->lhs.lhs_data.data == NULL) {
1019 0 : return NT_STATUS_NO_MEMORY;
1020 : }
1021 :
1022 853848 : ok = dcerpc_floor_pack_rhs_if_version_data(mem_ctx, syntax,
1023 : &floor->rhs.uuid.unknown);
1024 853848 : if (!ok) {
1025 0 : data_blob_free(&floor->lhs.lhs_data);
1026 0 : return NT_STATUS_NO_MEMORY;
1027 : }
1028 :
1029 853848 : return NT_STATUS_OK;
1030 : }
1031 :
1032 7021 : char *dcerpc_floor_get_rhs_data(TALLOC_CTX *mem_ctx, struct epm_floor *epm_floor)
1033 : {
1034 7021 : switch (epm_floor->lhs.protocol) {
1035 5365 : case EPM_PROTOCOL_TCP:
1036 5365 : if (epm_floor->rhs.tcp.port == 0) return NULL;
1037 5358 : return talloc_asprintf(mem_ctx, "%"PRIu16, epm_floor->rhs.tcp.port);
1038 :
1039 1 : case EPM_PROTOCOL_UDP:
1040 1 : if (epm_floor->rhs.udp.port == 0) return NULL;
1041 0 : return talloc_asprintf(mem_ctx, "%"PRIu16, epm_floor->rhs.udp.port);
1042 :
1043 3 : case EPM_PROTOCOL_HTTP:
1044 3 : if (epm_floor->rhs.http.port == 0) return NULL;
1045 2 : return talloc_asprintf(mem_ctx, "%"PRIu16, epm_floor->rhs.http.port);
1046 :
1047 24 : case EPM_PROTOCOL_IP:
1048 24 : return talloc_strdup(mem_ctx, epm_floor->rhs.ip.ipaddr);
1049 :
1050 0 : case EPM_PROTOCOL_NCACN:
1051 0 : return NULL;
1052 :
1053 0 : case EPM_PROTOCOL_NCADG:
1054 0 : return NULL;
1055 :
1056 472 : case EPM_PROTOCOL_SMB:
1057 472 : if (strlen(epm_floor->rhs.smb.unc) == 0) return NULL;
1058 469 : return talloc_strdup(mem_ctx, epm_floor->rhs.smb.unc);
1059 :
1060 682 : case EPM_PROTOCOL_NAMED_PIPE:
1061 682 : if (strlen(epm_floor->rhs.named_pipe.path) == 0) return NULL;
1062 680 : return talloc_strdup(mem_ctx, epm_floor->rhs.named_pipe.path);
1063 :
1064 472 : case EPM_PROTOCOL_NETBIOS:
1065 472 : if (strlen(epm_floor->rhs.netbios.name) == 0) return NULL;
1066 4 : return talloc_strdup(mem_ctx, epm_floor->rhs.netbios.name);
1067 :
1068 0 : case EPM_PROTOCOL_NCALRPC:
1069 0 : return NULL;
1070 :
1071 0 : case EPM_PROTOCOL_VINES_SPP:
1072 0 : return talloc_asprintf(mem_ctx, "%"PRIu16, epm_floor->rhs.vines_spp.port);
1073 :
1074 0 : case EPM_PROTOCOL_VINES_IPC:
1075 0 : return talloc_asprintf(mem_ctx, "%"PRIu16, epm_floor->rhs.vines_ipc.port);
1076 :
1077 0 : case EPM_PROTOCOL_STREETTALK:
1078 0 : return talloc_strdup(mem_ctx, epm_floor->rhs.streettalk.streettalk);
1079 :
1080 2 : case EPM_PROTOCOL_UNIX_DS:
1081 2 : if (strlen(epm_floor->rhs.unix_ds.path) == 0) return NULL;
1082 2 : return talloc_strdup(mem_ctx, epm_floor->rhs.unix_ds.path);
1083 :
1084 0 : case EPM_PROTOCOL_NULL:
1085 0 : return NULL;
1086 :
1087 0 : default:
1088 0 : DEBUG(0,("Unsupported lhs protocol %d\n", epm_floor->lhs.protocol));
1089 0 : break;
1090 : }
1091 :
1092 0 : return NULL;
1093 : }
1094 :
1095 1616245 : static NTSTATUS dcerpc_floor_set_rhs_data(TALLOC_CTX *mem_ctx,
1096 : struct epm_floor *epm_floor,
1097 : const char *data)
1098 : {
1099 1616245 : if (data == NULL) {
1100 1189250 : data = "";
1101 : }
1102 :
1103 1616245 : switch (epm_floor->lhs.protocol) {
1104 238873 : case EPM_PROTOCOL_TCP:
1105 243023 : epm_floor->rhs.tcp.port = atoi(data);
1106 1593257 : return NT_STATUS_OK;
1107 :
1108 0 : case EPM_PROTOCOL_UDP:
1109 1 : epm_floor->rhs.udp.port = atoi(data);
1110 1 : return NT_STATUS_OK;
1111 :
1112 27240 : case EPM_PROTOCOL_HTTP:
1113 27745 : epm_floor->rhs.http.port = atoi(data);
1114 27745 : return NT_STATUS_OK;
1115 :
1116 143572 : case EPM_PROTOCOL_IP:
1117 143572 : if (!is_ipaddress_v4(data)) {
1118 142524 : data = "0.0.0.0";
1119 : }
1120 143572 : epm_floor->rhs.ip.ipaddr = talloc_strdup(mem_ctx, data);
1121 143572 : NT_STATUS_HAVE_NO_MEMORY(epm_floor->rhs.ip.ipaddr);
1122 143572 : return NT_STATUS_OK;
1123 :
1124 335403 : case EPM_PROTOCOL_NCACN:
1125 335403 : epm_floor->rhs.ncacn.minor_version = 0;
1126 335403 : return NT_STATUS_OK;
1127 :
1128 1 : case EPM_PROTOCOL_NCADG:
1129 1 : epm_floor->rhs.ncadg.minor_version = 0;
1130 1 : return NT_STATUS_OK;
1131 :
1132 394809 : case EPM_PROTOCOL_SMB:
1133 394809 : epm_floor->rhs.smb.unc = talloc_strdup(mem_ctx, data);
1134 394809 : NT_STATUS_HAVE_NO_MEMORY(epm_floor->rhs.smb.unc);
1135 394809 : return NT_STATUS_OK;
1136 :
1137 182469 : case EPM_PROTOCOL_NAMED_PIPE:
1138 182469 : epm_floor->rhs.named_pipe.path = talloc_strdup(mem_ctx, data);
1139 182469 : NT_STATUS_HAVE_NO_MEMORY(epm_floor->rhs.named_pipe.path);
1140 182469 : return NT_STATUS_OK;
1141 :
1142 197698 : case EPM_PROTOCOL_NETBIOS:
1143 197698 : epm_floor->rhs.netbios.name = talloc_strdup(mem_ctx, data);
1144 197698 : NT_STATUS_HAVE_NO_MEMORY(epm_floor->rhs.netbios.name);
1145 197698 : return NT_STATUS_OK;
1146 :
1147 91520 : case EPM_PROTOCOL_NCALRPC:
1148 91520 : return NT_STATUS_OK;
1149 :
1150 0 : case EPM_PROTOCOL_VINES_SPP:
1151 0 : epm_floor->rhs.vines_spp.port = atoi(data);
1152 0 : return NT_STATUS_OK;
1153 :
1154 0 : case EPM_PROTOCOL_VINES_IPC:
1155 0 : epm_floor->rhs.vines_ipc.port = atoi(data);
1156 0 : return NT_STATUS_OK;
1157 :
1158 0 : case EPM_PROTOCOL_STREETTALK:
1159 0 : epm_floor->rhs.streettalk.streettalk = talloc_strdup(mem_ctx, data);
1160 0 : NT_STATUS_HAVE_NO_MEMORY(epm_floor->rhs.streettalk.streettalk);
1161 0 : return NT_STATUS_OK;
1162 :
1163 4 : case EPM_PROTOCOL_UNIX_DS:
1164 4 : epm_floor->rhs.unix_ds.path = talloc_strdup(mem_ctx, data);
1165 4 : NT_STATUS_HAVE_NO_MEMORY(epm_floor->rhs.unix_ds.path);
1166 4 : return NT_STATUS_OK;
1167 :
1168 0 : case EPM_PROTOCOL_NULL:
1169 0 : return NT_STATUS_OK;
1170 :
1171 0 : default:
1172 0 : DEBUG(0,("Unsupported lhs protocol %d\n", epm_floor->lhs.protocol));
1173 0 : break;
1174 : }
1175 :
1176 0 : return NT_STATUS_NOT_SUPPORTED;
1177 : }
1178 :
1179 0 : enum dcerpc_transport_t dcerpc_transport_by_endpoint_protocol(int prot)
1180 : {
1181 0 : size_t i;
1182 :
1183 : /* Find a transport that has 'prot' as 4th protocol */
1184 0 : for (i=0;i<ARRAY_SIZE(transports);i++) {
1185 0 : if (transports[i].num_protocols >= 2 &&
1186 0 : transports[i].protseq[1] == prot) {
1187 0 : return transports[i].transport;
1188 : }
1189 : }
1190 :
1191 : /* Unknown transport */
1192 0 : return (unsigned int)-1;
1193 : }
1194 :
1195 283053 : _PUBLIC_ enum dcerpc_transport_t dcerpc_transport_by_tower(const struct epm_tower *tower)
1196 : {
1197 5405 : size_t i;
1198 :
1199 : /* Find a transport that matches this tower */
1200 598603 : for (i=0;i<ARRAY_SIZE(transports);i++) {
1201 11794 : int j;
1202 598413 : if (transports[i].num_protocols != tower->num_floors - 2) {
1203 254968 : continue;
1204 : }
1205 :
1206 1188679 : for (j = 0; j < transports[i].num_protocols && j < MAX_PROTSEQ; j++) {
1207 905816 : if (transports[i].protseq[j] != tower->floors[j+2].lhs.protocol) {
1208 59173 : break;
1209 : }
1210 : }
1211 :
1212 343445 : if (j == transports[i].num_protocols) {
1213 282863 : return transports[i].transport;
1214 : }
1215 : }
1216 :
1217 : /* Unknown transport */
1218 190 : return (unsigned int)-1;
1219 : }
1220 :
1221 93525 : _PUBLIC_ const char *derpc_transport_string_by_transport(enum dcerpc_transport_t t)
1222 : {
1223 716 : size_t i;
1224 :
1225 115516 : for (i=0; i<ARRAY_SIZE(transports); i++) {
1226 115516 : if (t == transports[i].transport) {
1227 93525 : return transports[i].name;
1228 : }
1229 : }
1230 0 : return NULL;
1231 : }
1232 :
1233 180086 : _PUBLIC_ enum dcerpc_transport_t dcerpc_transport_by_name(const char *name)
1234 : {
1235 1771 : size_t i;
1236 :
1237 180086 : if (name == NULL) {
1238 0 : return NCA_UNKNOWN;
1239 : }
1240 :
1241 280299 : for (i=0; i<ARRAY_SIZE(transports);i++) {
1242 280299 : if (strcasecmp(name, transports[i].name) == 0) {
1243 180086 : return transports[i].transport;
1244 : }
1245 : }
1246 :
1247 0 : return NCA_UNKNOWN;
1248 : }
1249 :
1250 501 : _PUBLIC_ NTSTATUS dcerpc_binding_from_tower(TALLOC_CTX *mem_ctx,
1251 : struct epm_tower *tower,
1252 : struct dcerpc_binding **b_out)
1253 : {
1254 35 : NTSTATUS status;
1255 35 : struct dcerpc_binding *b;
1256 35 : enum dcerpc_transport_t transport;
1257 35 : struct ndr_syntax_id abstract_syntax;
1258 501 : char *endpoint = NULL;
1259 501 : char *host = NULL;
1260 :
1261 : /*
1262 : * A tower needs to have at least 4 floors to carry useful
1263 : * information. Floor 3 is the transport identifier which defines
1264 : * how many floors are required at least.
1265 : */
1266 501 : if (tower->num_floors < 4) {
1267 0 : return NT_STATUS_INVALID_PARAMETER;
1268 : }
1269 :
1270 501 : status = dcerpc_parse_binding(mem_ctx, "", &b);
1271 501 : if (!NT_STATUS_IS_OK(status)) {
1272 0 : return status;
1273 : }
1274 :
1275 501 : transport = dcerpc_transport_by_tower(tower);
1276 501 : if (transport == NCA_UNKNOWN) {
1277 0 : talloc_free(b);
1278 0 : return NT_STATUS_NOT_SUPPORTED;
1279 : }
1280 :
1281 501 : status = dcerpc_binding_set_transport(b, transport);
1282 501 : if (!NT_STATUS_IS_OK(status)) {
1283 0 : talloc_free(b);
1284 0 : return status;
1285 : }
1286 :
1287 : /* Set abstract syntax */
1288 501 : status = dcerpc_floor_get_uuid_full(&tower->floors[0], &abstract_syntax);
1289 501 : if (!NT_STATUS_IS_OK(status)) {
1290 0 : talloc_free(b);
1291 0 : return status;
1292 : }
1293 :
1294 501 : status = dcerpc_binding_set_abstract_syntax(b, &abstract_syntax);
1295 501 : if (!NT_STATUS_IS_OK(status)) {
1296 0 : talloc_free(b);
1297 0 : return status;
1298 : }
1299 :
1300 : /* Ignore floor 1, it contains the NDR version info */
1301 :
1302 : /* Set endpoint */
1303 501 : errno = 0;
1304 501 : if (tower->num_floors >= 4) {
1305 501 : endpoint = dcerpc_floor_get_rhs_data(b, &tower->floors[3]);
1306 : }
1307 501 : if (errno != 0) {
1308 0 : int saved_errno = errno;
1309 0 : talloc_free(b);
1310 0 : return map_nt_error_from_unix_common(saved_errno);
1311 : }
1312 :
1313 501 : status = dcerpc_binding_set_string_option(b, "endpoint", endpoint);
1314 501 : if (!NT_STATUS_IS_OK(status)) {
1315 0 : talloc_free(b);
1316 0 : return status;
1317 : }
1318 501 : TALLOC_FREE(endpoint);
1319 :
1320 : /* Set network address */
1321 501 : errno = 0;
1322 501 : if (tower->num_floors >= 5) {
1323 496 : host = dcerpc_floor_get_rhs_data(b, &tower->floors[4]);
1324 : }
1325 501 : if (errno != 0) {
1326 0 : int saved_errno = errno;
1327 0 : talloc_free(b);
1328 0 : return map_nt_error_from_unix_common(saved_errno);
1329 : }
1330 :
1331 501 : status = dcerpc_binding_set_string_option(b, "host", host);
1332 501 : if (!NT_STATUS_IS_OK(status)) {
1333 0 : talloc_free(b);
1334 0 : return status;
1335 : }
1336 501 : status = dcerpc_binding_set_string_option(b, "target_hostname", host);
1337 501 : if (!NT_STATUS_IS_OK(status)) {
1338 0 : talloc_free(b);
1339 0 : return status;
1340 : }
1341 501 : TALLOC_FREE(host);
1342 :
1343 501 : *b_out = b;
1344 501 : return NT_STATUS_OK;
1345 : }
1346 :
1347 490908 : _PUBLIC_ struct dcerpc_binding *dcerpc_binding_dup(TALLOC_CTX *mem_ctx,
1348 : const struct dcerpc_binding *b)
1349 : {
1350 9136 : struct dcerpc_binding *n;
1351 9136 : uint32_t count;
1352 :
1353 490908 : n = talloc_zero(mem_ctx, struct dcerpc_binding);
1354 490908 : if (n == NULL) {
1355 0 : return NULL;
1356 : }
1357 :
1358 490908 : n->transport = b->transport;
1359 490908 : n->object = b->object;
1360 490908 : n->flags = b->flags;
1361 490908 : n->assoc_group_id = b->assoc_group_id;
1362 :
1363 490908 : if (b->object_string != NULL) {
1364 25 : n->object_string = talloc_strdup(n, b->object_string);
1365 25 : if (n->object_string == NULL) {
1366 0 : goto nomem;
1367 : }
1368 : }
1369 490908 : if (b->host != NULL) {
1370 66424 : n->host = talloc_strdup(n, b->host);
1371 66424 : if (n->host == NULL) {
1372 0 : goto nomem;
1373 : }
1374 : }
1375 :
1376 490908 : if (b->target_hostname != NULL) {
1377 66424 : n->target_hostname = talloc_strdup(n, b->target_hostname);
1378 66424 : if (n->target_hostname == NULL) {
1379 0 : goto nomem;
1380 : }
1381 : }
1382 :
1383 490908 : if (b->target_principal != NULL) {
1384 20059 : n->target_principal = talloc_strdup(n, b->target_principal);
1385 20059 : if (n->target_principal == NULL) {
1386 0 : goto nomem;
1387 : }
1388 : }
1389 :
1390 490908 : if (b->endpoint != NULL) {
1391 448321 : n->endpoint = talloc_strdup(n, b->endpoint);
1392 448321 : if (n->endpoint == NULL) {
1393 0 : goto nomem;
1394 : }
1395 : }
1396 :
1397 555979 : for (count = 0; b->options && b->options[count]; count++);
1398 :
1399 490908 : if (count > 0) {
1400 1845 : uint32_t i;
1401 :
1402 50435 : n->options = talloc_array(n, const char *, count + 1);
1403 50435 : if (n->options == NULL) {
1404 0 : goto nomem;
1405 : }
1406 :
1407 115506 : for (i = 0; i < count; i++) {
1408 65071 : n->options[i] = talloc_strdup(n->options, b->options[i]);
1409 65071 : if (n->options[i] == NULL) {
1410 0 : goto nomem;
1411 : }
1412 : }
1413 50435 : n->options[count] = NULL;
1414 : }
1415 :
1416 481772 : return n;
1417 0 : nomem:
1418 0 : TALLOC_FREE(n);
1419 0 : return NULL;
1420 : }
1421 :
1422 426924 : _PUBLIC_ NTSTATUS dcerpc_binding_build_tower(TALLOC_CTX *mem_ctx,
1423 : const struct dcerpc_binding *binding,
1424 : struct epm_tower *tower)
1425 : {
1426 426924 : const enum epm_protocol *protseq = NULL;
1427 426924 : size_t i, num_protocols = 0;
1428 7160 : struct ndr_syntax_id abstract_syntax;
1429 7160 : NTSTATUS status;
1430 :
1431 : /* Find transport */
1432 944885 : for (i=0;i<ARRAY_SIZE(transports);i++) {
1433 944885 : if (transports[i].transport == binding->transport) {
1434 426924 : protseq = transports[i].protseq;
1435 426924 : num_protocols = transports[i].num_protocols;
1436 426924 : break;
1437 : }
1438 : }
1439 :
1440 426924 : if (i == ARRAY_SIZE(transports)) {
1441 0 : DEBUG(0, ("Unable to find transport with id '%d'\n", binding->transport));
1442 0 : return NT_STATUS_UNSUCCESSFUL;
1443 : }
1444 :
1445 426924 : tower->num_floors = 2 + num_protocols;
1446 426924 : tower->floors = talloc_array(mem_ctx, struct epm_floor, tower->num_floors);
1447 426924 : if (tower->floors == NULL) {
1448 0 : return NT_STATUS_NO_MEMORY;
1449 : }
1450 :
1451 : /* Floor 0 */
1452 426924 : abstract_syntax = dcerpc_binding_get_abstract_syntax(binding);
1453 426924 : status = dcerpc_floor_pack_uuid_full(tower->floors,
1454 : &tower->floors[0],
1455 : &abstract_syntax);
1456 426924 : if (!NT_STATUS_IS_OK(status)) {
1457 0 : return status;
1458 : }
1459 :
1460 : /* Floor 1 */
1461 434084 : status = dcerpc_floor_pack_uuid_full(tower->floors,
1462 426924 : &tower->floors[1],
1463 : &ndr_transfer_syntax_ndr);
1464 426924 : if (!NT_STATUS_IS_OK(status)) {
1465 0 : return status;
1466 : }
1467 :
1468 : /* Floor 2 to num_protocols */
1469 1616174 : for (i = 0; i < num_protocols; i++) {
1470 1189250 : tower->floors[2 + i].lhs.protocol = protseq[i];
1471 1189250 : tower->floors[2 + i].lhs.lhs_data = data_blob_null;
1472 1189250 : ZERO_STRUCT(tower->floors[2 + i].rhs);
1473 1209221 : status = dcerpc_floor_set_rhs_data(tower->floors,
1474 1189250 : &tower->floors[2 + i],
1475 : NULL);
1476 1189250 : if (!NT_STATUS_IS_OK(status)) {
1477 0 : return status;
1478 : }
1479 : }
1480 :
1481 : /* The 4th floor contains the endpoint */
1482 426924 : if (num_protocols >= 2 && binding->endpoint) {
1483 428148 : status = dcerpc_floor_set_rhs_data(tower->floors,
1484 421127 : &tower->floors[3],
1485 414106 : binding->endpoint);
1486 421127 : if (!NT_STATUS_IS_OK(status)) {
1487 0 : return status;
1488 : }
1489 : }
1490 :
1491 : /* The 5th contains the network address */
1492 426924 : if (num_protocols >= 3 && binding->host) {
1493 6014 : status = dcerpc_floor_set_rhs_data(tower->floors,
1494 5868 : &tower->floors[4],
1495 5722 : binding->host);
1496 5868 : if (!NT_STATUS_IS_OK(status)) {
1497 0 : return status;
1498 : }
1499 : }
1500 :
1501 426924 : return NT_STATUS_OK;
1502 : }
|