r26696: Some more minor pidl samba3 client cosmetics.
[samba.git] / source4 / pidl / lib / Parse / Pidl / Samba3 / ClientNDR.pm
index fa629e6101c21a63fbc90fffb494f072e27ec678..01b77c0c9fd6c0e15b545d858ab7950190fa78f0 100644 (file)
 
 package Parse::Pidl::Samba3::ClientNDR;
 
+use Exporter;
+@ISA = qw(Exporter);
+@EXPORT_OK = qw(GenerateFunctionInEnv ParseFunction $res $res_hdr);
+
 use strict;
-use Parse::Pidl::Typelist qw(hasType getType mapType scalar_is_reference);
-use Parse::Pidl::Util qw(has_property ParseExpr is_constant);
+use Parse::Pidl qw(fatal warning);
+use Parse::Pidl::Typelist qw(hasType getType mapTypeName scalar_is_reference);
+use Parse::Pidl::Util qw(has_property is_constant ParseExpr);
 use Parse::Pidl::NDR qw(GetPrevLevel GetNextLevel ContainsDeferred);
 use Parse::Pidl::Samba4 qw(DeclLong);
+use Parse::Pidl::Samba4::NDR::Parser qw(GenerateFunctionInEnv);
 
 use vars qw($VERSION);
 $VERSION = '0.01';
 
