1 /*-------------------------------------------------------------------------
2 SDCCsystem - SDCC system & pipe functions
4 Written By - Sandeep Dutta . sandeep.dutta@usa.net (1999)
6 This program is free software; you can redistribute it and/or modify it
7 under the terms of the GNU General Public License as published by the
8 Free Software Foundation; either version 2, or (at your option) any
11 This program is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 GNU General Public License for more details.
16 You should have received a copy of the GNU General Public License
17 along with this program; if not, write to the Free Software
18 Foundation, 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
20 In other words, you are welcome to use, share and improve this program.
21 You are forbidden to forbid anyone else to use, share and improve
22 what you give them. Help stamp out software-hoarding!
23 -------------------------------------------------------------------------*/
26 /* avoid DATADIR definition clash :-( */
39 #include "SDCCglobl.h"
41 #include "dbuf_string.h"
42 #include "SDCCsystem.h"
46 set
*binPathSet
= NULL
; /* set of binary paths */
50 * get command and arguments from command line
54 split_command (const char *cmd_line
, char **command
, char **params
)
56 const char *p
, *cmd_start
;
61 /* skip leading spaces */
62 for (p
= cmd_line
; isspace (*p
); p
++)
81 while (*p
!= '\0' && !isspace (*p
))
86 while (*p
!= '\0' && *p
!= delim
)
93 str
= Safe_alloc (len
+ 1);
94 strncpy (str
, cmd_start
, len
);
101 /* skip spaces before parameters */
107 *params
= Safe_strdup (p
);
113 * 1) if the command is specified by path, try it
114 * 2) try to find the command in predefined path's
122 file_exists (const char* path
)
124 DWORD attr
= GetFileAttributes (path
);
125 return attr
!= INVALID_FILE_ATTRIBUTES
126 && 0 == (attr
& FILE_ATTRIBUTE_DIRECTORY
);
130 * cmd.exe via _popen doesn't accept something like this:
131 * "program" "argument"
133 * ""program with possible spaces" "argument with spaces""
135 * ""program with possible spaces" argument_without_spaces"
136 * (note: the first and last " are also needed).
137 * The arguments at this point already contain double quotes,
138 * if necessary, so the following code formats the line as:
139 * ""program" arguments".
142 #define EXE_EXT ".exe"
145 * merge command and parameters to command line
149 merge_command (const char *command
, const char *params
)
153 /* allocate extra space for '""', '"' ' and ""\0" */
154 dbuf_init (&dbuf
, strlen (command
) + strlen (params
) + 6);
156 dbuf_append_str (&dbuf
, "\"\"");
157 dbuf_append_str (&dbuf
, command
);
158 dbuf_append_str (&dbuf
, "\" ");
159 dbuf_append_str (&dbuf
, params
);
160 dbuf_append_str (&dbuf
, "\"");
162 return dbuf_detach_c_str (&dbuf
);
167 * check if path/command exist by converting it to short file name
168 * if it exists, compose with args and return it
172 compose_command_line (const char *path
, const char *command
, const char *args
)
175 struct dbuf_s cmdPath
;
177 dbuf_init (&cmdPath
, PATH_MAX
);
180 dbuf_makePath (&cmdPath
, path
, command
);
182 dbuf_append_str (&cmdPath
, command
);
184 /* Does cmdPath or cmdPath.exe exist? */
185 cmd_exists
= file_exists (dbuf_c_str (&cmdPath
));
188 dbuf_append_str (&cmdPath
, EXE_EXT
);
189 cmd_exists
= file_exists (dbuf_c_str (&cmdPath
));
193 /* compose the command line */
194 return merge_command (dbuf_c_str (&cmdPath
), args
);
198 /* path/command not found */
205 get_path (const char *cmd
)
212 /* get the command */
213 split_command (cmd
, &command
, &args
);
215 if (NULL
== (cmdLine
= compose_command_line(NULL
, command
, args
)))
217 /* not an absolute path: try to find the command in predefined binary paths */
218 if (NULL
!= (path
= (char *)setFirstItem (binPathSet
)))
220 while (NULL
== (cmdLine
= compose_command_line (path
, command
, args
)) &&
221 NULL
!= (path
= (char *)setNextItem (binPathSet
)))
227 /* didn't found the command in predefined binary paths: try with PATH */
230 if (NULL
!= (envPath
= getenv("PATH")))
232 /* make a local copy; strtok() will modify it */
233 envPath
= Safe_strdup (envPath
);
235 if (NULL
!= (path
= strtok (envPath
, ";")))
237 while (NULL
== (cmdLine
= compose_command_line (path
, command
, args
)) &&
238 NULL
!= (path
= strtok (NULL
, ";")))
246 /* didn't found it; probably this won't help neither :-( */
248 cmdLine
= merge_command (command
, args
);
261 * merge command and parameters to command line
265 merge_command (const char *command
, const char *params
)
268 char *s
= shell_escape (command
);
270 /* allocate extra space for ' ' and '\0' */
271 dbuf_init (&dbuf
, strlen (command
) + strlen (params
) + 2);
273 dbuf_append_str (&dbuf
, s
);
274 dbuf_append (&dbuf
, " ", 1);
275 dbuf_append_str (&dbuf
, params
);
277 return dbuf_detach_c_str (&dbuf
);
282 * check if the path is relative or absolute (if contains the dir separator)
286 has_path (const char *path
)
288 return dbuf_splitPath (path
, NULL
, NULL
);
293 get_path (const char *cmd
)
295 const char *cmdLine
= NULL
;
300 /* get the command */
301 split_command (cmd
, &command
, &args
);
303 if (!has_path (command
))
305 /* try to find the command in predefined binary paths */
306 if (NULL
!= (path
= (char *)setFirstItem (binPathSet
)))
313 dbuf_init (&dbuf
, PATH_MAX
);
314 dbuf_makePath (&dbuf
, path
, command
);
315 cmdPath
= dbuf_detach (&dbuf
);
318 if (0 == access (cmdPath
, X_OK
))
320 /* compose the command line */
321 cmdLine
= merge_command (cmdPath
, args
);
324 } while (NULL
!= (path
= (char *)setNextItem (binPathSet
)));
327 cmdLine
= merge_command (command
, args
);
337 * the command is defined with absolute path:
343 return Safe_strdup (cmd
);
350 * call an external program with arguments
354 sdcc_system (const char *cmd
)
357 const char *cmdLine
= get_path (cmd
);
359 assert (NULL
!= cmdLine
);
361 if (options
.verboseExec
)
362 printf ("+ %s\n", cmdLine
);
364 e
= system (cmdLine
);
366 if (options
.verboseExec
&& e
)
367 printf ("+ %s returned errorcode %d\n", cmdLine
, e
);
376 * pipe an external program with arguments
380 #define sdcc_popen_read(cmd) _popen ((cmd), "rt")
382 sdcc_pclose (FILE *fp
)
387 #define sdcc_popen_read(cmd) popen ((cmd), "r")
389 sdcc_pclose (FILE *fp
)
396 sdcc_popen (const char *cmd
)
399 const char *cmdLine
= get_path (cmd
);
401 assert (NULL
!= cmdLine
);
403 if (options
.verboseExec
)
405 printf ("+ %s\n", cmdLine
);
408 fp
= sdcc_popen_read (cmdLine
);