Logo AND Algorithmique Numérique Distribuée

Public GIT Repository
Indent the rest of the code (examples, buildtools, doc...) except for examples/SMPI...
[simgrid.git] / tools / gras / unix_stub_generator.c
1 /*      $Id$     */
2
3 /* gras_stub_generator - creates the main() to use a GRAS program           */
4
5 /* Copyright (c) 2003-2007 Martin Quinson, Arnaud Legrand, Malek Cherier.   */
6 /* All rights reserved.                                                     */
7
8 /* This program is free software; you can redistribute it and/or modify it
9  * under the terms of the license (GNU LGPL) which comes with this package. */
10
11 /* specific to Borland Compiler */
12 #ifdef __BORLANDDC__
13 #pragma hdrstop
14 #endif
15
16 #include <stdio.h>
17 #include "xbt/sysdep.h"
18 #include "xbt/function_types.h"
19 #include "xbt/log.h"
20 #include "surf/surfxml_parse.h"
21 #include "surf/surf.h"
22 #include "portable.h"           /* Needed for the time of the SIMIX convertion */
23 #include "gras_stub_generator.h"
24
25 #include <stdarg.h>
26
27 #ifdef __BORLANDC__
28 /* stupid stubs so that it compiles on windows */
29 void generate_sim(char *project)
30 {
31 }
32
33 void generate_rl(char *project)
34 {
35 }
36
37 void generate_makefile_local(char *project, char *deployment)
38 {
39 }
40 #else
41 /* real implementation */
42 XBT_LOG_EXTERNAL_DEFAULT_CATEGORY(stubgen);
43
44
45 #define WARN "/***********\n * DO NOT EDIT! THIS FILE HAS BEEN AUTOMATICALLY GENERATED FROM %s BY gras_stub_generator\n ***********/\n"
46 #define SIM_SOURCENAME  "_%s_simulator.c"
47 #define SIM_OBJNAME  "_%s_simulator.o"
48 #define SIM_BINARYNAME  "%s_simulator"
49 #define SIM_SOURCENAME_LDADD  "%s_simulator_LDADD"
50 #define SIM_SOURCENAME_SOURCES  "%s_simulator_SOURCES"
51 #define RL_SOURCENAME  "_%s_%s.c"
52 #define RL_OBJNAME  "_%s_%s.o"
53 #define RL_BINARYNAME  "%s_%s"
54 #define RL_SOURCENAME_LDADD  "%s_%s_LDADD"
55 #define RL_SOURCENAME_SOURCES  "%s_%s_SOURCES"
56 #define MAKEFILE_FILENAME_AM  "%s.Makefile.am"
57 #define MAKEFILE_FILENAME_LOCAL  "%s.mk"
58 #define MAKEFILE_FILENAME_REMOTE  "%s.Makefile.remote"
59 #define DEPLOYMENT  "%s.deploy.sh"
60
61 char *warning = NULL;
62
63 /**********************************************/
64 /**** Generate the file for the simulator *****/
65 /**********************************************/
66
67 const char *SIM_PREEMBULE =
68     "/* specific to Borland Compiler */\n"
69     "#ifdef __BORLANDC__\n"
70     "#pragma hdrstop\n"
71     "#endif\n\n"
72     "#include <stdlib.h>\n"
73     "#include <stdio.h>\n"
74     "#include \"msg/msg.h\"\n"
75     "#include <gras.h>\n" "\n" "char *gras_log=NULL;\n";
76
77
78 #define SIM_LAUNCH_FUNC  \
79 "int launch_%s(int argc, char **argv) {\n" \
80 "  char **myargv=argv;\n" \
81 "  int myargc=argc;\n" \
82 "  int i;\n" \
83 "  int retcode;\n"\
84 "    \n"\
85 "  if (gras_log) {\n"\
86 "    myargv=malloc((argc+1) * sizeof(char**));\n" \
87 "    for (i=0; i<argc; i++)\n" \
88 "      myargv[i] = argv[i];\n" \
89 "    myargv[myargc++] = gras_log;\n" \
90 "  }\n" \
91 "  retcode = %s(myargc,myargv);\n" \
92 "  if (myargv != argv)\n" \
93 "    free(myargv);\n" \
94 "  return retcode;\n" \
95 "}\n"
96
97 const char *SIM_MAIN_POSTEMBULE = "\n"
98     "\n"
99     "  gras_launch_application(argv[2]);\n"
100     "\n"
101     "  /*  Run the simulation */\n"
102     "  gras_main();\n"
103     "\n"
104     "  /* cleanup the place */\n"
105     "  gras_clean();\n"
106     "  if (gras_log)\n" "    free(gras_log);\n" "  return 0;\n" "}\n";
107
108
109
110
111 void generate_sim(char *project)
112 {
113   xbt_dict_cursor_t cursor = NULL;
114   char *key = NULL;
115   void *data = NULL;
116   char *filename = NULL;
117   FILE *FICOUT = NULL;
118
119   /* Output file: <projet>_simulator.c */
120   filename = xbt_new(char, strlen(project) + strlen(SIM_SOURCENAME));
121   sprintf(filename, SIM_SOURCENAME, project);
122
123   FICOUT = fopen(filename, "w");
124
125   xbt_assert1(FICOUT, "Unable to open %s for writing", filename);
126
127   fprintf(FICOUT, "%s\n", warning);
128   fprintf(FICOUT, "%s", SIM_PREEMBULE);
129
130   xbt_dict_foreach(process_function_set, cursor, key, data) {
131     fprintf(FICOUT, "int %s(int argc,char *argv[]);\n", key);
132   }
133
134   fprintf(FICOUT, "\n");
135
136   xbt_dict_foreach(process_function_set, cursor, key, data) {
137     fprintf(FICOUT, "int launch_%s(int argc,char *argv[]);\n", key);
138   }
139
140   fprintf(FICOUT, "\n%s\n", warning);
141
142   xbt_dict_foreach(process_function_set, cursor, key, data) {
143     fprintf(FICOUT, SIM_LAUNCH_FUNC, key, key);
144   }
145   fprintf(FICOUT, "\n%s\n", warning);
146
147   fprintf(FICOUT, "%s", "/* specific to Borland Compiler */\n"
148           "#ifdef __BORLANDDC__\n" "#pragma argsused\n" "#endif\n\n");
149
150   fprintf(FICOUT, "%s", "int main (int argc,char *argv[]) {\n"
151           "\n"
152           "  /*  Simulation setup */\n"
153           "  gras_global_init(&argc,argv);\n"
154           "  if (argc != 3) {\n"
155           "    fprintf(stderr, \"Usage: %s platform.xml deployment.xml [--log=...]\\n\",argv[0]);\n"
156           "    exit(1);\n" "  }\n" "\n");
157   fprintf(FICOUT,
158           "  gras_create_environment(argv[1]);\n"
159           "\n" "  /*  Application deployment */\n");
160   xbt_dict_foreach(process_function_set, cursor, key, data) {
161     fprintf(FICOUT, "  gras_function_register(\"%s\", launch_%s);\n", key,
162             key);
163   }
164   fprintf(FICOUT, "%s", SIM_MAIN_POSTEMBULE);
165   fclose(FICOUT);
166   free(filename);
167 }
168
169 /**********************************************/
170 /**** Generate the file for the real life *****/
171 /**********************************************/
172 void generate_rl(char *project)
173 {
174   xbt_dict_cursor_t cursor = NULL;
175   char *key = NULL;
176   void *data = NULL;
177   char *filename = NULL;
178   FILE *FICOUT = NULL;
179
180   xbt_dict_foreach(process_function_set, cursor, key, data) {
181     filename =
182         xbt_new(char,
183                 strlen(project) + strlen(RL_SOURCENAME) + strlen(key));
184
185     sprintf(filename, RL_SOURCENAME, project, key);
186
187     FICOUT = fopen(filename, "w");
188     xbt_assert1(FICOUT, "Unable to open %s for writing", filename);
189
190     fprintf(FICOUT, "\n%s\n", warning);
191     fprintf(FICOUT, "/* specific to Borland Compiler */\n"
192             "#ifdef __BORLANDC__\n"
193             "#pragma hdrstop\n"
194             "#endif\n\n"
195             "#include <stdio.h>\n"
196             "#include <signal.h>\n"
197             "#include <gras.h>\n"
198             "\n"
199             "XBT_PUBLIC_DATA(const char *) _gras_procname;\n"
200             "/* user code */\n"
201             "int %s(int argc, char *argv[]);\n"
202             "\n"
203             "/* specific to Borland Compiler */\n"
204             "#ifdef __BORLANDC__\n"
205             "#pragma argsused\n"
206             "#endif\n\n"
207             "int main(int argc, char *argv[]){\n"
208             "  int errcode;\n"
209             "\n"
210             "  _gras_procname = \"%s\";\n"
211             "  errcode=%s(argc,argv);\n"
212             " \n" "  return errcode;\n" "}\n", key, key, key);
213     fprintf(FICOUT, "\n%s\n", warning);
214     fclose(FICOUT);
215     free(filename);
216   }
217 }
218
219 void generate_makefile_am(char *project, char *deployment)
220 {
221   xbt_dict_cursor_t cursor = NULL;
222   char *key = NULL;
223   void *data = NULL;
224   char *filename = NULL;
225   FILE *FICOUT = NULL;
226
227   filename = xbt_new(char, strlen(project) + strlen(MAKEFILE_FILENAME_AM));
228   sprintf(filename, MAKEFILE_FILENAME_AM, project);
229
230   FICOUT = fopen(filename, "w");
231   xbt_assert1(FICOUT, "Unable to open %s for writing", filename);
232
233   fprintf(FICOUT, "# AUTOMAKE variable definition\n");
234   fprintf(FICOUT, "INCLUDES= @CFLAGS_SimGrid@\n\n");
235   fprintf(FICOUT, "PROGRAMS=");
236   fprintf(FICOUT, SIM_BINARYNAME, project);
237
238   xbt_dict_foreach(process_function_set, cursor, key, data) {
239     fprintf(FICOUT, " ");
240     fprintf(FICOUT, RL_BINARYNAME, project, key);
241   }
242
243   fprintf(FICOUT, "\n\n");
244   fprintf(FICOUT, SIM_SOURCENAME_SOURCES, project);
245   fprintf(FICOUT, "=\t");
246   fprintf(FICOUT, SIM_SOURCENAME, project);
247   fprintf(FICOUT, " %s.c\n", project);
248   fprintf(FICOUT, SIM_SOURCENAME_LDADD, project);
249   fprintf(FICOUT, "=\tpath/to/libsimgrid.a\n\n");
250
251   xbt_dict_foreach(process_function_set, cursor, key, data) {
252     fprintf(FICOUT, RL_SOURCENAME_SOURCES, project, key);
253     fprintf(FICOUT, "=\t");
254     fprintf(FICOUT, RL_SOURCENAME, project, key);
255     fprintf(FICOUT, " %s.c\n", project);
256     fprintf(FICOUT, RL_SOURCENAME_LDADD, project, key);
257     fprintf(FICOUT, "=\tpath/to/libgras.a\n\n");
258   }
259
260   fprintf(FICOUT,
261           "\n# cleanup temps (allowing the user to add extra clean files)\n");
262   fprintf(FICOUT, "CLEANFILES?= \n");
263   fprintf(FICOUT, "CLEANFILES+= ");
264   fprintf(FICOUT, SIM_SOURCENAME, project);
265
266   xbt_dict_foreach(process_function_set, cursor, key, data) {
267     fprintf(FICOUT, " ");
268     fprintf(FICOUT, RL_SOURCENAME, project, key);
269   }
270   fprintf(FICOUT, "\n");
271
272   fprintf(FICOUT, "\n# generate temps\n");
273   fprintf(FICOUT,
274           "\n# A rule to generate the source file each time the deployment file changes\n");
275
276   xbt_dict_foreach(process_function_set, cursor, key, data) {
277     fprintf(FICOUT, RL_SOURCENAME, project, key);
278     fprintf(FICOUT, " ");
279   }
280   fprintf(FICOUT, SIM_SOURCENAME, project);
281   fprintf(FICOUT, ": %s\n", deployment);
282   fprintf(FICOUT, "\tgras_stub_generator %s %s >/dev/null\n", project,
283           deployment);
284   fclose(FICOUT);
285 }
286
287 void generate_makefile_local(char *project, char *deployment)
288 {
289   xbt_dict_cursor_t cursor = NULL;
290   char *key = NULL;
291   void *data = NULL;
292   char *filename = NULL;
293   FILE *FICOUT = NULL;
294
295   filename =
296       xbt_new(char, strlen(project) + strlen(MAKEFILE_FILENAME_LOCAL));
297   sprintf(filename, MAKEFILE_FILENAME_LOCAL, project);
298
299   FICOUT = fopen(filename, "w");
300   xbt_assert1(FICOUT, "Unable to open %s for writing", filename);
301   free(filename);
302
303   fprintf(FICOUT,
304           "\n"
305           "####\n"
306           "#### THIS FILE WAS GENERATED, DO NOT EDIT BEFORE RENAMING IT\n"
307           "####\n\n\n");
308
309   fprintf(FICOUT, "## Variable declarations\n"
310           "PROJECT_NAME=%s\n" "DISTDIR=gras-$(PROJECT_NAME)\n\n", project);
311
312   fprintf(FICOUT,
313           "# Set the GRAS_ROOT environment variable to the path under which you installed SimGrid\n"
314           "# Compilation will fail if you don't do so\n"
315           "GRAS_ROOT?= $(shell if [ -e /usr/local/lib/libgras.so ] ; then echo /usr/local ; else echo \"\\\"<<<< GRAS_ROOT undefined !!! >>>>\\\"\"; fi)\n\n"
316           "# You can fiddle the following to make it fit your taste\n"
317           "INCLUDES = -I$(GRAS_ROOT)/include\n"
318           "CFLAGS ?= -O3 -w -g -Wall\n"
319           "LIBS_SIM = -lm  -L$(GRAS_ROOT)/lib/ -lsimgrid\n"
320           "LIBS_RL = -lm  -L$(GRAS_ROOT)/lib/ -lgras\n" "LIBS = \n" "\n");
321
322   fprintf(FICOUT, "PRECIOUS_C_FILES ?= %s.c\n", project);
323
324   fprintf(FICOUT, "GENERATED_C_FILES = ");
325   fprintf(FICOUT, SIM_SOURCENAME " ", project);
326   xbt_dict_foreach(process_function_set, cursor, key, data) {
327     fprintf(FICOUT, RL_SOURCENAME " ", project, key);
328   }
329   fprintf(FICOUT, "\n");
330
331   fprintf(FICOUT,
332           "OBJ_FILES = $(patsubst %%.c,%%.o,$(PRECIOUS_C_FILES))\n");
333
334   fprintf(FICOUT, "BIN_FILES = ");
335
336   fprintf(FICOUT, SIM_BINARYNAME " ", project);
337   xbt_dict_foreach(process_function_set, cursor, key, data) {
338     fprintf(FICOUT, RL_BINARYNAME " ", project, key);
339   }
340   fprintf(FICOUT, "\n");
341
342   fprintf(FICOUT,
343           "\n"
344           "## By default, build all the binaries\n"
345           "all: $(BIN_FILES)\n" "\n");
346
347   fprintf(FICOUT,
348           "\n## generate temps: regenerate the source file each time the deployment file changes\n");
349   xbt_dict_foreach(process_function_set, cursor, key, data) {
350     fprintf(FICOUT, RL_SOURCENAME, project, key);
351     fprintf(FICOUT, " ");
352   }
353   fprintf(FICOUT, SIM_SOURCENAME, project);
354   fprintf(FICOUT, ": %s\n", deployment);
355   fprintf(FICOUT, "\tgras_stub_generator %s %s >/dev/null\n", project,
356           deployment);
357
358   fprintf(FICOUT, "\n## Generate the binaries\n");
359   fprintf(FICOUT, SIM_BINARYNAME ": " SIM_OBJNAME " $(OBJ_FILES)\n",
360           project, project);
361   fprintf(FICOUT,
362           "\t$(CC) $(INCLUDES) $(DEFS) $(CFLAGS) $^ $(LIBS_SIM) $(LIBS) $(LDADD) -o $@ \n");
363   xbt_dict_foreach(process_function_set, cursor, key, data) {
364     fprintf(FICOUT, RL_BINARYNAME " : " RL_OBJNAME " $(OBJ_FILES)\n",
365             project, key, project, key);
366     fprintf(FICOUT,
367             "\t$(CC) $(INCLUDES) $(DEFS) $(CFLAGS) $^ $(LIBS_RL) $(LIBS) $(LDADD) -o $@ \n");
368   }
369   fprintf(FICOUT,
370           "\n"
371           "%%: %%.o\n"
372           "\t$(CC) $(INCLUDES) $(DEFS) $(CFLAGS) $^ $(LIBS) $(LDADD) -o $@ \n"
373           "\n"
374           "%%.o: %%.c\n"
375           "\t$(CC) $(INCLUDES) $(DEFS) $(CFLAGS) -c -o $@ $<\n" "\n");
376
377   fprintf(FICOUT,
378           "## Rules for tarballs and cleaning\n"
379           "DIST_FILES= $(EXTRA_DIST) $(GENERATED_C_FILES) $(PRECIOUS_C_FILES) "
380           MAKEFILE_FILENAME_LOCAL " " /*MAKEFILE_FILENAME_REMOTE */ "\n"
381           "distdir: $(DIST_FILES)\n" "\trm -rf $(DISTDIR)\n"
382           "\tmkdir -p $(DISTDIR)\n" "\tcp $^ $(DISTDIR)\n" "\n"
383           "dist: clean distdir\n"
384           "\ttar c $(DISTDIR) | gzip -c9 > $(DISTDIR).tar.gz\n" "\n",
385           project /*, project */ );
386
387   fprintf(FICOUT,
388           "clean:\n"
389           "\trm -f $(CLEANFILES) $(BIN_FILES) $(OBJ_FILES) *~ %s.o "
390           SIM_OBJNAME, project, project);
391   xbt_dict_foreach(process_function_set, cursor, key, data) {
392     fprintf(FICOUT, " " RL_OBJNAME, project, key);
393   }
394   fprintf(FICOUT,
395           "\n"
396           "\trm -rf $(DISTDIR)\n"
397           "\n" ".SUFFIXES:\n" ".PHONY : clean\n" "\n");
398   /* 
399      fprintf(FICOUT, "############ REMOTE COMPILING #########\n");
400      fprintf(FICOUT,
401      "MACHINES ?= ");
402      xbt_dict_foreach(machine_set,cursor,key,data) {
403      fprintf(FICOUT, "%s ",key);
404      }
405      fprintf(FICOUT,"\n");
406
407      fprintf(FICOUT,
408      "INSTALL_PATH ?='$$HOME/tmp/src' ### Has to be an absolute path !!! \n"
409      "GRAS_ROOT ?='$(INSTALL_PATH)' ### Has to be an absolute path !!! \n"
410      "SRCDIR ?= ./\n"
411      "SIMGRID_URL ?=http://gcl.ucsd.edu/simgrid/dl/\n"
412      "SIMGRID_VERSION ?=2.92\n"
413      "GRAS_PROJECT ?= %s\n"
414      "GRAS_PROJECT_URL ?= http://www-id.imag.fr/Laboratoire/Membres/Legrand_Arnaud/gras_test/\n"
415      "\n"
416      "remote:\n"
417      "\t@echo;echo \"----[ Compile the package on remote hosts ]----\"\n"
418      "\t@test -e $(SRCDIR)/buildlogs/ || mkdir -p $(SRCDIR)/buildlogs/\n"
419      "\t for site in $(MACHINES) ; do \\\n"
420      "\t   machine=`echo $$site |sed 's/^\\([^%%]*\\)%%.*$$/\\1/'`;\\\n"
421      "\t   machine2=`echo $$site |sed 's/^\\([^%%]*\\)%%\\(.*\\)$$/\\2/'`;\\\n"
422      "\t   cmd_mkdir=\"\\\"sh -c 'env INSTALL_PATH=$(INSTALL_PATH) GRAS_ROOT=$(GRAS_ROOT) \\\n"
423      "\t                        SIMGRID_URL=$(SIMGRID_URL) SIMGRID_VERSION=$(SIMGRID_VERSION) GRAS_PROJECT=$(GRAS_PROJECT) \\\n"
424      "\t                        GRAS_PROJECT_URL=$(GRAS_PROJECT_URL)  mkdir -p $(INSTALL_PATH) 2>&1'\\\"\";\\\n"
425      "\t   cmd_make=\"\\\"sh -c 'env INSTALL_PATH=$(INSTALL_PATH) GRAS_ROOT=$(GRAS_ROOT) \\\n"
426      "\t                        SIMGRID_URL=$(SIMGRID_URL) SIMGRID_VERSION=$(SIMGRID_VERSION) GRAS_PROJECT=$(GRAS_PROJECT) \\\n"
427      "\t                        GRAS_PROJECT_URL=$(GRAS_PROJECT_URL)  make -C $(INSTALL_PATH) -f "MAKEFILE_FILENAME_REMOTE" $(ACTION) 2>&1'\\\"\";\\\n"
428      "\t   if echo $$site | grep  '%%' >/dev/null ; then \\\n"
429      "\t     echo \"----[ Compile on $$machine2 (behind $$machine) ]----\";\\\n"
430      "\t   else \\\n"
431      "\t     machine=$$site;\\\n"
432      "\t     echo \"----[ Compile on $$machine ]----\";\\\n"
433      "\t   fi;\\\n"
434      "\t   if echo $$site | grep  '%%' >/dev/null ; then \\\n"
435      "\t     if ssh $$machine \"ssh -A $$machine2 $$cmd_mkdir\" 2>&1 > $(SRCDIR)/buildlogs/$$site.log;\\\n"
436      "\t     then true; else failed=1;echo \"Failed (check $(SRCDIR)/buildlogs/$$site.log)\"; fi;\\\n"
437      "\t   else \\\n"
438      "\t     if ssh $$machine \"eval $$cmd_mkdir\" 2>&1 > $(SRCDIR)/buildlogs/$$site.log ;\\\n"
439      "\t     then true; else failed=1;echo \"Failed (check $(SRCDIR)/buildlogs/$$site.log)\"; fi; \\\n"
440      "\t   fi;\\\n"
441      "\t   echo \"-- Copy the data over\"; \\\n"
442      "\t   scp "MAKEFILE_FILENAME_REMOTE" $$site:$(INSTALL_PATH) ;\\\n"
443      "\t   echo \"-- Compiling... (the output gets into $(SRCDIR)/buildlogs/$$site.log)\"; \\\n"
444      "\t   if echo $$site | grep  '%%' >/dev/null ; then \\\n"
445      "\t     if ssh $$machine \"ssh -A $$machine2 $$cmd_make\" 2>&1 >> $(SRCDIR)/buildlogs/$$site.log;\\\n"
446      "\t     then echo \"Sucessful\"; else failed=1;echo \"Failed (check $(SRCDIR)/buildlogs/$$site.log)\"; fi;echo; \\\n"
447      "\t   else \\\n"
448      "\t     if ssh $$machine \"eval $$cmd_make\" 2>&1 >> $(SRCDIR)/buildlogs/$$site.log ;\\\n"
449      "\t     then echo \"Sucessful\"; else failed=1;echo \"Failed (check $(SRCDIR)/buildlogs/$$site.log)\"; fi;echo; \\\n"
450      "\t   fi;\\\n"
451      "\t done;\n",project,project,project);
452    */
453   fclose(FICOUT);
454 }
455
456 static void print(void *p)
457 {
458   printf("%p", p);
459 }
460 #endif