-my $res;
-my $res_hdr;
-my $tabs = "";
-sub indent() { $tabs.="\t"; }
-sub deindent() { $tabs = substr($tabs, 1); }
-sub pidl($) { $res .= $tabs.(shift)."\n"; }
-sub pidl_hdr($) { $res_hdr .= (shift)."\n"; }
-sub fatal($$) { my ($e,$s) = @_; die("$e->{ORIGINAL}->{FILE}:$e->{ORIGINAL}->{LINE}: $s\n"); }
-sub warning($$) { my ($e,$s) = @_; warn("$e->{ORIGINAL}->{FILE}:$e->{ORIGINAL}->{LINE}: $s\n"); }
-sub fn_declare($) { my ($n) = @_; pidl $n; pidl_hdr "$n;"; }
-
-sub ParseFunction($$)
+sub indent($) { my ($self) = @_; $self->{tabs}.="\t"; }
+sub deindent($) { my ($self) = @_; $self->{tabs} = substr($self->{tabs}, 1); }
+sub pidl($$) { my ($self,$txt) = @_; $self->{res} .= $txt ? "$self->{tabs}$txt\n" : "\n"; }
+sub pidl_hdr($$) { my ($self, $txt) = @_; $self->{res_hdr} .= "$txt\n"; } 
+sub fn_declare($$) { my ($self,$n) = @_; $self->pidl($n); $self->pidl_hdr("$n;"); }
+
+sub new($)
+{
+       my ($class) = shift;
+       my $self = { res => "", res_hdr => "", tabs => "" };
+       bless($self, $class);
+}
+
+sub ParseFunction($$$)
 {
-       my ($if,$fn) = @_;
+       my ($self, $if, $fn) = @_;
 
        my $inargs = "";
        my $defargs = "";
-       my $uif = uc($if->{NAME});
-       my $ufn = "DCERPC_".uc($fn->{NAME});
+       my $uif = uc($if);
+       my $ufn = "NDR_".uc($fn->{NAME});
 
        foreach (@{$fn->{ELEMENTS}}) {
                $defargs .= ", " . DeclLong($_);
        }
-       fn_declare "NTSTATUS rpccli_$fn->{NAME}(struct rpc_pipe_client *cli, TALLOC_CTX *mem_ctx$defargs)";
-       pidl "{";
-       indent;
-       pidl "struct $fn->{NAME} r;";
-       pidl "NTSTATUS status;";
-       pidl "";
-       pidl "/* In parameters */";
+
+       if (defined($fn->{RETURN_TYPE}) && ($fn->{RETURN_TYPE} eq "WERROR")) {
+               $defargs .= ", WERROR *werror";
+       }
+
+       $self->fn_declare("NTSTATUS rpccli_$fn->{NAME}(struct rpc_pipe_client *cli, TALLOC_CTX *mem_ctx$defargs)");
+       $self->pidl("{");
+       $self->indent;
+       $self->pidl("struct $fn->{NAME} r;");
+       $self->pidl("NTSTATUS status;");
+       $self->pidl("");
+       $self->pidl("/* In parameters */");
 
        foreach (@{$fn->{ELEMENTS}}) {
                if (grep(/in/, @{$_->{DIRECTION}})) {
-                       pidl "r.in.$_->{NAME} = $_->{NAME};";
+                       $self->pidl("r.in.$_->{NAME} = $_->{NAME};");
                } 
        }
 
-       pidl "status = cli_do_rpc_ndr(cli, mem_ctx, PI_$uif, $ufn, &r, (ndr_pull_flags_fn_t)ndr_pull_$fn->{NAME}, (ndr_push_flags_fn_t)ndr_push_$fn->{NAME});";
-       pidl "if (NT_STATUS_IS_ERR(status)) {";
-       pidl "\treturn status;";
-       pidl "}";
-       pidl "";
-       pidl "/* Return variables */";
+       $self->pidl("");
+       $self->pidl("if (DEBUGLEVEL >= 10) {");
+       $self->indent;
+       $self->pidl("NDR_PRINT_IN_DEBUG($fn->{NAME}, &r);");
+       $self->deindent;
+       $self->pidl("}");
+       $self->pidl("");
+       $self->pidl("status = cli_do_rpc_ndr(cli, mem_ctx, PI_$uif, &ndr_table_$if, $ufn, &r);");
+       $self->pidl("");
+
+       $self->pidl("if (!NT_STATUS_IS_OK(status)) {");
+       $self->indent;
+       $self->pidl("return status;");
+       $self->deindent;
+       $self->pidl("}");
+
+       $self->pidl("");
+       $self->pidl("if (DEBUGLEVEL >= 10) {");
+       $self->indent;
+       $self->pidl("NDR_PRINT_OUT_DEBUG($fn->{NAME}, &r);");
+       $self->deindent;
+       $self->pidl("}");
+       $self->pidl("");
+       $self->pidl("if (NT_STATUS_IS_ERR(status)) {");
+       $self->indent;
+       $self->pidl("return status;");
+       $self->deindent;
+       $self->pidl("}");
+       $self->pidl("");
+       $self->pidl("/* Return variables */");
        foreach my $e (@{$fn->{ELEMENTS}}) {
                next unless (grep(/out/, @{$e->{DIRECTION}}));
-
-               fatal($e, "[out] argument is not a pointer") if ($e->{LEVELS}[0]->{TYPE} ne "POINTER");
-
-               pidl "*$e->{NAME} = *r.out.$e->{NAME};";
+               my $level = 0;
+
+               fatal($e->{ORIGINAL}, "[out] argument is not a pointer or array") if ($e->{LEVELS}[0]->{TYPE} ne "POINTER" and $e->{LEVELS}[0]->{TYPE} ne "ARRAY");
+
+               if ($e->{LEVELS}[0]->{TYPE} eq "POINTER") {
+                       $level = 1;
+                       if ($e->{LEVELS}[0]->{POINTER_TYPE} ne "ref") {
+                               $self->pidl("if ($e->{NAME} && r.out.$e->{NAME}) {");
+                               $self->indent;
+                       }
+               }
+
+               if ($e->{LEVELS}[$level]->{TYPE} eq "ARRAY") {
+                       # This is a call to GenerateFunctionInEnv intentionally. 
+                       # Since the data is being copied into a user-provided data 
+                       # structure, the user should be able to know the size beforehand 
+                       # to allocate a structure of the right size.
+                       my $env = GenerateFunctionInEnv($fn, "r.");
+                       my $size_is = ParseExpr($e->{LEVELS}[$level]->{SIZE_IS}, $env, $e->{ORIGINAL});
+                       $self->pidl("memcpy($e->{NAME}, r.out.$e->{NAME}, $size_is);");
+               } else {
+                       $self->pidl("*$e->{NAME} = *r.out.$e->{NAME};");
+               }
+
+               if ($e->{LEVELS}[0]->{TYPE} eq "POINTER") {
+                       if ($e->{LEVELS}[0]->{POINTER_TYPE} ne "ref") {
+                               $self->deindent;
+                               $self->pidl("}");
+                       }
+               }
        }
 
-       pidl"";
-       pidl "/* Return result */";
+       $self->pidl("");
+       $self->pidl("/* Return result */");
        if (not $fn->{RETURN_TYPE}) {
-               pidl "return NT_STATUS_OK;";
+               $self->pidl("return NT_STATUS_OK;");
        } elsif ($fn->{RETURN_TYPE} eq "NTSTATUS") {
-               pidl "return r.out.result;";
+               $self->pidl("return r.out.result;");
        } elsif ($fn->{RETURN_TYPE} eq "WERROR") {
-               pidl "return werror_to_ntstatus(r.out.result);";
+               $self->pidl("if (werror) {");
+               $self->indent;
+               $self->pidl("*werror = r.out.result;");
+               $self->deindent;
+               $self->pidl("}");
+               $self->pidl("");
+               $self->pidl("return werror_to_ntstatus(r.out.result);");
        } else {
-               pidl "/* Sorry, don't know how to convert $fn->{RETURN_TYPE} to NTSTATUS */";
-               pidl "return NT_STATUS_OK;";
+               warning($fn->{ORIGINAL}, "Unable to convert $fn->{RETURN_TYPE} to NTSTATUS");
+               $self->pidl("return NT_STATUS_OK;");
        }
 
-       deindent;
-       pidl "}";
-       pidl "";
+       $self->deindent;
+       $self->pidl("}");
+       $self->pidl("");
 }
 
