1 ###################################################
2 # client calls generator
3 # Copyright tridge@samba.org 2003
4 # Copyright jelmer@samba.org 2005-2006
5 # released under the GNU GPL
7 package Parse::Pidl::Samba4::NDR::Client;
11 @EXPORT_OK = qw(Parse);
13 use Parse::Pidl qw(fatal warning error);
14 use Parse::Pidl::Util qw(has_property ParseExpr);
15 use Parse::Pidl::NDR qw(ContainsPipe);
16 use Parse::Pidl::Typelist qw(mapTypeName);
17 use Parse::Pidl::Samba4 qw(choose_header is_intree DeclLong);
18 use Parse::Pidl::Samba4::Header qw(GenerateFunctionInEnv GenerateFunctionOutEnv);
20 use vars qw($VERSION);
25 sub indent($) { my ($self) = @_; $self->{tabs}.="\t"; }
26 sub deindent($) { my ($self) = @_; $self->{tabs} = substr($self->{tabs}, 1); }
27 sub pidl($$) { my ($self,$txt) = @_; $self->{res} .= $txt ? "$self->{tabs}$txt\n" : "\n"; }
28 sub pidl_hdr($$) { my ($self, $txt) = @_; $self->{res_hdr} .= "$txt\n"; }
29 sub pidl_both($$) { my ($self, $txt) = @_; $self->{hdr} .= "$txt\n"; $self->{res_hdr} .= "$txt\n"; }
30 sub fn_declare($$) { my ($self,$n) = @_; $self->pidl($n); $self->pidl_hdr("$n;"); }
35 my $nt = int((length($s)+1)/8);
37 my $ns = (length($s)-$lt);
38 return "\t"x($nt)." "x($ns);
44 my $self = { res => "", res_hdr => "", tabs => "" };
48 sub ParseFunctionHasPipes($$)
52 foreach my $e (@{$fn->{ELEMENTS}}) {
53 return 1 if ContainsPipe($e, $e->{LEVELS}[0]);
59 sub ParseFunction_r_State($$$$)
61 my ($self, $if, $fn, $name) = @_;
64 $self->pidl("struct dcerpc_$name\_r_state {");
66 $self->pidl("TALLOC_CTX *out_mem_ctx;");
70 $self->pidl("static void dcerpc_$name\_r_done(struct tevent_req *subreq);");
74 sub ParseFunction_r_Send($$$$)
76 my ($self, $if, $fn, $name) = @_;
79 my $proto = "struct tevent_req *dcerpc_$name\_r_send(TALLOC_CTX *mem_ctx,\n";
80 $proto .= "\tstruct tevent_context *ev,\n",
81 $proto .= "\tstruct dcerpc_binding_handle *h,\n",
82 $proto .= "\tstruct $name *r)";
84 $self->fn_declare($proto);
89 $self->pidl("struct tevent_req *req;");
90 $self->pidl("struct dcerpc_$name\_r_state *state;");
91 $self->pidl("struct tevent_req *subreq;");
94 $self->pidl("req = tevent_req_create(mem_ctx, &state,");
95 $self->pidl("\t\t\tstruct dcerpc_$name\_r_state);");
96 $self->pidl("if (req == NULL) {");
98 $self->pidl("return NULL;");
104 foreach my $e (@{$fn->{ELEMENTS}}) {
105 next unless grep(/out/, @{$e->{DIRECTION}});
106 next if ContainsPipe($e, $e->{LEVELS}[0]);
112 if ($out_params > 0) {
113 $self->pidl("state->out_mem_ctx = talloc_new(state);");
114 $self->pidl("if (tevent_req_nomem(state->out_mem_ctx, req)) {");
116 $self->pidl("return tevent_req_post(req, ev);");
119 $submem = "state->out_mem_ctx";
121 $self->pidl("state->out_mem_ctx = NULL;");
126 $self->pidl("subreq = dcerpc_binding_handle_call_send(state, ev, h,");
127 $self->pidl("\t\tNULL, &ndr_table_$if->{NAME},");
128 $self->pidl("\t\tNDR_$uname, $submem, r);");
129 $self->pidl("if (tevent_req_nomem(subreq, req)) {");
131 $self->pidl("return tevent_req_post(req, ev);");
134 $self->pidl("tevent_req_set_callback(subreq, dcerpc_$name\_r_done, req);");
137 $self->pidl("return req;");
143 sub ParseFunction_r_Done($$$$)
145 my ($self, $if, $fn, $name) = @_;
146 my $uname = uc $name;
148 my $proto = "static void dcerpc_$name\_r_done(struct tevent_req *subreq)";
150 $self->pidl("$proto");
154 $self->pidl("struct tevent_req *req =");
155 $self->pidl("\ttevent_req_callback_data(subreq,");
156 $self->pidl("\tstruct tevent_req);");
157 $self->pidl("NTSTATUS status;");
160 $self->pidl("status = dcerpc_binding_handle_call_recv(subreq);");
161 $self->pidl("TALLOC_FREE(subreq);");
162 $self->pidl("if (tevent_req_nterror(req, status)) {");
164 $self->pidl("return;");
169 $self->pidl("tevent_req_done(req);");
175 sub ParseFunction_r_Recv($$$$)
177 my ($self, $if, $fn, $name) = @_;
178 my $uname = uc $name;
180 my $proto = "NTSTATUS dcerpc_$name\_r_recv(struct tevent_req *req, TALLOC_CTX *mem_ctx)";
182 $self->fn_declare($proto);
187 $self->pidl("struct dcerpc_$name\_r_state *state =");
188 $self->pidl("\ttevent_req_data(req,");
189 $self->pidl("\tstruct dcerpc_$name\_r_state);");
190 $self->pidl("NTSTATUS status;");
193 $self->pidl("if (tevent_req_is_nterror(req, &status)) {");
195 $self->pidl("tevent_req_received(req);");
196 $self->pidl("return status;");
201 $self->pidl("talloc_steal(mem_ctx, state->out_mem_ctx);");
204 $self->pidl("tevent_req_received(req);");
205 $self->pidl("return NT_STATUS_OK;");
211 sub ParseFunction_r_Sync($$$$)
213 my ($self, $if, $fn, $name) = @_;
214 my $uname = uc $name;
216 if ($self->ParseFunctionHasPipes($fn)) {
217 $self->pidl_both("/*");
218 $self->pidl_both(" * The following function is skipped because");
219 $self->pidl_both(" * it uses pipes:");
220 $self->pidl_both(" *");
221 $self->pidl_both(" * dcerpc_$name\_r()");
222 $self->pidl_both(" */");
223 $self->pidl_both("");
227 my $proto = "NTSTATUS dcerpc_$name\_r(struct dcerpc_binding_handle *h, TALLOC_CTX *mem_ctx, struct $name *r)";
229 $self->fn_declare($proto);
233 $self->pidl("NTSTATUS status;");
236 $self->pidl("status = dcerpc_binding_handle_call(h,");
237 $self->pidl("\t\tNULL, &ndr_table_$if->{NAME},");
238 $self->pidl("\t\tNDR_$uname, mem_ctx, r);");
240 $self->pidl("return status;");
247 sub ElementDirection($)
251 return "[in,out]" if (has_property($e, "in") and has_property($e, "out"));
252 return "[in]" if (has_property($e, "in"));
253 return "[out]" if (has_property($e, "out"));
257 sub HeaderProperties($$)
259 my($props,$ignores) = @_;
262 foreach my $d (keys %{$props}) {
263 next if (grep(/^$d$/, @$ignores));
264 if($props->{$d} ne "1") {
265 $ret.= "$d($props->{$d}),";
272 return "[" . substr($ret, 0, -1) . "]";
276 sub ParseCopyArgument($$$$$)
278 my ($self, $fn, $e, $r, $i) = @_;
279 my $l = $e->{LEVELS}[0];
281 if ($l->{TYPE} eq "ARRAY" and $l->{IS_FIXED} == 1) {
282 $self->pidl("memcpy(${r}$e->{NAME}, ${i}$e->{NAME}, sizeof(${r}$e->{NAME}));");
284 $self->pidl("${r}$e->{NAME} = ${i}$e->{NAME};");
288 sub ParseInvalidResponse($$)
290 my ($self, $type) = @_;
292 if ($type eq "sync") {
293 $self->pidl("return NT_STATUS_INVALID_NETWORK_RESPONSE;");
294 } elsif ($type eq "async") {
295 $self->pidl("tevent_req_nterror(req, NT_STATUS_INVALID_NETWORK_RESPONSE);");
296 $self->pidl("return;");
298 die("ParseInvalidResponse($type)");
302 sub ParseOutputArgument($$$$$$)
304 my ($self, $fn, $e, $r, $o, $invalid_response_type) = @_;
307 if ($e->{LEVELS}[0]->{TYPE} ne "POINTER" and $e->{LEVELS}[0]->{TYPE} ne "ARRAY") {
308 fatal($e->{ORIGINAL}, "[out] argument is not a pointer or array");
312 if ($e->{LEVELS}[0]->{TYPE} eq "POINTER") {
314 if ($e->{LEVELS}[0]->{POINTER_TYPE} ne "ref") {
315 $self->pidl("if ($o$e->{NAME} && ${r}out.$e->{NAME}) {");
320 if ($e->{LEVELS}[$level]->{TYPE} eq "ARRAY") {
321 # This is a call to GenerateFunctionInEnv intentionally.
322 # Since the data is being copied into a user-provided data
323 # structure, the user should be able to know the size beforehand
324 # to allocate a structure of the right size.
325 my $in_env = GenerateFunctionInEnv($fn, $r);
326 my $out_env = GenerateFunctionOutEnv($fn, $r);
327 my $l = $e->{LEVELS}[$level];
330 if (grep(/in/, @{$e->{DIRECTION}})) {
331 $in_var = ParseExpr($e->{NAME}, $in_env, $e->{ORIGINAL});
333 my $out_var = ParseExpr($e->{NAME}, $out_env, $e->{ORIGINAL});
335 my $in_size_is = undef;
336 my $out_size_is = undef;
337 my $out_length_is = undef;
339 my $avail_len = undef;
340 my $needed_len = undef;
344 my $copy_len_var = "_copy_len_$e->{NAME}";
345 $self->pidl("size_t $copy_len_var;");
347 if (not defined($l->{SIZE_IS})) {
348 if (not $l->{IS_ZERO_TERMINATED}) {
349 fatal($e->{ORIGINAL}, "no size known for [out] array `$e->{NAME}'");
351 if (has_property($e, "charset")) {
352 $avail_len = "ndr_charset_length($in_var, CH_UNIX)";
353 $needed_len = "ndr_charset_length($out_var, CH_UNIX)";
355 $avail_len = "ndr_string_length($in_var, sizeof(*$in_var))";
356 $needed_len = "ndr_string_length($out_var, sizeof(*$out_var))";
362 $in_size_is = ParseExpr($l->{SIZE_IS}, $in_env, $e->{ORIGINAL});
363 $out_size_is = ParseExpr($l->{SIZE_IS}, $out_env, $e->{ORIGINAL});
364 $out_length_is = $out_size_is;
365 if (defined($l->{LENGTH_IS})) {
366 $out_length_is = ParseExpr($l->{LENGTH_IS}, $out_env, $e->{ORIGINAL});
368 if (has_property($e, "charset")) {
369 if (defined($in_var)) {
370 $avail_len = "ndr_charset_length($in_var, CH_UNIX)";
372 $avail_len = $out_length_is;
374 $needed_len = "ndr_charset_length($out_var, CH_UNIX)";
378 if ($out_size_is ne $in_size_is) {
379 $self->pidl("if (($out_size_is) > ($in_size_is)) {");
381 $self->ParseInvalidResponse($invalid_response_type);
385 if ($out_length_is ne $out_size_is) {
386 $self->pidl("if (($out_length_is) > ($out_size_is)) {");
388 $self->ParseInvalidResponse($invalid_response_type);
392 if (defined($needed_len)) {
393 $self->pidl("$copy_len_var = $needed_len;");
394 $self->pidl("if ($copy_len_var > $avail_len) {");
396 $self->ParseInvalidResponse($invalid_response_type);
400 $self->pidl("$copy_len_var = $out_length_is;");
403 if (has_property($e, "charset")) {
404 $self->pidl("memcpy(discard_const_p(uint8_t *, $o$e->{NAME}), $out_var, $copy_len_var * sizeof(*$o$e->{NAME}));");
406 $self->pidl("memcpy($o$e->{NAME}, $out_var, $copy_len_var * sizeof(*$o$e->{NAME}));");
412 $self->pidl("*$o$e->{NAME} = *${r}out.$e->{NAME};");
415 if ($e->{LEVELS}[0]->{TYPE} eq "POINTER") {
416 if ($e->{LEVELS}[0]->{POINTER_TYPE} ne "ref") {
423 sub ParseFunction_State($$$$)
425 my ($self, $if, $fn, $name) = @_;
427 my $state_str = "struct dcerpc_$name\_state";
428 my $done_fn = "dcerpc_$name\_done";
430 $self->pidl("$state_str {");
432 $self->pidl("struct $name orig;");
433 $self->pidl("struct $name tmp;");
434 $self->pidl("TALLOC_CTX *out_mem_ctx;");
438 $self->pidl("static void $done_fn(struct tevent_req *subreq);");
442 sub ParseFunction_Send($$$$)
444 my ($self, $if, $fn, $name) = @_;
447 my $state_str = "struct dcerpc_$name\_state";
448 my $done_fn = "dcerpc_$name\_done";
449 my $out_mem_ctx = "dcerpc_$name\_out_memory";
450 my $fn_str = "struct tevent_req *dcerpc_$name\_send";
451 my $pad = genpad($fn_str);
453 $fn_args .= "TALLOC_CTX *mem_ctx";
454 $fn_args .= ",\n" . $pad . "struct tevent_context *ev";
455 $fn_args .= ",\n" . $pad . "struct dcerpc_binding_handle *h";
457 foreach (@{$fn->{ELEMENTS}}) {
458 my $dir = ElementDirection($_);
459 my $prop = HeaderProperties($_->{PROPERTIES}, ["in", "out"]);
460 $fn_args .= ",\n" . $pad . DeclLong($_, "_") . " /* $dir $prop */";
463 $self->fn_declare("$fn_str($fn_args)");
466 $self->pidl("struct tevent_req *req;");
467 $self->pidl("$state_str *state;");
468 $self->pidl("struct tevent_req *subreq;");
470 $self->pidl("req = tevent_req_create(mem_ctx, &state,");
471 $self->pidl("\t\t\t$state_str);");
472 $self->pidl("if (req == NULL) {");
474 $self->pidl("return NULL;");
477 $self->pidl("state->out_mem_ctx = NULL;");
480 $self->pidl("/* In parameters */");
481 foreach my $e (@{$fn->{ELEMENTS}}) {
482 next unless (grep(/in/, @{$e->{DIRECTION}}));
484 $self->ParseCopyArgument($fn, $e, "state->orig.in.", "_");
489 $self->pidl("/* Out parameters */");
490 foreach my $e (@{$fn->{ELEMENTS}}) {
491 next unless grep(/out/, @{$e->{DIRECTION}});
493 $self->ParseCopyArgument($fn, $e, "state->orig.out.", "_");
495 next if ContainsPipe($e, $e->{LEVELS}[0]);
501 if (defined($fn->{RETURN_TYPE})) {
502 $self->pidl("/* Result */");
503 $self->pidl("ZERO_STRUCT(state->orig.out.result);");
507 if ($out_params > 0) {
508 $self->pidl("state->out_mem_ctx = talloc_named_const(state, 0,");
509 $self->pidl("\t\t \"$out_mem_ctx\");");
510 $self->pidl("if (tevent_req_nomem(state->out_mem_ctx, req)) {");
512 $self->pidl("return tevent_req_post(req, ev);");
518 $self->pidl("/* make a temporary copy, that we pass to the dispatch function */");
519 $self->pidl("state->tmp = state->orig;");
522 $self->pidl("subreq = dcerpc_$name\_r_send(state, ev, h, &state->tmp);");
523 $self->pidl("if (tevent_req_nomem(subreq, req)) {");
525 $self->pidl("return tevent_req_post(req, ev);");
528 $self->pidl("tevent_req_set_callback(subreq, $done_fn, req);");
529 $self->pidl("return req;");
535 sub ParseFunction_Done($$$$)
537 my ($self, $if, $fn, $name) = @_;
539 my $state_str = "struct dcerpc_$name\_state";
540 my $done_fn = "dcerpc_$name\_done";
542 $self->pidl("static void $done_fn(struct tevent_req *subreq)");
545 $self->pidl("struct tevent_req *req = tevent_req_callback_data(");
546 $self->pidl("\tsubreq, struct tevent_req);");
547 $self->pidl("$state_str *state = tevent_req_data(");
548 $self->pidl("\treq, $state_str);");
549 $self->pidl("NTSTATUS status;");
550 $self->pidl("TALLOC_CTX *mem_ctx;");
553 $self->pidl("if (state->out_mem_ctx) {");
555 $self->pidl("mem_ctx = state->out_mem_ctx;");
557 $self->pidl("} else {");
559 $self->pidl("mem_ctx = state;");
564 $self->pidl("status = dcerpc_$name\_r_recv(subreq, mem_ctx);");
565 $self->pidl("TALLOC_FREE(subreq);");
566 $self->pidl("if (tevent_req_nterror(req, status)) {");
568 $self->pidl("return;");
573 $self->pidl("/* Copy out parameters */");
574 foreach my $e (@{$fn->{ELEMENTS}}) {
575 next if ContainsPipe($e, $e->{LEVELS}[0]);
576 next unless (grep(/out/, @{$e->{DIRECTION}}));
578 $self->ParseOutputArgument($fn, $e,
585 if (defined($fn->{RETURN_TYPE})) {
586 $self->pidl("/* Copy result */");
587 $self->pidl("state->orig.out.result = state->tmp.out.result;");
591 $self->pidl("/* Reset temporary structure */");
592 $self->pidl("ZERO_STRUCT(state->tmp);");
595 $self->pidl("tevent_req_done(req);");
601 sub ParseFunction_Recv($$$$)
603 my ($self, $if, $fn, $name) = @_;
606 my $state_str = "struct dcerpc_$name\_state";
607 my $fn_str = "NTSTATUS dcerpc_$name\_recv";
608 my $pad = genpad($fn_str);
610 $fn_args .= "struct tevent_req *req,\n" . $pad . "TALLOC_CTX *mem_ctx";
612 if (defined($fn->{RETURN_TYPE})) {
613 $fn_args .= ",\n" . $pad . mapTypeName($fn->{RETURN_TYPE}). " *result";
616 $self->fn_declare("$fn_str($fn_args)");
619 $self->pidl("$state_str *state = tevent_req_data(");
620 $self->pidl("\treq, $state_str);");
621 $self->pidl("NTSTATUS status;");
623 $self->pidl("if (tevent_req_is_nterror(req, &status)) {");
625 $self->pidl("tevent_req_received(req);");
626 $self->pidl("return status;");
631 $self->pidl("/* Steal possible out parameters to the callers context */");
632 $self->pidl("talloc_steal(mem_ctx, state->out_mem_ctx);");
635 if (defined($fn->{RETURN_TYPE})) {
636 $self->pidl("/* Return result */");
637 $self->pidl("*result = state->orig.out.result;");
641 $self->pidl("tevent_req_received(req);");
642 $self->pidl("return NT_STATUS_OK;");
648 sub ParseFunction_Sync($$$$)
650 my ($self, $if, $fn, $name) = @_;
652 if ($self->ParseFunctionHasPipes($fn)) {
653 $self->pidl_both("/*");
654 $self->pidl_both(" * The following function is skipped because");
655 $self->pidl_both(" * it uses pipes:");
656 $self->pidl_both(" *");
657 $self->pidl_both(" * dcerpc_$name()");
658 $self->pidl_both(" */");
659 $self->pidl_both("");
663 my $uname = uc $name;
665 my $fn_str = "NTSTATUS dcerpc_$name";
666 my $pad = genpad($fn_str);
668 $fn_args .= "struct dcerpc_binding_handle *h,\n" . $pad . "TALLOC_CTX *mem_ctx";
670 foreach (@{$fn->{ELEMENTS}}) {
671 my $dir = ElementDirection($_);
672 my $prop = HeaderProperties($_->{PROPERTIES}, ["in", "out"]);
673 $fn_args .= ",\n" . $pad . DeclLong($_, "_") . " /* $dir $prop */";
676 if (defined($fn->{RETURN_TYPE})) {
677 $fn_args .= ",\n" . $pad . mapTypeName($fn->{RETURN_TYPE}). " *result";
680 $self->fn_declare("$fn_str($fn_args)");
683 $self->pidl("struct $name r;");
684 $self->pidl("NTSTATUS status;");
687 $self->pidl("/* In parameters */");
688 foreach my $e (@{$fn->{ELEMENTS}}) {
689 next unless (grep(/in/, @{$e->{DIRECTION}}));
691 $self->ParseCopyArgument($fn, $e, "r.in.", "_");
695 $self->pidl("/* Out parameters */");
696 foreach my $e (@{$fn->{ELEMENTS}}) {
697 next unless grep(/out/, @{$e->{DIRECTION}});
699 $self->ParseCopyArgument($fn, $e, "r.out.", "_");
703 if (defined($fn->{RETURN_TYPE})) {
704 $self->pidl("/* Result */");
705 $self->pidl("ZERO_STRUCT(r.out.result);");
709 $self->pidl("status = dcerpc_$name\_r(h, mem_ctx, &r);");
710 $self->pidl("if (!NT_STATUS_IS_OK(status)) {");
712 $self->pidl("return status;");
717 $self->pidl("/* Return variables */");
718 foreach my $e (@{$fn->{ELEMENTS}}) {
719 next if ContainsPipe($e, $e->{LEVELS}[0]);
720 next unless (grep(/out/, @{$e->{DIRECTION}}));
722 $self->ParseOutputArgument($fn, $e, "r.", "_", "sync");
726 $self->pidl("/* Return result */");
727 if ($fn->{RETURN_TYPE}) {
728 $self->pidl("*result = r.out.result;");
732 $self->pidl("return NT_STATUS_OK;");
739 #####################################################################
741 sub ParseFunction($$$)
743 my ($self, $if, $fn) = @_;
745 if ($self->ParseFunctionHasPipes($fn)) {
746 $self->pidl_both("/*");
747 $self->pidl_both(" * The following function is skipped because");
748 $self->pidl_both(" * it uses pipes:");
749 $self->pidl_both(" *");
750 $self->pidl_both(" * dcerpc_$fn->{NAME}_r_send()");
751 $self->pidl_both(" * dcerpc_$fn->{NAME}_r_recv()");
752 $self->pidl_both(" * dcerpc_$fn->{NAME}_r()");
753 $self->pidl_both(" *");
754 $self->pidl_both(" * dcerpc_$fn->{NAME}_send()");
755 $self->pidl_both(" * dcerpc_$fn->{NAME}_recv()");
756 $self->pidl_both(" * dcerpc_$fn->{NAME}()");
757 $self->pidl_both(" */");
758 $self->pidl_both("");
759 warning($fn->{ORIGINAL}, "$fn->{NAME}: dcerpc client does not support pipe yet");
763 $self->ParseFunction_r_State($if, $fn, $fn->{NAME});
764 $self->ParseFunction_r_Send($if, $fn, $fn->{NAME});
765 $self->ParseFunction_r_Done($if, $fn, $fn->{NAME});
766 $self->ParseFunction_r_Recv($if, $fn, $fn->{NAME});
767 $self->ParseFunction_r_Sync($if, $fn, $fn->{NAME});
769 foreach my $e (@{$fn->{ELEMENTS}}) {
770 next unless (grep(/out/, @{$e->{DIRECTION}}));
772 my $reason = "is not a pointer or array";
774 # TODO: make this fatal at NDR level
775 if ($e->{LEVELS}[0]->{TYPE} eq "POINTER") {
776 if ($e->{LEVELS}[1]->{TYPE} eq "DATA" and
777 $e->{LEVELS}[1]->{DATA_TYPE} eq "string") {
778 $reason = "is a pointer to type 'string'";
779 } elsif ($e->{LEVELS}[1]->{TYPE} eq "ARRAY" and
780 $e->{LEVELS}[1]->{IS_ZERO_TERMINATED}) {
782 } elsif ($e->{LEVELS}[1]->{TYPE} eq "ARRAY" and
783 not defined($e->{LEVELS}[1]->{SIZE_IS})) {
784 $reason = "is a pointer to an unsized array";
789 if ($e->{LEVELS}[0]->{TYPE} eq "ARRAY") {
790 if (not defined($e->{LEVELS}[0]->{SIZE_IS})) {
791 $reason = "is an unsized array";
797 $self->pidl_both("/*");
798 $self->pidl_both(" * The following functions are skipped because");
799 $self->pidl_both(" * an [out] argument $e->{NAME} $reason:");
800 $self->pidl_both(" *");
801 $self->pidl_both(" * dcerpc_$fn->{NAME}_send()");
802 $self->pidl_both(" * dcerpc_$fn->{NAME}_recv()");
803 $self->pidl_both(" * dcerpc_$fn->{NAME}()");
804 $self->pidl_both(" */");
805 $self->pidl_both("");
807 error($e->{ORIGINAL}, "$fn->{NAME}: [out] argument '$e->{NAME}' $reason, skip client functions");
811 $self->ParseFunction_State($if, $fn, $fn->{NAME});
812 $self->ParseFunction_Send($if, $fn, $fn->{NAME});
813 $self->ParseFunction_Done($if, $fn, $fn->{NAME});
814 $self->ParseFunction_Recv($if, $fn, $fn->{NAME});
815 $self->ParseFunction_Sync($if, $fn, $fn->{NAME});
822 #####################################################################
823 # parse the interface definitions
824 sub ParseInterface($$)
826 my ($self, $if) = @_;
827 my $ifu = uc($if->{NAME});
829 $self->pidl_hdr("#ifndef _HEADER_RPC_$if->{NAME}");
830 $self->pidl_hdr("#define _HEADER_RPC_$if->{NAME}");
833 if (defined $if->{PROPERTIES}->{uuid}) {
834 $self->pidl_hdr("extern const struct ndr_interface_table ndr_table_$if->{NAME};");
838 $self->pidl("/* $if->{NAME} - client functions generated by pidl */");
841 foreach my $fn (@{$if->{FUNCTIONS}}) {
842 next if defined($done{$fn->{NAME}});
843 next if has_property($fn, "noopnum");
844 next if has_property($fn, "todo");
845 $self->ParseFunction($if, $fn);
846 $done{$fn->{NAME}} = 1;
849 $self->pidl_hdr("#endif /* _HEADER_RPC_$if->{NAME} */");
854 my($self,$ndr,$header,$ndr_header,$client_header) = @_;
856 $self->pidl("/* client functions auto-generated by pidl */");
859 $self->pidl("#include \"includes.h\"");
861 $self->pidl("#ifndef _GNU_SOURCE");
862 $self->pidl("#define _GNU_SOURCE");
863 $self->pidl("#endif");
864 $self->pidl("#include <stdio.h>");
865 $self->pidl("#include <stdbool.h>");
866 $self->pidl("#include <stdlib.h>");
867 $self->pidl("#include <stdint.h>");
868 $self->pidl("#include <stdarg.h>");
869 $self->pidl("#include <string.h>");
870 $self->pidl("#include <core/ntstatus.h>");
872 $self->pidl("#include <tevent.h>");
873 $self->pidl(choose_header("lib/util/tevent_ntstatus.h", "util/tevent_ntstatus.h")."");
874 $self->pidl("#include \"$ndr_header\"");
875 $self->pidl("#include \"$client_header\"");
878 $self->pidl_hdr(choose_header("librpc/rpc/dcerpc.h", "dcerpc.h")."");
879 $self->pidl_hdr("#include \"$header\"");
881 foreach my $x (@{$ndr}) {
882 ($x->{TYPE} eq "INTERFACE") && $self->ParseInterface($x);
885 return ($self->{res},$self->{res_hdr});