ldb_map objectClass munging: Don't hard-code 'extensibleObject'.
[samba.git] / source4 / pidl / lib / Parse / Pidl / Samba3 / ServerNDR.pm
1 ###################################################
2 # Samba3 server generator for IDL structures
3 # on top of Samba4 style NDR functions
4 # Copyright jelmer@samba.org 2005-2006
5 # released under the GNU GPL
6
7 package Parse::Pidl::Samba3::ServerNDR;
8
9 use strict;
10 use Parse::Pidl qw(warning fatal);
11 use Parse::Pidl::Typelist qw(mapTypeName scalar_is_reference);
12 use Parse::Pidl::Util qw(ParseExpr has_property is_constant);
13 use Parse::Pidl::NDR qw(GetNextLevel);
14 use Parse::Pidl::Samba4 qw(ElementStars DeclLong);
15 use Parse::Pidl::Samba4::NDR::Parser qw(GenerateFunctionOutEnv);
16
17 use vars qw($VERSION);
18 $VERSION = '0.01';
19
20 my $res;
21 my $res_hdr;
22 my $tabs = "";
23 sub indent() { $tabs.="\t"; }
24 sub deindent() { $tabs = substr($tabs, 1); }
25 sub pidl($) { my ($txt) = @_; $res .= $txt?$tabs.(shift)."\n":"\n"; }
26 sub pidl_hdr($) { $res_hdr .= (shift)."\n"; }
27 sub fn_declare($) { my ($n) = @_; pidl $n; pidl_hdr "$n;"; }
28
29 sub DeclLevel($$)
30 {
31         my ($e, $l) = @_;
32         my $res = "";
33
34         if (has_property($e, "charset")) {
35                 $res .= "const char";
36         } else {
37                 $res .= mapTypeName($e->{TYPE});
38         }
39
40         my $stars = ElementStars($e, $l);
41
42         $res .= " ".$stars unless ($stars eq "");
43
44         return $res;
45 }
46
47 sub AllocOutVar($$$$)
48 {
49         my ($e, $mem_ctx, $name, $env) = @_;
50
51         my $l = $e->{LEVELS}[0];
52
53         # we skip pointer to arrays
54         if ($l->{TYPE} eq "POINTER") {
55                 my $nl = GetNextLevel($e, $l);
56                 $l = $nl if ($nl->{TYPE} eq "ARRAY");
57         }
58
59         # we don't support multi-dimentional arrays yet
60         if ($l->{TYPE} eq "ARRAY") {
61                 my $nl = GetNextLevel($e, $l);
62                 if ($nl->{TYPE} eq "ARRAY") {
63                         fatal($e->{ORIGINAL},"multi-dimentional [out] arrays are not supported!");
64                 }
65         }
66
67         if ($l->{TYPE} eq "ARRAY") {
68                 my $size = ParseExpr($l->{SIZE_IS}, $env, $e);
69                 pidl "$name = talloc_zero_array($mem_ctx, " . DeclLevel($e, 1) . ", $size);";
70         } else {
71                 pidl "$name = talloc_zero($mem_ctx, " . DeclLevel($e, 1) . ");";
72         }
73
74         pidl "if ($name == NULL) {";
75         pidl "\ttalloc_free($mem_ctx);";
76         pidl "\treturn false;";
77         pidl "}";
78         pidl "";
79 }
80
81 sub ParseFunction($$)
82 {
83         my ($if,$fn) = @_;
84
85         my $op = "NDR_".uc($fn->{NAME});
86
87         pidl "static bool api_$fn->{NAME}(pipes_struct *p)";
88         pidl "{";
89         indent;
90         pidl "const struct ndr_interface_call *call;";
91         pidl "struct ndr_pull *pull;";
92         pidl "struct ndr_push *push;";
93         pidl "enum ndr_err_code ndr_err;";
94         pidl "DATA_BLOB blob;";
95         pidl "struct $fn->{NAME} *r;";
96         pidl "";
97         pidl "call = &ndr_table_$if->{NAME}.calls[$op];";
98         pidl "";
99         pidl "r = talloc(NULL, struct $fn->{NAME});";
100         pidl "if (r == NULL) {";
101         pidl "\treturn false;";
102         pidl "}";
103         pidl "";
104         pidl "if (!prs_data_blob(&p->in_data.data, &blob, r)) {";
105         pidl "\ttalloc_free(r);";
106         pidl "\treturn false;";
107         pidl "}";
108         pidl "";
109         pidl "pull = ndr_pull_init_blob(&blob, r);";
110         pidl "if (pull == NULL) {";
111         pidl "\ttalloc_free(r);";
112         pidl "\treturn false;";
113         pidl "}";
114         pidl "";
115         pidl "pull->flags |= LIBNDR_FLAG_REF_ALLOC;";
116         pidl "ndr_err = call->ndr_pull(pull, NDR_IN, r);";
117         pidl "if (!NDR_ERR_CODE_IS_SUCCESS(ndr_err)) {";
118         pidl "\ttalloc_free(r);";
119         pidl "\treturn false;";
120         pidl "}";
121         pidl "";
122         pidl "if (DEBUGLEVEL >= 10) {";
123         pidl "\tNDR_PRINT_IN_DEBUG($fn->{NAME}, r);";
124         pidl "}";
125         pidl "";
126
127         my $env = GenerateFunctionOutEnv($fn);
128         my $hasout = 0;
129         foreach (@{$fn->{ELEMENTS}}) {
130                 if (grep(/out/, @{$_->{DIRECTION}})) { $hasout = 1; }
131         }
132
133         pidl "ZERO_STRUCT(r->out);" if ($hasout);
134
135         my $proto = "_$fn->{NAME}(pipes_struct *p, struct $fn->{NAME} *r";
136         my $ret = "_$fn->{NAME}(p, r";
137         foreach (@{$fn->{ELEMENTS}}) {
138                 my @dir = @{$_->{DIRECTION}};
139                 if (grep(/in/, @dir) and grep(/out/, @dir)) {
140                         pidl "r->out.$_->{NAME} = r->in.$_->{NAME};";
141                 } elsif (grep(/out/, @dir) and not
142                                  has_property($_, "represent_as")) {
143                         AllocOutVar($_, "r", "r->out.$_->{NAME}", $env);
144                 }
145         }
146         $ret .= ")";
147         $proto .= ");";
148
149         if ($fn->{RETURN_TYPE}) {
150                 $ret = "r->out.result = $ret";
151                 $proto = "$fn->{RETURN_TYPE} $proto";
152         } else {
153                 $proto = "void $proto";
154         }
155
156         pidl_hdr "$proto";
157         pidl "$ret;";
158
159         pidl "";
160         pidl "if (p->rng_fault_state) {";
161         pidl "\ttalloc_free(r);";
162         pidl "\t/* Return true here, srv_pipe_hnd.c will take care */";
163         pidl "\treturn true;";
164         pidl "}";
165         pidl "";
166         pidl "if (DEBUGLEVEL >= 10) {";
167         pidl "\tNDR_PRINT_OUT_DEBUG($fn->{NAME}, r);";
168         pidl "}";
169         pidl "";
170         pidl "push = ndr_push_init_ctx(r);";
171         pidl "if (push == NULL) {";
172         pidl "\ttalloc_free(r);";
173         pidl "\treturn false;";
174         pidl "}";
175         pidl "";
176         pidl "ndr_err = call->ndr_push(push, NDR_OUT, r);";
177         pidl "if (!NDR_ERR_CODE_IS_SUCCESS(ndr_err)) {";
178         pidl "\ttalloc_free(r);";
179         pidl "\treturn false;";
180         pidl "}";
181         pidl "";
182         pidl "blob = ndr_push_blob(push);";
183         pidl "if (!prs_copy_data_in(&p->out_data.rdata, (const char *)blob.data, (uint32_t)blob.length)) {";
184         pidl "\ttalloc_free(r);";
185         pidl "\treturn false;";
186         pidl "}";
187         pidl "";
188         pidl "talloc_free(r);";
189         pidl "";
190         pidl "return true;";
191         deindent;
192         pidl "}";
193         pidl "";
194 }
195
196 sub ParseInterface($)
197 {
198         my $if = shift;
199
200         my $uif = uc($if->{NAME});
201
202         pidl_hdr "#ifndef __SRV_$uif\__";
203         pidl_hdr "#define __SRV_$uif\__";
204         ParseFunction($if, $_) foreach (@{$if->{FUNCTIONS}});
205
206         pidl "";
207         pidl "/* Tables */";
208         pidl "static struct api_struct api_$if->{NAME}_cmds[] = ";
209         pidl "{";
210         indent;
211
212         foreach (@{$if->{FUNCTIONS}}) {
213                 pidl "{\"" . uc($_->{NAME}) . "\", NDR_" . uc($_->{NAME}) . ", api_$_->{NAME}},";
214         }
215
216         deindent;
217         pidl "};";
218
219         pidl "";
220
221         pidl_hdr "void $if->{NAME}_get_pipe_fns(struct api_struct **fns, int *n_fns);";
222         pidl "void $if->{NAME}_get_pipe_fns(struct api_struct **fns, int *n_fns)";
223         pidl "{";
224         indent;
225         pidl "*fns = api_$if->{NAME}_cmds;";
226         pidl "*n_fns = sizeof(api_$if->{NAME}_cmds) / sizeof(struct api_struct);";
227         deindent;
228         pidl "}";
229         pidl "";
230
231         pidl_hdr "NTSTATUS rpc_$if->{NAME}_init(void);";
232         pidl "NTSTATUS rpc_$if->{NAME}_init(void)";
233         pidl "{";
234         pidl "\treturn rpc_pipe_register_commands(SMB_RPC_INTERFACE_VERSION, \"$if->{NAME}\", \"$if->{NAME}\", api_$if->{NAME}_cmds, sizeof(api_$if->{NAME}_cmds) / sizeof(struct api_struct));";
235         pidl "}";
236
237         pidl_hdr "#endif /* __SRV_$uif\__ */";
238 }
239
240 sub Parse($$$)
241 {
242         my($ndr,$header,$ndr_header) = @_;
243
244         $res = "";
245         $res_hdr = "";
246
247         pidl "/*";
248         pidl " * Unix SMB/CIFS implementation.";
249         pidl " * server auto-generated by pidl. DO NOT MODIFY!";
250         pidl " */";
251         pidl "";
252         pidl "#include \"includes.h\"";
253         pidl "#include \"$header\"";
254         pidl_hdr "#include \"$ndr_header\"";
255         pidl "";
256
257         foreach (@$ndr) {
258                 ParseInterface($_) if ($_->{TYPE} eq "INTERFACE");
259         }
260
261         return ($res, $res_hdr);
262 }
263
264 1;