-sub ParseInterface($)
+sub ParseInterface($$)
 {
-       my $if = shift;
+       my ($self, $if) = @_;
 
        my $uif = uc($if->{NAME});
 
-       pidl_hdr "#ifndef __CLI_$uif\__";
-       pidl_hdr "#define __CLI_$uif\__";
-       ParseFunction($if, $_) foreach (@{$if->{FUNCTIONS}});
-       pidl_hdr "#endif /* __CLI_$uif\__ */";
+       $self->pidl_hdr("#ifndef __CLI_$uif\__");
+       $self->pidl_hdr("#define __CLI_$uif\__");
+       $self->ParseFunction($if->{NAME}, $_) foreach (@{$if->{FUNCTIONS}});
+       $self->pidl_hdr("#endif /* __CLI_$uif\__ */");
 }
 
-sub Parse($$$)
+sub Parse($$$$)
 {
-       my($ndr,$header,$ndr_header) = @_;
-
-       $res = "";
-       $res_hdr = "";
-
-       pidl "/*";
-       pidl " * Unix SMB/CIFS implementation.";
-       pidl " * client auto-generated by pidl. DO NOT MODIFY!";
-       pidl " */";
-       pidl "";
-       pidl "#include \"includes.h\"";
-       pidl "#include \"$header\"";
-       pidl_hdr "#include \"$ndr_header\"";
-       pidl "";
+       my($self,$ndr,$header,$ndr_header) = @_;
+
+       $self->pidl("/*");
+       $self->pidl(" * Unix SMB/CIFS implementation.");
+       $self->pidl(" * client auto-generated by pidl. DO NOT MODIFY!");
+       $self->pidl(" */");
+       $self->pidl("");
+       $self->pidl("#include \"includes.h\"");
+       $self->pidl("#include \"$header\"");
+       $self->pidl_hdr("#include \"$ndr_header\"");
+       $self->pidl("");
        
        foreach (@$ndr) {
-               ParseInterface($_) if ($_->{TYPE} eq "INTERFACE");
+               $self->ParseInterface($_) if ($_->{TYPE} eq "INTERFACE");
        }
 
-       return ($res, $res_hdr);
+       return ($self->{res}, $self->{res_hdr});
 }
 
 1;