3 /* gras_stub_generator - creates the main() to use a GRAS program */
5 /* Copyright (c) 2003-2007 Martin Quinson, Arnaud Legrand, Malek Cherier. */
6 /* All rights reserved. */
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. */
11 /* specific to Borland Compiler */
17 #include "xbt/sysdep.h"
18 #include "xbt/function_types.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"
28 /* stupid stubs so that it compiles on windows */
29 void generate_sim(char *project)
33 void generate_rl(char *project)
37 void generate_makefile_local(char *project, char *deployment)
41 /* real implementation */
42 XBT_LOG_EXTERNAL_DEFAULT_CATEGORY(stubgen);
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"
63 /**********************************************/
64 /**** Generate the file for the simulator *****/
65 /**********************************************/
67 const char *SIM_PREEMBULE =
68 "/* specific to Borland Compiler */\n"
69 "#ifdef __BORLANDC__\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";
78 #define SIM_LAUNCH_FUNC \
79 "int launch_%s(int argc, char **argv) {\n" \
80 " char **myargv=argv;\n" \
81 " int myargc=argc;\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" \
91 " retcode = %s(myargc,myargv);\n" \
92 " if (myargv != argv)\n" \
94 " return retcode;\n" \
97 const char *SIM_MAIN_POSTEMBULE = "\n"
99 " gras_launch_application(argv[2]);\n"
101 " /* Run the simulation */\n"
104 " /* cleanup the place */\n"
106 " if (gras_log)\n" " free(gras_log);\n" " return 0;\n" "}\n";
111 void generate_sim(char *project)
113 xbt_dict_cursor_t cursor = NULL;
116 char *filename = NULL;
119 /* Output file: <projet>_simulator.c */
120 filename = xbt_new(char, strlen(project) + strlen(SIM_SOURCENAME));
121 sprintf(filename, SIM_SOURCENAME, project);
123 FICOUT = fopen(filename, "w");
125 xbt_assert1(FICOUT, "Unable to open %s for writing", filename);
127 fprintf(FICOUT, "%s\n", warning);
128 fprintf(FICOUT, "%s", SIM_PREEMBULE);
130 xbt_dict_foreach(process_function_set, cursor, key, data) {
131 fprintf(FICOUT, "int %s(int argc,char *argv[]);\n", key);
134 fprintf(FICOUT, "\n");
136 xbt_dict_foreach(process_function_set, cursor, key, data) {
137 fprintf(FICOUT, "int launch_%s(int argc,char *argv[]);\n", key);
140 fprintf(FICOUT, "\n%s\n", warning);
142 xbt_dict_foreach(process_function_set, cursor, key, data) {
143 fprintf(FICOUT, SIM_LAUNCH_FUNC, key, key);
145 fprintf(FICOUT, "\n%s\n", warning);
147 fprintf(FICOUT, "%s", "/* specific to Borland Compiler */\n"
148 "#ifdef __BORLANDDC__\n" "#pragma argsused\n" "#endif\n\n");
150 fprintf(FICOUT, "%s", "int main (int argc,char *argv[]) {\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");
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,
164 fprintf(FICOUT, "%s", SIM_MAIN_POSTEMBULE);
169 /**********************************************/
170 /**** Generate the file for the real life *****/
171 /**********************************************/
172 void generate_rl(char *project)
174 xbt_dict_cursor_t cursor = NULL;
177 char *filename = NULL;
180 xbt_dict_foreach(process_function_set, cursor, key, data) {
183 strlen(project) + strlen(RL_SOURCENAME) + strlen(key));
185 sprintf(filename, RL_SOURCENAME, project, key);
187 FICOUT = fopen(filename, "w");
188 xbt_assert1(FICOUT, "Unable to open %s for writing", filename);
190 fprintf(FICOUT, "\n%s\n", warning);
191 fprintf(FICOUT, "/* specific to Borland Compiler */\n"
192 "#ifdef __BORLANDC__\n"
195 "#include <stdio.h>\n"
196 "#include <signal.h>\n"
197 "#include <gras.h>\n"
199 "XBT_PUBLIC_DATA(const char *) _gras_procname;\n"
201 "int %s(int argc, char *argv[]);\n"
203 "/* specific to Borland Compiler */\n"
204 "#ifdef __BORLANDC__\n"
207 "int main(int argc, char *argv[]){\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);
219 void generate_makefile_am(char *project, char *deployment)
221 xbt_dict_cursor_t cursor = NULL;
224 char *filename = NULL;
227 filename = xbt_new(char, strlen(project) + strlen(MAKEFILE_FILENAME_AM));
228 sprintf(filename, MAKEFILE_FILENAME_AM, project);
230 FICOUT = fopen(filename, "w");
231 xbt_assert1(FICOUT, "Unable to open %s for writing", filename);
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);
238 xbt_dict_foreach(process_function_set, cursor, key, data) {
239 fprintf(FICOUT, " ");
240 fprintf(FICOUT, RL_BINARYNAME, project, key);
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");
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");
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);
266 xbt_dict_foreach(process_function_set, cursor, key, data) {
267 fprintf(FICOUT, " ");
268 fprintf(FICOUT, RL_SOURCENAME, project, key);
270 fprintf(FICOUT, "\n");
272 fprintf(FICOUT, "\n# generate temps\n");
274 "\n# A rule to generate the source file each time the deployment file changes\n");
276 xbt_dict_foreach(process_function_set, cursor, key, data) {
277 fprintf(FICOUT, RL_SOURCENAME, project, key);
278 fprintf(FICOUT, " ");
280 fprintf(FICOUT, SIM_SOURCENAME, project);
281 fprintf(FICOUT, ": %s\n", deployment);
282 fprintf(FICOUT, "\tgras_stub_generator %s %s >/dev/null\n", project,
287 void generate_makefile_local(char *project, char *deployment)
289 xbt_dict_cursor_t cursor = NULL;
292 char *filename = NULL;
296 xbt_new(char, strlen(project) + strlen(MAKEFILE_FILENAME_LOCAL));
297 sprintf(filename, MAKEFILE_FILENAME_LOCAL, project);
299 FICOUT = fopen(filename, "w");
300 xbt_assert1(FICOUT, "Unable to open %s for writing", filename);
306 "#### THIS FILE WAS GENERATED, DO NOT EDIT BEFORE RENAMING IT\n"
309 fprintf(FICOUT, "## Variable declarations\n"
310 "PROJECT_NAME=%s\n" "DISTDIR=gras-$(PROJECT_NAME)\n\n", project);
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");
322 fprintf(FICOUT, "PRECIOUS_C_FILES ?= %s.c\n", project);
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);
329 fprintf(FICOUT, "\n");
332 "OBJ_FILES = $(patsubst %%.c,%%.o,$(PRECIOUS_C_FILES))\n");
334 fprintf(FICOUT, "BIN_FILES = ");
336 fprintf(FICOUT, SIM_BINARYNAME " ", project);
337 xbt_dict_foreach(process_function_set, cursor, key, data) {
338 fprintf(FICOUT, RL_BINARYNAME " ", project, key);
340 fprintf(FICOUT, "\n");
344 "## By default, build all the binaries\n"
345 "all: $(BIN_FILES)\n" "\n");
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, " ");
353 fprintf(FICOUT, SIM_SOURCENAME, project);
354 fprintf(FICOUT, ": %s\n", deployment);
355 fprintf(FICOUT, "\tgras_stub_generator %s %s >/dev/null\n", project,
358 fprintf(FICOUT, "\n## Generate the binaries\n");
359 fprintf(FICOUT, SIM_BINARYNAME ": " SIM_OBJNAME " $(OBJ_FILES)\n",
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);
367 "\t$(CC) $(INCLUDES) $(DEFS) $(CFLAGS) $^ $(LIBS_RL) $(LIBS) $(LDADD) -o $@ \n");
372 "\t$(CC) $(INCLUDES) $(DEFS) $(CFLAGS) $^ $(LIBS) $(LDADD) -o $@ \n"
375 "\t$(CC) $(INCLUDES) $(DEFS) $(CFLAGS) -c -o $@ $<\n" "\n");
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 */ );
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);
396 "\trm -rf $(DISTDIR)\n"
397 "\n" ".SUFFIXES:\n" ".PHONY : clean\n" "\n");
399 fprintf(FICOUT, "############ REMOTE COMPILING #########\n");
402 xbt_dict_foreach(machine_set,cursor,key,data) {
403 fprintf(FICOUT, "%s ",key);
405 fprintf(FICOUT,"\n");
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"
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"
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"
431 "\t machine=$$site;\\\n"
432 "\t echo \"----[ Compile on $$machine ]----\";\\\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"
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"
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"
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"
451 "\t done;\n",project,project,project);
456 static void print(void *p)