Salome HOME
Fix a bug: compilation failed on Mandrake 10.1 with automake version 1.9
[tools/hxx2salome.git] / scripts / parse3.awk
1 # This awk program contains the type mapping tables - and the treatments
2 # for code generation
3 #
4 BEGIN { 
5 #
6 # file name generation
7   idl_file="code_idl"
8   hxx_file="code_hxx"
9   cxx_file="code_cxx"
10   class_i=class_name"_i"
11   print "\t// generated part" > idl_file
12   printf "    // generated part\n" > hxx_file
13   printf "//\n// generated part\n//\n" > cxx_file
14   print "Functions parsing (for debug)" > "parse_result"
15 #
16 #
17 # type mapping from c++ component to idl
18 #
19   idl_arg_type["int"]="in long"
20   idl_arg_type["double"]="in double"
21   idl_arg_type["float"]="in float"
22   idl_arg_type["long"]="in long"
23   idl_arg_type["short"]="in short"
24   idl_arg_type["unsigned"]="in unsigned long"
25   idl_arg_type["const char*"]="in string"
26   idl_arg_type["const std::string&"]="in string"
27   idl_arg_type["int&"]="out long"
28   idl_arg_type["double&"]="out double"
29   idl_arg_type["float&"]="out float"
30   idl_arg_type["long&"]="out long"
31   idl_arg_type["short&"]="out short"
32   idl_arg_type["unsigned&"]="out unsigned long"
33   idl_arg_type["std::string&"]="out string"
34   idl_arg_type["const MEDMEM::MESH&"]="in SALOME_MED::MESH"
35   idl_arg_type["const MEDMEM::MESH*"]="in SALOME_MED::MESH"
36   idl_arg_type["const MEDMEM::FIELD<double>*"]="in SALOME_MED::FIELDDOUBLE"
37   idl_arg_type["const MEDMEM::FIELD<double>&"]="in SALOME_MED::FIELDDOUBLE"
38   idl_arg_type["const std::vector<double>&"]="in SALOME::SenderDouble"
39   idl_arg_type["MEDMEM::FIELD<double>*&"]="out SALOME_MED::FIELDDOUBLE"
40   idl_arg_type["const MEDMEM::FIELD<int>*"]="in SALOME_MED::FIELDINT"
41   idl_arg_type["const MEDMEM::FIELD<int>&"]="in SALOME_MED::FIELDINT"
42   idl_arg_type["const std::vector<int>&"]="in SALOME::SenderInt"
43   idl_arg_type["MEDMEM::FIELD<int>*&"]="out SALOME_MED::FIELDINT"
44 #
45 #
46 # mapping for returned types
47 #
48   idl_rtn_type["void"]="void"
49   idl_rtn_type["int"]="long"
50   idl_rtn_type["double"]="double"
51   idl_rtn_type["float"]="float"
52   idl_rtn_type["long"]="long"
53   idl_rtn_type["short"]="short"
54   idl_rtn_type["unsigned"]="unsigned long"
55   idl_rtn_type["const char*"]="string"
56   idl_rtn_type["char*"]="string"
57   idl_rtn_type["std::string"]="string"
58   idl_rtn_type["const MEDMEM::MESH&"]="SALOME_MED::MESH"
59   idl_rtn_type["MEDMEM::MESH&"]="SALOME_MED::MESH"
60   idl_rtn_type["MEDMEM::MESH*"]="SALOME_MED::MESH"
61   idl_rtn_type["const MEDMEM::MESH*"]="SALOME_MED::MESH"
62   idl_rtn_type["const MEDMEM::FIELD<double>*"]="SALOME_MED::FIELDDOUBLE"
63   idl_rtn_type["MEDMEM::FIELD<double>*"]="SALOME_MED::FIELDDOUBLE"
64   idl_rtn_type["MEDMEM::FIELD<double>&"]="SALOME_MED::FIELDDOUBLE"
65   idl_rtn_type["const MEDMEM::FIELD<double>&"]="SALOME_MED::FIELDDOUBLE"
66   idl_rtn_type["std::vector<double>*"]="SALOME::SenderDouble"
67   idl_rtn_type["const MEDMEM::FIELD<int>*"]="SALOME_MED::FIELDINT"
68   idl_rtn_type["MEDMEM::FIELD<int>*"]="SALOME_MED::FIELDINT"
69   idl_rtn_type["MEDMEM::FIELD<int>&"]="SALOME_MED::FIELDINT"
70   idl_rtn_type["const MEDMEM::FIELD<int>&"]="SALOME_MED::FIELDINT"
71   idl_rtn_type["std::vector<int>*"]="SALOME::SenderInt"
72 #
73 #
74 # Corba mapping table (for argument's types and returned types)
75 #
76 # CNC commented types are not yet implemented
77   idl_impl_hxx["in long"]="CORBA::Long"
78   idl_impl_hxx["in double"]="CORBA::Double"
79   idl_impl_hxx["in float"]="CORBA::Float"
80   idl_impl_hxx["in short"]="CORBA::Short"
81   idl_impl_hxx["in unsigned long"]="CORBA::ULong"
82   idl_impl_hxx["in string"]="const char*"
83   idl_impl_hxx["out long"]="CORBA::Long_out"
84   idl_impl_hxx["out double"]="CORBA::Double_out"
85   idl_impl_hxx["out float"]="CORBA::Float_out"
86   idl_impl_hxx["out short"]="CORBA::Short_out"
87   idl_impl_hxx["out unsigned long"]="CORBA::ULong_out"
88   idl_impl_hxx["out string"]="CORBA::String_out"
89   idl_impl_hxx["in SALOME_MED::MESH"]="SALOME_MED::MESH_ptr"
90   idl_impl_hxx["in SALOME_MED::FIELDDOUBLE"]="SALOME_MED::FIELDDOUBLE_ptr"
91   idl_impl_hxx["in SALOME::SenderDouble"]="SALOME::SenderDouble_ptr"
92   idl_impl_hxx["out SALOME_MED::FIELDDOUBLE"]="SALOME_MED::FIELDDOUBLE_out"
93   idl_impl_hxx["in SALOME_MED::FIELDINT"]="SALOME_MED::FIELDINT_ptr"
94   idl_impl_hxx["in SALOME::SenderInt"]="SALOME::SenderInt_ptr"
95   idl_impl_hxx["out SALOME_MED::FIELDINT"]="SALOME_MED::FIELDINT_out"
96   idl_impl_hxx["void"]="void"
97   idl_impl_hxx["long"]="CORBA::Long"
98   idl_impl_hxx["double"]="CORBA::Double"
99   idl_impl_hxx["unsigned long"]="CORBA::ULong"
100   idl_impl_hxx["string"]="char*"
101   idl_impl_hxx["SALOME_MED::MESH"]="SALOME_MED::MESH_ptr"
102   idl_impl_hxx["SALOME_MED::FIELDDOUBLE"]="SALOME_MED::FIELDDOUBLE_ptr"
103   idl_impl_hxx["SALOME::SenderDouble"]="SALOME::SenderDouble_ptr"
104   idl_impl_hxx["SALOME_MED::FIELDINT"]="SALOME_MED::FIELDINT_ptr"
105   idl_impl_hxx["SALOME::SenderInt"]="SALOME::SenderInt_ptr"
106 #
107 #
108 # table for c++ code generation : argument's processing
109 #
110   cpp_impl_a["int"]="\tint _%s(%s);\n"
111   cpp_impl_a["double"]="\tdouble _%s(%s);\n"
112   cpp_impl_a["float"]="\tfloat _%s(%s);\n"
113   cpp_impl_a["long"]="\tlong _%s(%s);\n"
114   cpp_impl_a["short"]="\tshort _%s(%s);\n"
115   cpp_impl_a["unsigned"]="\tunsigned _%s(%s);\n"
116   cpp_impl_a["const char*"]="\tconst char* _%s(%s);\n"
117   cpp_impl_a["const std::string&"]="\tconst std::string _%s(%s);\n"
118   cpp_impl_a["int&"]="\tint _%s;\n"
119   cpp_impl_a["double&"]="\tdouble _%s;\n"
120   cpp_impl_a["float&"]="\tfloat _%s;\n"
121   cpp_impl_a["long&"]="\tlong _%s;\n"
122   cpp_impl_a["short&"]="\tshort _%s;\n"
123   cpp_impl_a["unsigned&"]="\tunsigned _%s;\n"
124   cpp_impl_a["std::string&"]="std::string _%s;\n"
125   cpp_impl_a["const MEDMEM::MESH&"]="\tMEDMEM::MESHClient* _%s = new MEDMEM::MESHClient(%s);\n" # MESHClient cannot be created on the stack (private constructor), so we create it on the heap and dereference it later (in treatment 4)
126   cpp_impl_a["const MEDMEM::MESH*"]="\tMEDMEM::MESHClient* _%s = new MEDMEM::MESHClient(%s);\n"
127   cpp_impl_a["MEDMEM::FIELD<double>*&"]="\tMEDMEM::FIELD<double>* _%s;\n"
128   cpp_impl_a["const MEDMEM::FIELD<double>*"]="\tstd::auto_ptr<MEDMEM::FIELD<double> > _%s ( new MEDMEM::FIELDClient<double>(%s) );\n"
129   cpp_impl_a["const MEDMEM::FIELD<double>&"]="\tMEDMEM::FIELDClient<double> _%s(%s);\n"
130   cpp_impl_a["const std::vector<double>&"]="\tlong _%s_size;\n\tdouble *_%s_value = ReceiverFactory::getValue(%s,_%s_size);\n"\
131              "\tstd::vector<double> _%s(_%s_value,_%s_value+_%s_size);\n\tdelete [] _%s_value;"
132   cpp_impl_a["MEDMEM::FIELD<int>*&"]="\tMEDMEM::FIELD<int>* _%s;\n"
133   cpp_impl_a["const MEDMEM::FIELD<int>*"]="\tstd::auto_ptr<MEDMEM::FIELD<int> > _%s ( new MEDMEM::FIELDClient<int>(%s) );\n"
134   cpp_impl_a["const MEDMEM::FIELD<int>&"]="\tMEDMEM::FIELDClient<int> _%s(%s);\n"
135   cpp_impl_a["const std::vector<int>&"]="\tlong _%s_size;\n\tint *_%s_value = ReceiverFactory::getValue(%s,_%s_size);\n"\
136              "\tstd::vector<int> _%s(_%s_value,_%s_value+_%s_size);\n\tdelete [] _%s_value;"
137 #
138 #
139 # table for c++ code generation : returned value processing
140 #
141   cpp_impl_b["void"]=""
142   cpp_impl_b["int"]="\tCORBA::Long _rtn_ior(_rtn_cpp);\n"
143   cpp_impl_b["double"]="\tCORBA::Double _rtn_ior(_rtn_cpp);\n"
144   cpp_impl_b["float"]="\tCORBA::Float _rtn_ior(_rtn_cpp);\n"
145   cpp_impl_b["long"]="\tCORBA::Long _rtn_ior(_rtn_cpp);\n"
146   cpp_impl_b["short"]="\tCORBA::Short _rtn_ior(_rtn_cpp);\n"
147   cpp_impl_b["unsigned"]="\tCORBA::ULong _rtn_ior(_rtn_cpp);\n"
148   cpp_impl_b["const char*"]="\tchar* _rtn_ior = CORBA::string_dup(_rtn_cpp);\n"
149   cpp_impl_b["char*"]="\tchar* _rtn_ior(_rtn_cpp);\n"
150   cpp_impl_b["std::string"]="\tchar* _rtn_ior=CORBA::string_dup(_rtn_cpp.c_str());\n"
151              "\tstd::copy(_rtn_cpp.begin(),_rtn_cpp.end(),_rtn_ior);\n"    
152   cpp_impl_b["const MEDMEM::MESH&"]=\
153              "\tMEDMEM::MESH_i * _rtn_mesh_i = new MEDMEM::MESH_i(const_cast<MEDMEM::MESH*>(&_rtn_cpp));\n"\
154              "\tSALOME_MED::MESH_ptr _rtn_ior = _rtn_mesh_i->_this();\n"
155   cpp_impl_b["MEDMEM::MESH&"]=\
156              "\tMEDMEM::MESH_i * _rtn_mesh_i = new MEDMEM::MESH_i(&_rtn_cpp);\n"\
157              "\tSALOME_MED::MESH_ptr _rtn_ior = _rtn_mesh_i->_this();\n"
158   cpp_impl_b["MEDMEM::MESH*"]=\
159              "\tMEDMEM::MESH_i * _rtn_mesh_i = new MEDMEM::MESH_i(_rtn_cpp);\n"\
160              "\tSALOME_MED::MESH_ptr _rtn_ior = _rtn_mesh_i->_this();\n"
161   cpp_impl_b["const MEDMEM::MESH*"]=\
162              "\tMEDMEM::MESH_i * _rtn_mesh_i = new MEDMEM::MESH_i(const_cast<MEDMEM::MESH*>(_rtn_cpp));\n"\
163              "\tSALOME_MED::MESH_ptr _rtn_ior = _rtn_mesh_i->_this();\n"
164   cpp_impl_b["const MEDMEM::FIELD<double>*"]=\
165              "\tMEDMEM::FIELDDOUBLE_i * _rtn_field_i = new MEDMEM::FIELDDOUBLE_i(const_cast<MEDMEM::FIELD<double>*>(_rtn_cpp),false);\n"\
166              "\tSALOME_MED::FIELDDOUBLE_ptr _rtn_ior = _rtn_field_i->_this();\n"
167   cpp_impl_b["MEDMEM::FIELD<double>*"]=\
168              "\tMEDMEM::FIELDDOUBLE_i * _rtn_field_i = new MEDMEM::FIELDDOUBLE_i(_rtn_cpp,true);\n"\
169              "\tSALOME_MED::FIELDDOUBLE_ptr _rtn_ior = _rtn_field_i->_this();\n"
170   cpp_impl_b["MEDMEM::FIELD<double>&"]=\
171              "\tMEDMEM::FIELDDOUBLE_i * _rtn_field_i = new MEDMEM::FIELDDOUBLE_i(&_rtn_cpp,false);\n"\
172              "\tSALOME_MED::FIELDDOUBLE_ptr _rtn_ior = _rtn_field_i->_this();\n"
173   cpp_impl_b["const MEDMEM::FIELD<double>&"]=\
174              "\tMEDMEM::FIELDDOUBLE_i * _rtn_field_i = new MEDMEM::FIELDDOUBLE_i(const_cast<MEDMEM::FIELD<double>*>(&_rtn_cpp),false);\n"\
175              "\tSALOME_MED::FIELDDOUBLE_ptr _rtn_ior = _rtn_field_i->_this();\n"
176   cpp_impl_b["std::vector<double>*"]=\
177              "\tSALOME::SenderDouble_ptr _rtn_ior = SenderFactory::buildSender(*this,&(*_rtn_cpp)[0],(*_rtn_cpp).size(),true);\n"
178   cpp_impl_b["const MEDMEM::FIELD<int>*"]=\
179              "\tMEDMEM::FIELDINT_i * _rtn_field_i = new MEDMEM::FIELDINT_i(const_cast<MEDMEM::FIELD<int>*>(_rtn_cpp),false);\n"\
180              "\tSALOME_MED::FIELDINT_ptr _rtn_ior = _rtn_field_i->_this();\n"
181   cpp_impl_b["MEDMEM::FIELD<int>*"]=\
182              "\tMEDMEM::FIELDINT_i * _rtn_field_i = new MEDMEM::FIELDINT_i(_rtn_cpp,true);\n"\
183              "\tSALOME_MED::FIELDINT_ptr _rtn_ior = _rtn_field_i->_this();\n"
184   cpp_impl_b["MEDMEM::FIELD<int>&"]=\
185              "\tMEDMEM::FIELDINT_i * _rtn_field_i = new MEDMEM::FIELDINT_i(&_rtn_cpp,false);\n"\
186              "\tSALOME_MED::FIELDINT_ptr _rtn_ior = _rtn_field_i->_this();\n"
187   cpp_impl_b["const MEDMEM::FIELD<int>&"]=\
188              "\tMEDMEM::FIELDINT_i * _rtn_field_i = new MEDMEM::FIELDINT_i(const_cast<MEDMEM::FIELD<int>*>(&_rtn_cpp),false);\n"\
189              "\tSALOME_MED::FIELDINT_ptr _rtn_ior = _rtn_field_i->_this();\n"
190   cpp_impl_b["std::vector<int>*"]=\
191              "\tSALOME::SenderInt_ptr _rtn_ior = SenderFactory::buildSender(*this,&(*_rtn_cpp)[0],(*_rtn_cpp).size(),true);\n"
192
193 #
194 #
195 # table for c++ code generation : out parameters processing and removeRef for reference counted objects
196 #
197   cpp_impl_c["MEDMEM::FIELD<double>*&"]=\
198              "\tMEDMEM::FIELDDOUBLE_i * %s_ior = new MEDMEM::FIELDDOUBLE_i(_%s, true);\n"\
199              "\t%s = %s_ior->_this();\n"
200   cpp_impl_c["MEDMEM::FIELD<int>*&"]=\
201              "\tMEDMEM::FIELDINT_i * %s_ior = new MEDMEM::FIELDINT_i(_%s, true);\n"\
202              "\t%s = %s_ior->_this();\n"
203   cpp_impl_c["std::string&"]="\t%s = CORBA::string_dup(_%s.c_str());\n"
204   cpp_impl_c["int&"]="\t%s = _%s"
205   cpp_impl_c["double&"]="\t%s = _%s"
206   cpp_impl_c["float&"]="\t%s = _%s"
207   cpp_impl_c["long&"]="\t%s = _%s"
208   cpp_impl_c["short&"]="\t%s = _%s"
209   cpp_impl_c["unsigned&"]="\t%s = _%s"
210   cpp_impl_c["const MEDMEM::MESH&"]="\t_%s->removeReference();\n"
211   cpp_impl_c["const MEDMEM::MESH*"]="\t_%s->removeReference();\n"
212 #
213 #
214 # record sep is ");\n" whith blanks all around, and optional "(" at the beginning
215   RS="[(]?[ \t]*\)[ \t]*;[ \t]*\n"  
216   FS="[ \t]*[(,][ \t]*"  # field sep is either "(" or "," surrounded by blanks 
217 }
218
219 # --------------------- treatment 1 ----------------------------------
220 #
221 #  extract from fields types, function name, and argument's names
222 #
223 {
224   print "Function : ",$0 >> "parse_result"  # print for debug
225   for (i=1; i<=NF; i++) {
226       print "\t-> ",i," : ",$i >> "parse_result"
227   }
228   ok1=0;ok=1
229   # check if returned type ($1) is one of the accepted types (idl_rtn_type)
230   for (cpptype in idl_rtn_type) {
231     if ( substr($1,1,length(cpptype)) == cpptype ) {
232       # if compatible, store returned type and function name
233       type[1]=cpptype
234       name[1]=substr($1,length(cpptype)+1)
235       sub("^[ \t]*","",name[1]) # get rid of leading blanks
236       ok1=1
237       break
238     }
239   }
240   ok*=ok1
241   # for each argument ($i), check if it is compatible (belongs to idl_arg_type)
242   for (i=2; i<=NF; i++) {
243     ok2=0
244     split($i,tab,"=") # get rid of default value
245     item=tab[1]
246     for (cpptype in idl_arg_type) {
247        if ( substr(item,1,length(cpptype)) == cpptype ) {
248           # if compatible, store argument type and name
249           type[i]=cpptype
250           name[i]=substr(item,length(cpptype)+1)
251           sub("^[ \t]*","",name[i]) # get rid of leading blanks
252           if ( length(name[i]) == 0 ) # automatic name if argument's name wasn't precised
253              name[i]=sprintf("_arg%d",i-1)
254           ok2=1
255           break
256        }
257     }
258     ok*=ok2 # ok=0 if one of the type is not compatible
259   }
260
261   # print compatibility 
262   if ( $0 !~ class_name ) { # constructor are not considered, but we don't print it
263       if ( ok == 0){ # if one of the c++ type is not compatible
264           printf "     [KO]     :  %s",$0
265       }
266       else
267           printf "     [OK]     :  %s",$0
268         
269       if ( $0 !~ /\(/  ) {
270           printf "(" # if there is no argument, parenthesis was suppressed, so we add it for printing
271       }
272       printf ");\n"
273   }    
274   if ( ok == 0) # pass to the next function if one of the c++ type is not compatible
275       next
276 }
277 #
278 # --------------------- treatment 2 ----------------------------------
279 #
280 #  generate the Corba interface (idl file)
281 #
282
283   printf "\t%s %s(", idl_rtn_type[type[1]],name[1] >> idl_file  # return type and name of function
284   if ( NF >= 2 ){  # if there is arguments, print them
285     for (i=2; i<=NF-1; i++)
286       printf "%s %s,",idl_arg_type[type[i]],name[i] >> idl_file
287     printf "%s %s", idl_arg_type[type[NF]],name[NF] >> idl_file
288   }
289   printf ");\n" >> idl_file
290 }  
291 #
292 # --------------------- treatment 3 ----------------------------------
293 #
294 #  generate the C++ implementation of component (hxx file)
295 #
296
297   printf "    %s %s(",idl_impl_hxx[idl_rtn_type[type[1]]],name[1] >> hxx_file
298   if ( NF >= 2 ){  # if there is arguments, print them
299       for (i=2; i<=NF-1; i++)
300           printf "%s %s,",idl_impl_hxx[idl_arg_type[type[i]]],name[i] >> hxx_file
301       printf "%s %s", idl_impl_hxx[idl_arg_type[type[NF]]],name[NF] >> hxx_file
302   }
303   printf ");\n" >> hxx_file
304 }
305 #
306 # --------------------- treatment 4 ----------------------------------
307 #
308 #  generate the C++ implementation of component (cxx file)
309 #
310 {
311   # a) generate the function declaration + macro declarations
312   func_name=class_name"_i::"name[1]
313   printf "%s %s(",idl_impl_hxx[idl_rtn_type[type[1]]],func_name >> cxx_file
314   if ( NF >= 2 ){  # if there is arguments, print them
315       for (i=2; i<=NF-1; i++)
316           printf "%s %s,",idl_impl_hxx[idl_arg_type[type[i]]],name[i] >> cxx_file
317       printf "%s %s", idl_impl_hxx[idl_arg_type[type[NF]]],name[NF] >> cxx_file
318   }
319   printf ")\n{\n\tbeginService(\"%s\");\n\tBEGIN_OF(\"%s\");\n",func_name,func_name >> cxx_file
320
321   # b) generate the argument processing part
322   if ( NF >= 2 ){
323       printf "//\tArguments processing\n" >> cxx_file
324       for (i=2; i<=NF; i++)
325           printf cpp_impl_a[type[i]],name[i],name[i],name[i],name[i],name[i],name[i],name[i],name[i],name[i] >> cxx_file
326   }
327
328   # c) generate the call to the c++ component
329   if ( type[1] == "void" ) # if return type is void, the call syntax is different.
330       printf "//\tCall cpp component\n\tcppCompo_->%s(",name[1] >> cxx_file
331   else
332       printf "//\tCall cpp component\n\t%s _rtn_cpp = cppCompo_->%s(",type[1],name[1] >> cxx_file
333   if ( NF >= 2 ){  # if there is arguments, print them
334       for (i=2; i<=NF; i++) {
335           # special treatment for some arguments
336           post=""
337           pre=""
338           if ( cpp_impl_a[type[i]] ~ "auto_ptr" )
339              post=".get()" # for auto_ptr argument, retrieve the raw pointer behind
340           if ( type[i] == "const MEDMEM::MESH&" )
341              pre="*"  # we cannot create MESHClient on the stack (private constructor), so we create it on the heap and dereference it
342           if ( i < NF )
343              post=post"," # separator between arguments
344           printf " %s_%s%s",pre,name[i],post >> cxx_file
345       }
346   }
347
348   # d) generate the post_processing of returned and out parameters
349   printf ");\n//\tPost-processing & return\n" >> cxx_file
350   for (i=2; i<=NF; i++)
351       printf cpp_impl_c[type[i]],name[i],name[i],name[i],name[i] >> cxx_file  # process for out parameters
352   printf cpp_impl_b[type[1]] >> cxx_file  # process for returned value
353   printf "\tendService(\"%s\");\n\tEND_OF(\"%s\");\n",func_name,func_name >> cxx_file
354   if ( type[1] != "void" )
355       printf "\treturn _rtn_ior;\n" >> cxx_file
356   printf "}\n\n" >> cxx_file
357 }
358 #
359 #
360 END {
361 # CNC peut ĂȘtre mis dans le template directement printf "\nprivate:\n    std::auto_ptr<%s> cppImpl_;\n",class_name >> hxx_file
362 }