2 * viking -- GPS Data and Topo Analyzer, Explorer, and Manager
4 * Copyright (C) 2003-2005, Evan Battaglia <gtoevan@gmx.net>
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
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, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
22 /* babel.c: running external programs and redirecting to TRWLayers.
23 * GPSBabel may not be necessary for everything -- for instance,
24 * use a_babel_convert_from_shellcommand with input_file_type == NULL
25 * for an external program that outputs GPX.
36 #ifdef HAVE_SYS_WAIT_H
43 #include <glib/gstdio.h>
45 /* in the future we could have support for other shells (change command strings), or not use a shell at all */
46 #define BASH_LOCATION "/bin/bash"
48 gboolean
a_babel_convert( VikTrwLayer
*vt
, const char *babelargs
, BabelStatusFunc cb
, gpointer user_data
)
54 gchar
*bargs
= g_strconcat(babelargs
, " -i gpx", NULL
);
56 if ((fd_src
= g_file_open_tmp("tmp-viking.XXXXXX", &name_src
, NULL
)) >= 0) {
57 f
= fdopen(fd_src
, "w");
58 a_gpx_write_file(vt
, f
);
61 ret
= a_babel_convert_from ( vt
, bargs
, cb
, name_src
, user_data
);
70 /* Runs args[0] with the arguments and uses the GPX module
71 * to import the GPX data into layer vt. Assumes that upon
72 * running the command, the data will appear in the (usually
73 * temporary) file name_dst.
75 * cb: callback that is run upon new data from STDOUT (?)
76 * (TODO: STDERR would be nice since we usually redirect STDOUT)
77 * user_data: passed along to cb
79 * returns TRUE on success
82 gboolean
babel_general_convert_from( VikTrwLayer
*vt
, BabelStatusFunc cb
, gchar
**args
, const gchar
*name_dst
, gpointer user_data
)
90 PROCESS_INFORMATION pi
;
93 ZeroMemory( &si
, sizeof(si
) );
94 ZeroMemory( &pi
, sizeof(pi
) );
96 si
.dwFlags
= STARTF_USESHOWWINDOW
;
97 si
.wShowWindow
= SW_HIDE
;
99 cmd
= g_strjoinv( " ", args
);
100 args2
= g_strsplit(cmd
, "\\", 0);
102 cmd
= g_strjoinv( "\\\\", args2
);
104 args2
= g_strsplit(cmd
, "/", 0);
106 cmd
= g_strjoinv( "\\\\", args2
);
109 NULL
, // No module name (use command line).
110 (LPTSTR
)cmd
, // Command line.
111 NULL
, // Process handle not inheritable.
112 NULL
, // Thread handle not inheritable.
113 FALSE
, // Set handle inheritance to FALSE.
114 0, // No creation flags.
115 NULL
, // Use parent's environment block.
116 NULL
, // Use parent's starting directory.
117 &si
, // Pointer to STARTUPINFO structure.
118 &pi
) // Pointer to PROCESS_INFORMATION structure.
120 g_warning( "CreateProcess failed");
124 WaitForSingleObject(pi
.hProcess
, INFINITE
);
125 WaitForSingleObject(pi
.hThread
, INFINITE
);
127 CloseHandle(pi
.hThread
);
128 CloseHandle(pi
.hProcess
);
131 cb(BABEL_DONE
, NULL
, user_data
);
133 f
= g_fopen(name_dst
, "r");
134 a_gpx_read_file( vt
, f
);
147 gboolean
babel_general_convert_from( VikTrwLayer
*vt
, BabelStatusFunc cb
, gchar
**args
, const gchar
*name_dst
, gpointer user_data
)
149 gboolean ret
= FALSE
;
151 GError
*error
= NULL
;
156 if (!g_spawn_async_with_pipes (NULL
, args
, NULL
, 0, NULL
, NULL
, &pid
, NULL
, &babel_stdout
, NULL
, &error
)) {
157 g_warning("Error : %s", error
->message
);
163 diag
= fdopen(babel_stdout
, "r");
164 setvbuf(diag
, NULL
, _IONBF
, 0);
166 while (fgets(line
, sizeof(line
), diag
)) {
168 cb(BABEL_DIAG_OUTPUT
, line
, user_data
);
171 cb(BABEL_DONE
, NULL
, user_data
);
174 waitpid(pid
, NULL
, 0);
175 g_spawn_close_pid(pid
);
177 f
= g_fopen(name_dst
, "r");
179 a_gpx_read_file ( vt
, f
);
190 gboolean
a_babel_convert_from( VikTrwLayer
*vt
, const char *babelargs
, BabelStatusFunc cb
, const char *from
, gpointer user_data
)
195 gboolean ret
= FALSE
;
199 if ((fd_dst
= g_file_open_tmp("tmp-viking.XXXXXX", &name_dst
, NULL
)) >= 0) {
203 gpsbabel_loc
= g_find_program_in_path("gpsbabel");
206 gchar
*unbuffer_loc
= g_find_program_in_path("unbuffer");
207 cmd
= g_strdup_printf ( "%s%s%s %s -o gpx %s %s",
208 unbuffer_loc
? unbuffer_loc
: "",
209 unbuffer_loc
? " " : "",
216 g_free ( unbuffer_loc
);
218 if ( g_shell_parse_argv(cmd
, &nb_args
, &args
, NULL
) ) {
219 ret
= babel_general_convert_from ( vt
, cb
, args
, name_dst
, user_data
);
231 /* Runs the input command in a shell (bash) and optionally uses GPSBabel to convert from input_file_type.
232 * If input_file_type is NULL, doesn't use GPSBabel. Input must be GPX (or Geocaching *.loc)
234 * Uses babel_general_convert_from to actually run the command. This function
235 * prepares the command and temporary file, and sets up the arguments for bash.
237 gboolean
a_babel_convert_from_shellcommand ( VikTrwLayer
*vt
, const char *input_cmd
, const char *input_file_type
, BabelStatusFunc cb
, gpointer user_data
)
241 gboolean ret
= FALSE
;
244 if ((fd_dst
= g_file_open_tmp("tmp-viking.XXXXXX", &name_dst
, NULL
)) >= 0) {
245 gchar
*shell_command
;
246 if ( input_file_type
)
247 shell_command
= g_strdup_printf("%s | gpsbabel -i %s -f - -o gpx -F %s", input_cmd
, input_file_type
, name_dst
);
249 shell_command
= g_strdup_printf("%s > %s", input_cmd
, name_dst
);
251 g_debug("%s: %s", __FUNCTION__
, shell_command
);
254 args
= g_malloc(sizeof(gchar
*)*4);
255 args
[0] = BASH_LOCATION
;
257 args
[2] = shell_command
;
260 ret
= babel_general_convert_from ( vt
, cb
, args
, name_dst
, user_data
);
262 g_free ( shell_command
);
270 gboolean
a_babel_convert_from_url ( VikTrwLayer
*vt
, const char *url
, const char *input_type
, BabelStatusFunc cb
, gpointer user_data
)
272 static DownloadOptions options
= {NULL
, 0, a_check_html_file
};
275 gboolean ret
= FALSE
;
279 g_debug("%s: input_type=%s url=%s", __FUNCTION__
, input_type
, url
);
281 if ((fd_src
= g_file_open_tmp("tmp-viking.XXXXXX", &name_src
, NULL
)) >= 0) {
285 babelargs
= g_strdup_printf(" -i %s", input_type
);
287 fetch_ret
= a_http_download_get_url(url
, "", name_src
, &options
);
289 ret
= a_babel_convert_from( vt
, babelargs
, NULL
, name_src
, NULL
);
300 gboolean
babel_general_convert_to( VikTrwLayer
*vt
, BabelStatusFunc cb
, gchar
**args
, const gchar
*name_src
, gpointer user_data
)
306 if (!a_file_export(vt
, name_src
, FILE_TYPE_GPX
)) {
307 g_warning("%s(): error exporting to %s", __FUNCTION__
, name_src
);
312 PROCESS_INFORMATION pi
;
314 ZeroMemory( &si
, sizeof(si
) );
315 ZeroMemory( &pi
, sizeof(pi
) );
317 si
.dwFlags
= STARTF_USESHOWWINDOW
;
318 si
.wShowWindow
= SW_HIDE
;
321 cmd
= g_strjoinv( " ", args
);
322 args2
= g_strsplit(cmd
, "\\", 0);
323 cmd
= g_strjoinv( "\\\\", args2
);
325 args2
= g_strsplit(cmd
, "/", 0);
327 cmd
= g_strjoinv( "\\\\", args2
);
330 NULL
, // No module name (use command line).
331 (LPTSTR
)cmd
, // Command line.
332 NULL
, // Process handle not inheritable.
333 NULL
, // Thread handle not inheritable.
334 FALSE
, // Set handle inheritance to FALSE.
335 0, // No creation flags.
336 NULL
, // Use parent's environment block.
337 NULL
, // Use parent's starting directory.
338 &si
, // Pointer to STARTUPINFO structure.
339 &pi
) // Pointer to PROCESS_INFORMATION structure.
341 g_warning( "CreateProcess failed" );
346 WaitForSingleObject(pi
.hProcess
, INFINITE
);
347 WaitForSingleObject(pi
.hThread
, INFINITE
);
349 CloseHandle(pi
.hThread
);
350 CloseHandle(pi
.hProcess
);
353 cb(BABEL_DONE
, NULL
, user_data
);
366 gboolean
babel_general_convert_to( VikTrwLayer
*vt
, BabelStatusFunc cb
, gchar
**args
, const gchar
*name_src
, gpointer user_data
)
368 gboolean ret
= FALSE
;
370 GError
*error
= NULL
;
373 if (!a_file_export(vt
, name_src
, FILE_TYPE_GPX
)) {
374 g_warning("%s(): error exporting to %s", __FUNCTION__
, name_src
);
378 if (!g_spawn_async_with_pipes (NULL
, args
, NULL
, 0, NULL
, NULL
, &pid
, NULL
, &babel_stdout
, NULL
, &error
)) {
379 g_warning("Error : %s", error
->message
);
385 diag
= fdopen(babel_stdout
, "r");
386 setvbuf(diag
, NULL
, _IONBF
, 0);
388 while (fgets(line
, sizeof(line
), diag
)) {
390 cb(BABEL_DIAG_OUTPUT
, line
, user_data
);
393 cb(BABEL_DONE
, NULL
, user_data
);
396 waitpid(pid
, NULL
, 0);
397 g_spawn_close_pid(pid
);
406 gboolean
a_babel_convert_to( VikTrwLayer
*vt
, const char *babelargs
, BabelStatusFunc cb
, const char *to
, gpointer user_data
)
411 gboolean ret
= FALSE
;
415 if ((fd_src
= g_file_open_tmp("tmp-viking.XXXXXX", &name_src
, NULL
)) >= 0) {
419 gpsbabel_loc
= g_find_program_in_path("gpsbabel");
422 gchar
*unbuffer_loc
= g_find_program_in_path("unbuffer");
423 cmd
= g_strdup_printf ( "%s%s%s %s -i gpx %s %s",
424 unbuffer_loc
? unbuffer_loc
: "",
425 unbuffer_loc
? " " : "",
432 g_free ( unbuffer_loc
);
433 g_debug ( "%s: %s", __FUNCTION__
, cmd
);
434 if ( g_shell_parse_argv(cmd
, &nb_args
, &args
, NULL
) ) {
435 ret
= babel_general_convert_to ( vt
, cb
, args
, name_src
, user_data
);