1 ###################################################
2 # EJS function wrapper generator
3 # Copyright jelmer@samba.org 2005
4 # Copyright Andrew Tridgell 2005
5 # released under the GNU GPL
7 package Parse::Pidl::Samba4::EJS;
11 @EXPORT_OK = qw(check_null_pointer fn_declare TypeFunctionName);
14 use Parse::Pidl::Typelist qw(typeHasBody);
15 use Parse::Pidl::CUtil qw(get_pointer_to get_value_of);
16 use Parse::Pidl::Util qw(has_property ParseExpr);
17 use Parse::Pidl::NDR qw(GetPrevLevel GetNextLevel);
18 use Parse::Pidl::Samba4::Header qw(GenerateStructEnv GenerateFunctionInEnv
19 GenerateFunctionOutEnv);
21 use vars qw($VERSION);
26 my $self = { res => "", res_hdr => "", tabs => "", constants => {}};
33 $self->{res_hdr} .= shift;
40 $self->{res} .= $self->{tabs};
49 $self->{tabs} .= "\t";
55 $self->{tabs} = substr($self->{tabs}, 0, -1);
58 #####################################################################
59 # check that a variable we get from ParseExpr isn't a null pointer
60 sub check_null_pointer($$)
62 my ($self, $size) = @_;
64 my $size2 = substr($size, 1);
65 $self->pidl("if ($size2 == NULL) return NT_STATUS_INVALID_PARAMETER_MIX;");
69 #####################################################################
70 # work out is a parse function should be declared static or not
73 my ($self,$fn,$decl) = @_;
75 if (has_property($fn, "public")) {
76 $self->pidl_hdr("$decl;\n");
77 $self->pidl("_PUBLIC_ $decl");
79 $self->pidl("static $decl");
83 ###########################
84 # pull a scalar element
85 sub EjsPullScalar($$$$$$$)
87 my ($self, $e, $l, $var, $name, $env) = @_;
89 return if (has_property($e, "value"));
91 if (ref($e->{TYPE}) eq "HASH" and not defined($e->{TYPE}->{NAME})) {
92 $self->EjsTypePull($e->{TYPE}, $var);
94 my $pl = Parse::Pidl::NDR::GetPrevLevel($e, $l);
95 $var = get_pointer_to($var);
96 # have to handle strings specially :(
97 if (Parse::Pidl::Typelist::scalar_is_reference($e->{TYPE})
98 and (defined($pl) and $pl->{TYPE} eq "POINTER")) {
99 $var = get_pointer_to($var);
103 if (ref($e->{TYPE}) eq "HASH") {
104 $t = "$e->{TYPE}->{TYPE}_$e->{TYPE}->{NAME}";
108 $self->pidl("EJS_CHECK(ejs_pull_$t(ejs, v, $name, $var));");
112 ###########################
113 # pull a pointer element
114 sub EjsPullPointer($$$$$$)
116 my ($self, $e, $l, $var, $name, $env) = @_;
117 $self->pidl("if (ejs_pull_null(ejs, v, $name)) {");
119 if ($l->{POINTER_TYPE} eq "ref") {
120 $self->pidl("return NT_STATUS_INVALID_PARAMETER_MIX;");
122 $self->pidl("$var = NULL;");
125 $self->pidl("} else {");
127 $self->pidl("EJS_ALLOC(ejs, $var);");
128 $var = get_value_of($var);
129 $self->EjsPullElement($e, GetNextLevel($e, $l), $var, $name, $env);
134 ###########################
135 # pull a string element
136 sub EjsPullString($$$$$$)
138 my ($self, $e, $l, $var, $name, $env) = @_;
139 my $pl = GetPrevLevel($e, $l);
140 $var = get_pointer_to($var);
141 if (defined($pl) and $pl->{TYPE} eq "POINTER") {
142 $var = get_pointer_to($var);
144 $self->pidl("EJS_CHECK(ejs_pull_string(ejs, v, $name, $var));");
147 ###########################
148 # pull an array element
149 sub EjsPullArray($$$$$$)
151 my ($self, $e, $l, $var, $name, $env) = @_;
152 my $nl = GetNextLevel($e, $l);
153 my $length = ParseExpr($l->{LENGTH_IS}, $env, $e);
154 my $size = ParseExpr($l->{SIZE_IS}, $env, $e);
155 my $pl = GetPrevLevel($e, $l);
156 if ($pl && $pl->{TYPE} eq "POINTER") {
157 $var = get_pointer_to($var);
159 # uint8 arrays are treated as data blobs
160 if ($nl->{TYPE} eq 'DATA' && $e->{TYPE} eq 'uint8') {
161 if (!$l->{IS_FIXED}) {
162 $self->check_null_pointer($size);
163 $self->pidl("EJS_ALLOC_N(ejs, $var, $size);");
165 $self->check_null_pointer($length);
166 $self->pidl("ejs_pull_array_uint8(ejs, v, $name, $var, $length);");
169 my $avar = $var . "[i]";
172 $self->pidl("uint32_t i;");
173 if (!$l->{IS_FIXED}) {
174 $self->pidl("EJS_ALLOC_N(ejs, $var, $size);");
176 $self->pidl("for (i=0;i<$length;i++) {");
178 $self->pidl("char *id = talloc_asprintf(ejs, \"%s.%u\", $name, i);");
179 $self->EjsPullElement($e, $nl, $avar, "id", $env);
180 $self->pidl("talloc_free(id);");
183 $self->pidl("ejs_push_uint32(ejs, v, $name \".length\", &i);");
188 ###########################
189 # pull a switch element
190 sub EjsPullSwitch($$$$$$)
192 my ($self, $e, $l, $var, $name, $env) = @_;
193 my $switch_var = ParseExpr($l->{SWITCH_IS}, $env, $e);
194 $self->pidl("ejs_set_switch(ejs, $switch_var);");
195 $self->EjsPullElement($e, GetNextLevel($e, $l), $var, $name, $env);
198 ###########################
199 # pull a structure element
200 sub EjsPullElement($$$$$$)
202 my ($self, $e, $l, $var, $name, $env) = @_;
203 if (($l->{TYPE} eq "POINTER")) {
204 $self->EjsPullPointer($e, $l, $var, $name, $env);
205 } elsif (has_property($e, "charset")) {
206 $self->EjsPullString($e, $l, $var, $name, $env);
207 } elsif ($l->{TYPE} eq "ARRAY") {
208 $self->EjsPullArray($e, $l, $var, $name, $env);
209 } elsif ($l->{TYPE} eq "DATA") {
210 $self->EjsPullScalar($e, $l, $var, $name, $env);
211 } elsif (($l->{TYPE} eq "SWITCH")) {
212 $self->EjsPullSwitch($e, $l, $var, $name, $env);
214 $self->pidl("return ejs_panic(ejs, \"unhandled pull type $l->{TYPE}\");");
218 #############################################
219 # pull a structure/union element at top level
220 sub EjsPullElementTop($$$)
222 my ($self, $e, $env) = @_;
223 my $l = $e->{LEVELS}[0];
224 my $var = ParseExpr($e->{NAME}, $env, $e);
225 my $name = "\"$e->{NAME}\"";
226 $self->EjsPullElement($e, $l, $var, $name, $env);
229 ###########################
231 sub EjsStructPull($$$)
233 my ($self, $d, $varname) = @_;
234 my $env = GenerateStructEnv($d, $varname);
235 $self->pidl("EJS_CHECK(ejs_pull_struct_start(ejs, &v, name));");
236 foreach my $e (@{$d->{ELEMENTS}}) {
237 $self->EjsPullElementTop($e, $env);
241 ###########################
243 sub EjsUnionPull($$$)
245 my ($self, $d, $varname) = @_;
246 my $have_default = 0;
247 $self->pidl("EJS_CHECK(ejs_pull_struct_start(ejs, &v, name));");
248 $self->pidl("switch (ejs->switch_var) {");
250 foreach my $e (@{$d->{ELEMENTS}}) {
251 if ($e->{CASE} eq "default") {
254 $self->pidl("$e->{CASE}:");
256 if ($e->{TYPE} ne "EMPTY") {
257 $self->EjsPullElementTop($e, { $e->{NAME} => "$varname->$e->{NAME}"});
259 $self->pidl("break;");
262 if (! $have_default) {
263 $self->pidl("default:");
265 $self->pidl("return ejs_panic(ejs, \"Bad switch value\");");
272 ##############################################
273 # put the enum elements in the constants array
274 sub EjsEnumConstant($$)
277 return unless (defined($d->{ELEMENTS}));
279 foreach my $e (@{$d->{ELEMENTS}}) {
282 if ($el =~ /^(.*)=\s*(.*)\s*$/) {
286 $self->{constants}->{$el} = $v;
291 ###########################
295 my ($self, $d, $varname) = @_;
296 $self->EjsEnumConstant($d);
297 $self->pidl("unsigned e;");
298 $self->pidl("EJS_CHECK(ejs_pull_enum(ejs, v, name, &e));");
299 $self->pidl("*$varname = e;");
302 ###########################
304 sub EjsBitmapPull($$$)
306 my ($self, $d, $varname) = @_;
307 my $type_fn = $d->{BASE_TYPE};
308 $self->pidl("EJS_CHECK(ejs_pull_$type_fn(ejs, v, name, $varname));");
311 sub EjsTypePullFunction($$$)
313 sub EjsTypePullFunction($$$);
314 my ($self, $d, $name) = @_;
315 return if (has_property($d, "noejs"));
317 if ($d->{TYPE} eq "TYPEDEF") {
318 $self->EjsTypePullFunction($d->{DATA}, $name);
322 if ($d->{TYPE} eq "STRUCT") {
323 $self->fn_declare($d, "NTSTATUS ejs_pull_$name(struct ejs_rpc *ejs, struct MprVar *v, const char *name, struct $name *r)");
324 } elsif ($d->{TYPE} eq "UNION") {
325 $self->fn_declare($d, "NTSTATUS ejs_pull_$name(struct ejs_rpc *ejs, struct MprVar *v, const char *name, union $name *r)");
326 } elsif ($d->{TYPE} eq "ENUM") {
327 $self->fn_declare($d, "NTSTATUS ejs_pull_$name(struct ejs_rpc *ejs, struct MprVar *v, const char *name, enum $name *r)");
328 } elsif ($d->{TYPE} eq "BITMAP") {
329 my($type_decl) = Parse::Pidl::Typelist::mapTypeName($d->{BASE_TYPE});
330 $self->fn_declare($d, "NTSTATUS ejs_pull_$name(struct ejs_rpc *ejs, struct MprVar *v, const char *name, $type_decl *r)");
335 $self->EjsTypePull($d, "r");
337 $self->pidl("return NT_STATUS_OK;");
344 my ($self, $d, $varname) = @_;
345 if ($d->{TYPE} eq 'STRUCT') {
346 $self->EjsStructPull($d, $varname);
347 } elsif ($d->{TYPE} eq 'UNION') {
348 $self->EjsUnionPull($d, $varname);
349 } elsif ($d->{TYPE} eq 'ENUM') {
350 $self->EjsEnumPull($d, $varname);
351 } elsif ($d->{TYPE} eq 'BITMAP') {
352 $self->EjsBitmapPull($d, $varname);
354 warn "Unhandled pull $varname of type $d->{TYPE}";
358 #####################
359 # generate a function
360 sub EjsPullFunction($$)
363 my $env = GenerateFunctionInEnv($d);
364 my $name = $d->{NAME};
366 $self->pidl("\nstatic NTSTATUS ejs_pull_$name(struct ejs_rpc *ejs, struct MprVar *v, struct $name *r)");
369 $self->pidl("EJS_CHECK(ejs_pull_struct_start(ejs, &v, \"input\"));");
371 # we pull non-array elements before array elements as arrays
372 # may have length_is() or size_is() properties that depend
373 # on the non-array elements
374 foreach my $e (@{$d->{ELEMENTS}}) {
375 next unless (grep(/in/, @{$e->{DIRECTION}}));
376 next if (has_property($e, "length_is") || has_property($e, "size_is"));
377 $self->EjsPullElementTop($e, $env);
380 foreach my $e (@{$d->{ELEMENTS}}) {
381 next unless (grep(/in/, @{$e->{DIRECTION}}));
382 next unless (has_property($e, "length_is") || has_property($e, "size_is"));
383 $self->EjsPullElementTop($e, $env);
386 $self->pidl("return NT_STATUS_OK;");
391 ###########################
392 # push a scalar element
393 sub EjsPushScalar($$$$$$)
395 my ($self, $e, $l, $var, $name, $env) = @_;
397 if (ref($e->{TYPE}) eq "HASH" and not defined($e->{TYPE}->{NAME})) {
398 $self->EjsTypePush($e->{TYPE}, get_pointer_to($var));
400 # have to handle strings specially :(
401 my $pl = GetPrevLevel($e, $l);
403 if ((not Parse::Pidl::Typelist::scalar_is_reference($e->{TYPE}))
404 or (defined($pl) and $pl->{TYPE} eq "POINTER")) {
405 $var = get_pointer_to($var);
408 $self->pidl("EJS_CHECK(".TypeFunctionName("ejs_push", $e->{TYPE})."(ejs, v, $name, $var));");
412 ###########################
413 # push a string element
414 sub EjsPushString($$$$$$)
416 my ($self, $e, $l, $var, $name, $env) = @_;
417 my $pl = GetPrevLevel($e, $l);
418 if (defined($pl) and $pl->{TYPE} eq "POINTER") {
419 $var = get_pointer_to($var);
421 $self->pidl("EJS_CHECK(ejs_push_string(ejs, v, $name, $var));");
424 ###########################
425 # push a pointer element
426 sub EjsPushPointer($$$$$$)
428 my ($self, $e, $l, $var, $name, $env) = @_;
429 $self->pidl("if (NULL == $var) {");
431 if ($l->{POINTER_TYPE} eq "ref") {
432 $self->pidl("return NT_STATUS_INVALID_PARAMETER_MIX;");
434 $self->pidl("EJS_CHECK(ejs_push_null(ejs, v, $name));");
437 $self->pidl("} else {");
439 $var = get_value_of($var);
440 $self->EjsPushElement($e, GetNextLevel($e, $l), $var, $name, $env);
445 ###########################
446 # push a switch element
447 sub EjsPushSwitch($$$$$$)
449 my ($self, $e, $l, $var, $name, $env) = @_;
450 my $switch_var = ParseExpr($l->{SWITCH_IS}, $env, $e);
451 $self->pidl("ejs_set_switch(ejs, $switch_var);");
452 $self->EjsPushElement($e, GetNextLevel($e, $l), $var, $name, $env);
455 ###########################
456 # push an array element
457 sub EjsPushArray($$$$$$)
459 my ($self, $e, $l, $var, $name, $env) = @_;
460 my $nl = GetNextLevel($e, $l);
461 my $length = ParseExpr($l->{LENGTH_IS}, $env, $e);
462 my $pl = GetPrevLevel($e, $l);
463 if ($pl && $pl->{TYPE} eq "POINTER") {
464 $var = get_pointer_to($var);
466 # uint8 arrays are treated as data blobs
467 if ($nl->{TYPE} eq 'DATA' && $e->{TYPE} eq 'uint8') {
468 $self->check_null_pointer($length);
469 $self->pidl("ejs_push_array_uint8(ejs, v, $name, $var, $length);");
472 my $avar = $var . "[i]";
475 $self->pidl("uint32_t i;");
476 $self->pidl("for (i=0;i<$length;i++) {");
478 $self->pidl("const char *id = talloc_asprintf(ejs, \"%s.%u\", $name, i);");
479 $self->EjsPushElement($e, $nl, $avar, "id", $env);
482 $self->pidl("ejs_push_uint32(ejs, v, $name \".length\", &i);");
487 ################################
488 # push a structure/union element
489 sub EjsPushElement($$$$$$)
491 my ($self, $e, $l, $var, $name, $env) = @_;
492 if (($l->{TYPE} eq "POINTER")) {
493 $self->EjsPushPointer($e, $l, $var, $name, $env);
494 } elsif (has_property($e, "charset")) {
495 $self->EjsPushString($e, $l, $var, $name, $env);
496 } elsif ($l->{TYPE} eq "ARRAY") {
497 $self->EjsPushArray($e, $l, $var, $name, $env);
498 } elsif ($l->{TYPE} eq "DATA") {
499 $self->EjsPushScalar($e, $l, $var, $name, $env);
500 } elsif (($l->{TYPE} eq "SWITCH")) {
501 $self->EjsPushSwitch($e, $l, $var, $name, $env);
503 $self->pidl("return ejs_panic(ejs, \"unhandled push type $l->{TYPE}\");");
507 #############################################
508 # push a structure/union element at top level
509 sub EjsPushElementTop($$$)
511 my ($self, $e, $env) = @_;
512 my $l = $e->{LEVELS}[0];
513 my $var = ParseExpr($e->{NAME}, $env, $e);
514 my $name = "\"$e->{NAME}\"";
515 $self->EjsPushElement($e, $l, $var, $name, $env);
518 ###########################
520 sub EjsStructPush($$$)
522 my ($self, $d, $varname) = @_;
523 my $env = GenerateStructEnv($d, $varname);
524 $self->pidl("EJS_CHECK(ejs_push_struct_start(ejs, &v, name));");
525 foreach my $e (@{$d->{ELEMENTS}}) {
526 $self->EjsPushElementTop($e, $env);
530 ###########################
532 sub EjsUnionPush($$$)
534 my ($self, $d, $varname) = @_;
535 my $have_default = 0;
536 $self->pidl("EJS_CHECK(ejs_push_struct_start(ejs, &v, name));");
537 $self->pidl("switch (ejs->switch_var) {");
539 foreach my $e (@{$d->{ELEMENTS}}) {
540 if ($e->{CASE} eq "default") {
543 $self->pidl("$e->{CASE}:");
545 if ($e->{TYPE} ne "EMPTY") {
546 $self->EjsPushElementTop($e, { $e->{NAME} => "$varname->$e->{NAME}"} );
548 $self->pidl("break;");
551 if (! $have_default) {
552 $self->pidl("default:");
554 $self->pidl("return ejs_panic(ejs, \"Bad switch value\");");
561 ###########################
565 my ($self, $d, $varname) = @_;
566 $self->EjsEnumConstant($d);
567 $self->pidl("unsigned e = ".get_value_of($varname).";");
568 $self->pidl("EJS_CHECK(ejs_push_enum(ejs, v, name, &e));");
571 ###########################
573 sub EjsBitmapPush($$$)
575 my ($self, $d, $varname) = @_;
576 return unless (defined($d->{ELEMENTS}));
577 my $type_fn = $d->{BASE_TYPE};
578 # put the bitmap elements in the constants array
579 foreach my $e (@{$d->{ELEMENTS}}) {
580 if ($e =~ /^(\w*)\s*(.*)\s*$/) {
583 $self->{constants}->{$bname} = $v;
586 $self->pidl("EJS_CHECK(ejs_push_$type_fn(ejs, v, name, $varname));");
589 sub EjsTypePushFunction($$$)
591 sub EjsTypePushFunction($$$);
592 my ($self, $d, $name) = @_;
593 return if (has_property($d, "noejs"));
597 if ($dt->{TYPE} eq "TYPEDEF") {
600 if ($dt->{TYPE} eq "STRUCT") {
601 $var = "const struct $name *r";
602 } elsif ($dt->{TYPE} eq "UNION") {
603 $var = "const union $name *r";
604 } elsif ($dt->{TYPE} eq "ENUM") {
605 $var = "const enum $name *r";
606 } elsif ($dt->{TYPE} eq "BITMAP") {
607 my($type_decl) = Parse::Pidl::Typelist::mapTypeName($dt->{BASE_TYPE});
608 $var = "const $type_decl *r";
610 $self->fn_declare($d, "NTSTATUS ".TypeFunctionName("ejs_push", $d) . "(struct ejs_rpc *ejs, struct MprVar *v, const char *name, $var)");
613 $self->EjsTypePush($d, "r");
614 $self->pidl("return NT_STATUS_OK;");
621 sub EjsTypePush($$$);
622 my ($self, $d, $varname) = @_;
624 if ($d->{TYPE} eq 'STRUCT') {
625 $self->EjsStructPush($d, $varname);
626 } elsif ($d->{TYPE} eq 'UNION') {
627 $self->EjsUnionPush($d, $varname);
628 } elsif ($d->{TYPE} eq 'ENUM') {
629 $self->EjsEnumPush($d, $varname);
630 } elsif ($d->{TYPE} eq 'BITMAP') {
631 $self->EjsBitmapPush($d, $varname);
632 } elsif ($d->{TYPE} eq 'TYPEDEF') {
633 $self->EjsTypePush($d->{DATA}, $varname);
635 warn "Unhandled push $varname of type $d->{TYPE}";
639 #####################
640 # generate a function
641 sub EjsPushFunction($$)
644 my $env = GenerateFunctionOutEnv($d);
646 $self->pidl("\nstatic NTSTATUS ejs_push_$d->{NAME}(struct ejs_rpc *ejs, struct MprVar *v, const struct $d->{NAME} *r)");
649 $self->pidl("EJS_CHECK(ejs_push_struct_start(ejs, &v, \"output\"));");
651 foreach my $e (@{$d->{ELEMENTS}}) {
652 next unless (grep(/out/, @{$e->{DIRECTION}}));
653 $self->EjsPushElementTop($e, $env);
656 if ($d->{RETURN_TYPE}) {
657 $self->pidl("EJS_CHECK(".TypeFunctionName("ejs_push", $d->{RETURN_TYPE})."(ejs, v, \"result\", &r->out.result));");
660 $self->pidl("return NT_STATUS_OK;");
665 #################################
666 # generate a ejs mapping function
669 my ($self, $d, $iface) = @_;
670 my $name = $d->{NAME};
671 my $callnum = uc("NDR_$name");
672 my $table = "&ndr_table_$iface";
674 $self->pidl("static int ejs_$name(int eid, int argc, struct MprVar **argv)");
677 $self->pidl("return ejs_rpc_call(eid, argc, argv, $table, $callnum, (ejs_pull_function_t)ejs_pull_$name, (ejs_push_function_t)ejs_push_$name);");
686 my ($self, $const) = @_;
687 $self->{constants}->{$const->{NAME}} = $const->{VALUE};
697 $self->pidl_hdr("#include \"librpc/gen_ndr/ndr_$_\_ejs\.h\"\n");
701 #####################################################################
702 # parse the interface definitions
703 sub EjsInterface($$$)
705 my($self,$interface,$needed) = @_;
707 my $name = $interface->{NAME};
709 $self->pidl_hdr("#ifndef _HEADER_EJS_$interface->{NAME}\n");
710 $self->pidl_hdr("#define _HEADER_EJS_$interface->{NAME}\n\n");
712 $self->pidl_hdr("\n");
714 foreach my $d (@{$interface->{TYPES}}) {
715 next unless (typeHasBody($d));
716 ($needed->{TypeFunctionName("ejs_push", $d)}) && $self->EjsTypePushFunction($d, $d->{NAME});
717 ($needed->{TypeFunctionName("ejs_pull", $d)}) && $self->EjsTypePullFunction($d, $d->{NAME});
720 foreach my $d (@{$interface->{FUNCTIONS}}) {
721 next if not defined($d->{OPNUM});
722 next if has_property($d, "noejs");
724 $self->EjsPullFunction($d);
725 $self->EjsPushFunction($d);
726 $self->EjsFunction($d, $name);
728 push (@fns, $d->{NAME});
731 foreach my $d (@{$interface->{CONSTS}}) {
735 $self->pidl("static int ejs_$name\_init(int eid, int argc, struct MprVar **argv)");
738 $self->pidl("struct MprVar *obj = mprInitObject(eid, \"$name\", argc, argv);");
740 $self->pidl("mprSetCFunction(obj, \"$_\", ejs_$_);");
742 foreach my $v (keys %{$self->{constants}}) {
743 my $value = $self->{constants}->{$v};
744 if (substr($value, 0, 1) eq "\"") {
745 $self->pidl("mprSetVar(obj, \"$v\", mprString($value));");
747 $self->pidl("mprSetVar(obj, \"$v\", mprCreateNumberVar($value));");
750 $self->pidl("return ejs_rpc_init(obj, \"$name\");");
754 $self->pidl("NTSTATUS ejs_init_$name(void)");
757 $self->pidl("ejsDefineCFunction(-1, \"$name\_init\", ejs_$name\_init, NULL, MPR_VAR_SCRIPT_HANDLE);");
758 $self->pidl("return NT_STATUS_OK;");
762 $self->pidl_hdr("\n");
763 $self->pidl_hdr("#endif /* _HEADER_EJS_$interface->{NAME} */\n");
766 #####################################################################
767 # parse a parsed IDL into a C header
770 my($self,$ndr,$hdr) = @_;
773 $ejs_hdr =~ s/.h$/_ejs.h/;
775 $self->pidl_hdr("/* header auto-generated by pidl */\n\n");
778 /* EJS wrapper functions auto-generated by pidl */
779 #include \"includes.h\"
780 #include \"librpc/rpc/dcerpc.h\"
781 #include \"lib/appweb/ejs/ejs.h\"
782 #include \"scripting/ejs/ejsrpc.h\"
783 #include \"scripting/ejs/smbcalls.h\"
784 #include \"librpc/gen_ndr/ndr_misc_ejs.h\"
786 #include \"$ejs_hdr\"
792 foreach my $x (@{$ndr}) {
793 ($x->{TYPE} eq "INTERFACE") && NeededInterface($x, \%needed);
796 foreach my $x (@$ndr) {
797 ($x->{TYPE} eq "INTERFACE") && $self->EjsInterface($x, \%needed);
798 ($x->{TYPE} eq "IMPORT") && $self->EjsImport(@{$x->{PATHS}});
801 return ($self->{res_hdr}, $self->{res});
804 sub NeededFunction($$)
806 my ($fn,$needed) = @_;
808 $needed->{"ejs_pull_$fn->{NAME}"} = 1;
809 $needed->{"ejs_push_$fn->{NAME}"} = 1;
811 foreach (@{$fn->{ELEMENTS}}) {
812 next if (has_property($_, "subcontext")); #FIXME: Support subcontexts
813 if (grep(/in/, @{$_->{DIRECTION}})) {
814 $needed->{TypeFunctionName("ejs_pull", $_->{TYPE})} = 1;
816 if (grep(/out/, @{$_->{DIRECTION}})) {
817 $needed->{TypeFunctionName("ejs_push", $_->{TYPE})} = 1;
825 my ($t,$needed,$req) = @_;
827 NeededType($t->{DATA}, $needed, $req) if ($t->{TYPE} eq "TYPEDEF");
829 return unless (($t->{TYPE} eq "STRUCT") or ($t->{TYPE} eq "UNION"));
831 return unless(typeHasBody($t));
833 foreach (@{$t->{ELEMENTS}}) {
834 next if (has_property($_, "subcontext")); #FIXME: Support subcontexts
836 if (ref($_->{TYPE}) ne "HASH" or defined($_->{TYPE}->{NAME})) {
837 $needed->{TypeFunctionName("ejs_$req", $_->{TYPE})} = 1;
839 NeededType($_->{TYPE}, $needed, $req) if (ref($_->{TYPE}) eq "HASH");
843 #####################################################################
844 # work out what parse functions are needed
845 sub NeededInterface($$)
847 my ($interface,$needed) = @_;
849 NeededFunction($_, $needed) foreach (@{$interface->{FUNCTIONS}});
851 foreach (reverse @{$interface->{TYPES}}) {
852 if (has_property($_, "public")) {
853 $needed->{TypeFunctionName("ejs_pull", $_)} = not has_property($_, "noejs");
854 $needed->{TypeFunctionName("ejs_push", $_)} = not has_property($_, "noejs");
857 NeededType($_, $needed, "pull") if ($needed->{TypeFunctionName("ejs_pull", $_)});
858 NeededType($_, $needed, "push") if ($needed->{TypeFunctionName("ejs_push", $_)});
862 sub TypeFunctionName($$)
864 my ($prefix, $t) = @_;
866 return "$prefix\_$t->{NAME}" if (ref($t) eq "HASH" and
867 $t->{TYPE} eq "TYPEDEF");
868 return "$prefix\_$t->{TYPE}_$t->{NAME}" if (ref($t) eq "HASH");
869 return "$prefix\_$t";