build: do not included generated files in distribution
[openocd.git] / src / openocd.c
1 /***************************************************************************
2 * Copyright (C) 2005 by Dominic Rath *
3 * Dominic.Rath@gmx.de *
4 * *
5 * Copyright (C) 2007-2010 Øyvind Harboe *
6 * oyvind.harboe@zylin.com *
7 * *
8 * Copyright (C) 2008 Richard Missenden *
9 * richard.missenden@googlemail.com *
10 * *
11 * This program is free software; you can redistribute it and/or modify *
12 * it under the terms of the GNU General Public License as published by *
13 * the Free Software Foundation; either version 2 of the License, or *
14 * (at your option) any later version. *
15 * *
16 * This program is distributed in the hope that it will be useful, *
17 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
18 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
19 * GNU General Public License for more details. *
20 * *
21 * You should have received a copy of the GNU General Public License *
22 * along with this program; if not, write to the *
23 * Free Software Foundation, Inc., *
24 * 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. *
25 ***************************************************************************/
26
27 #ifdef HAVE_CONFIG_H
28 #include "config.h"
29 #endif
30
31 #include "openocd.h"
32 #include <jtag/driver.h>
33 #include <jtag/jtag.h>
34 #include <transport/transport.h>
35 #include <helper/ioutil.h>
36 #include <helper/util.h>
37 #include <helper/configuration.h>
38 #include <flash/nor/core.h>
39 #include <flash/nand/core.h>
40 #include <pld/pld.h>
41 #include <flash/mflash.h>
42
43 #include <server/server.h>
44 #include <server/gdb_server.h>
45
46 #ifdef HAVE_STRINGS_H
47 #include <strings.h>
48 #endif
49
50
51 #define OPENOCD_VERSION \
52 "Open On-Chip Debugger " VERSION RELSTR " (" PKGBLDDATE ")"
53
54 /* Give scripts and TELNET a way to find out what version this is */
55 static int jim_version_command(Jim_Interp *interp, int argc,
56 Jim_Obj * const *argv)
57 {
58 if (argc > 2)
59 {
60 return JIM_ERR;
61 }
62 const char *str = "";
63 char * version_str;
64 version_str = OPENOCD_VERSION;
65
66 if (argc == 2)
67 str = Jim_GetString(argv[1], NULL);
68
69 if (strcmp("git", str) == 0)
70 {
71 version_str = GITVERSION;
72 }
73
74 Jim_SetResult(interp, Jim_NewStringObj(interp, version_str, -1));
75
76 return JIM_OK;
77 }
78
79 static int log_target_callback_event_handler(struct target *target, enum target_event event, void *priv)
80 {
81 switch (event)
82 {
83 case TARGET_EVENT_GDB_START:
84 target->display = 0;
85 break;
86 case TARGET_EVENT_GDB_END:
87 target->display = 1;
88 break;
89 case TARGET_EVENT_HALTED:
90 if (target->display)
91 {
92 /* do not display information when debugger caused the halt */
93 target_arch_state(target);
94 }
95 break;
96 default:
97 break;
98 }
99
100 return ERROR_OK;
101 }
102
103 static bool init_at_startup = true;
104
105 COMMAND_HANDLER(handle_noinit_command)
106 {
107 if (CMD_ARGC != 0)
108 return ERROR_COMMAND_SYNTAX_ERROR;
109 init_at_startup = false;
110 return ERROR_OK;
111 }
112
113 /* OpenOCD can't really handle failure of this command. Patches welcome! :-) */
114 COMMAND_HANDLER(handle_init_command)
115 {
116
117 if (CMD_ARGC != 0)
118 return ERROR_COMMAND_SYNTAX_ERROR;
119
120 int retval;
121 static int initialized = 0;
122 if (initialized)
123 return ERROR_OK;
124
125 initialized = 1;
126
127 retval = command_run_line(CMD_CTX, "target init");
128 if (ERROR_OK != retval)
129 return ERROR_FAIL;
130
131 if ((retval = adapter_init(CMD_CTX)) != ERROR_OK)
132 {
133 /* we must be able to set up the debug adapter */
134 return retval;
135 }
136
137 LOG_DEBUG("Debug Adapter init complete");
138
139 /* "transport init" verifies the expected devices are present;
140 * for JTAG, it checks the list of configured TAPs against
141 * what's discoverable, possibly with help from the platform's
142 * JTAG event handlers. (which require COMMAND_EXEC)
143 */
144 command_context_mode(CMD_CTX, COMMAND_EXEC);
145
146 retval = command_run_line(CMD_CTX, "transport init");
147 if (ERROR_OK != retval)
148 return ERROR_FAIL;
149
150 LOG_DEBUG("Examining targets...");
151 if (target_examine() != ERROR_OK)
152 LOG_DEBUG("target examination failed");
153
154 command_context_mode(CMD_CTX, COMMAND_CONFIG);
155
156 if (command_run_line(CMD_CTX, "flash init") != ERROR_OK)
157 return ERROR_FAIL;
158
159 if (command_run_line(CMD_CTX, "mflash init") != ERROR_OK)
160 return ERROR_FAIL;
161
162 if (command_run_line(CMD_CTX, "nand init") != ERROR_OK)
163 return ERROR_FAIL;
164
165 if (command_run_line(CMD_CTX, "pld init") != ERROR_OK)
166 return ERROR_FAIL;
167 command_context_mode(CMD_CTX, COMMAND_EXEC);
168
169 /* initialize telnet subsystem */
170 gdb_target_add_all(all_targets);
171
172 target_register_event_callback(log_target_callback_event_handler, CMD_CTX);
173
174 return ERROR_OK;
175 }
176
177 COMMAND_HANDLER(handle_add_script_search_dir_command)
178 {
179 if (CMD_ARGC != 1)
180 return ERROR_COMMAND_SYNTAX_ERROR;
181
182 add_script_search_dir(CMD_ARGV[0]);
183
184 return ERROR_OK;
185 }
186
187
188 static int jim_stacktrace_command(Jim_Interp *interp, int argc,
189 Jim_Obj * const *argv)
190 {
191 if (argc != 1)
192 {
193 return JIM_ERR;
194 }
195 Jim_Obj * stacktrace = Jim_DuplicateObj(interp, interp->stackTrace);
196
197 /* insert actual error site at beginning of list*/
198 Jim_Obj *procname = Jim_NewStringObj(interp, "", -1); /* Uhhh... don't know this one. */
199 Jim_ListInsertElements(interp, stacktrace, 0, 1, &procname);
200 Jim_Obj *filename = Jim_NewStringObj(interp, interp->errorFileName, -1);
201 Jim_ListInsertElements(interp, stacktrace, 1, 1, &filename);
202 Jim_Obj *line = Jim_NewIntObj(interp, interp->errorLine);
203 Jim_ListInsertElements(interp, stacktrace, 2, 1, &line);
204
205 Jim_SetResult(interp, stacktrace);
206
207 return JIM_OK;
208 }
209
210 static const struct command_registration openocd_command_handlers[] = {
211 {
212 .name = "version",
213 .jim_handler = jim_version_command,
214 .mode = COMMAND_ANY,
215 .help = "show program version",
216 },
217 {
218 .name = "noinit",
219 .handler = &handle_noinit_command,
220 .mode = COMMAND_CONFIG,
221 .help = "Prevent 'init' from being called at startup.",
222 },
223 {
224 .name = "init",
225 .handler = &handle_init_command,
226 .mode = COMMAND_ANY,
227 .help = "Initializes configured targets and servers. "
228 "Changes command mode from CONFIG to EXEC. "
229 "Unless 'noinit' is called, this command is "
230 "called automatically at the end of startup.",
231
232 },
233 {
234 .name = "add_script_search_dir",
235 .handler = &handle_add_script_search_dir_command,
236 .mode = COMMAND_ANY,
237 .help = "dir to search for config files and scripts",
238
239 },
240 {
241 .name = "stacktrace",
242 .jim_handler = jim_stacktrace_command,
243 .mode = COMMAND_ANY,
244 .help = "returns the stacktrace as a list of triples: proc, file, line."
245 "The stack trace is reset when a new stack trace is being built after "
246 "a new failure has occurred.",
247 },
248 COMMAND_REGISTRATION_DONE
249 };
250
251 static int openocd_register_commands(struct command_context *cmd_ctx)
252 {
253 return register_commands(cmd_ctx, NULL, openocd_command_handlers);
254 }
255
256 struct command_context *global_cmd_ctx;
257
258 /* NB! this fn can be invoked outside this file for non PC hosted builds
259 * NB! do not change to 'static'!!!!
260 */
261 struct command_context *setup_command_handler(Jim_Interp *interp)
262 {
263 log_init();
264 LOG_DEBUG("log_init: complete");
265
266 const char *startup = openocd_startup_tcl;
267 struct command_context *cmd_ctx = command_init(startup, interp);
268
269 /* register subsystem commands */
270 typedef int (*command_registrant_t)(struct command_context *cmd_ctx_value);
271 static const command_registrant_t command_registrants[] = {
272 &openocd_register_commands,
273 &server_register_commands,
274 &gdb_register_commands,
275 &log_register_commands,
276 &transport_register_commands,
277 &interface_register_commands,
278 &target_register_commands,
279 &flash_register_commands,
280 &nand_register_commands,
281 &pld_register_commands,
282 &mflash_register_commands,
283 NULL
284 };
285 for (unsigned i = 0; NULL != command_registrants[i]; i++)
286 {
287 int retval = (*command_registrants[i])(cmd_ctx);
288 if (ERROR_OK != retval)
289 {
290 command_done(cmd_ctx);
291 return NULL;
292 }
293 }
294 LOG_DEBUG("command registration: complete");
295
296 LOG_OUTPUT(OPENOCD_VERSION "\n"
297 "Licensed under GNU GPL v2\n");
298
299 global_cmd_ctx = cmd_ctx;
300
301 return cmd_ctx;
302 }
303
304 /** OpenOCD runtime meat that can become single-thread in future. It parse
305 * commandline, reads configuration, sets up the target and starts server loop.
306 * Commandline arguments are passed into this function from openocd_main().
307 */
308 static int openocd_thread(int argc, char *argv[], struct command_context *cmd_ctx)
309 {
310 int ret;
311
312 if (parse_cmdline_args(cmd_ctx, argc, argv) != ERROR_OK)
313 return EXIT_FAILURE;
314
315 if (server_preinit() != ERROR_OK)
316 return EXIT_FAILURE;
317
318 ret = parse_config_file(cmd_ctx);
319 if (ret != ERROR_OK)
320 return EXIT_FAILURE;
321
322 ret = server_init(cmd_ctx);
323 if (ERROR_OK != ret)
324 return EXIT_FAILURE;
325
326 ret = command_run_line(cmd_ctx, "init_targets");
327 if (ERROR_OK != ret)
328 ret = EXIT_FAILURE;
329
330 if (init_at_startup)
331 {
332 ret = command_run_line(cmd_ctx, "init");
333 if (ERROR_OK != ret)
334 return EXIT_FAILURE;
335 }
336
337 server_loop(cmd_ctx);
338
339 server_quit();
340
341 return ret;
342 }
343
344 /* normally this is the main() function entry, but if OpenOCD is linked
345 * into application, then this fn will not be invoked, but rather that
346 * application will have it's own implementation of main(). */
347 int openocd_main(int argc, char *argv[])
348 {
349 int ret;
350
351 /* initialize commandline interface */
352 struct command_context *cmd_ctx;
353
354 cmd_ctx = setup_command_handler(NULL);
355
356 if (util_init(cmd_ctx) != ERROR_OK)
357 return EXIT_FAILURE;
358
359 if (ioutil_init(cmd_ctx) != ERROR_OK)
360 return EXIT_FAILURE;
361
362 LOG_OUTPUT("For bug reports, read\n\t"
363 "http://openocd.berlios.de/doc/doxygen/bugs.html"
364 "\n");
365
366 command_context_mode(cmd_ctx, COMMAND_CONFIG);
367 command_set_output_handler(cmd_ctx, configuration_output_handler, NULL);
368
369 /* Start the executable meat that can evolve into thread in future. */
370 ret = openocd_thread(argc, argv, cmd_ctx);
371
372 unregister_all_commands(cmd_ctx, NULL);
373
374 /* free commandline interface */
375 command_done(cmd_ctx);
376
377 adapter_quit();
378
379 return ret;
380 }

Linking to existing account procedure

If you already have an account and want to add another login method you MUST first sign in with your existing account and then change URL to read https://review.openocd.org/login/?link to get to this page again but this time it'll work for linking. Thank you.

SSH host keys fingerprints

1024 SHA256:YKx8b7u5ZWdcbp7/4AeXNaqElP49m6QrwfXaqQGJAOk gerrit-code-review@openocd.zylin.com (DSA)
384 SHA256:jHIbSQa4REvwCFG4cq5LBlBLxmxSqelQPem/EXIrxjk gerrit-code-review@openocd.org (ECDSA)
521 SHA256:UAOPYkU9Fjtcao0Ul/Rrlnj/OsQvt+pgdYSZ4jOYdgs gerrit-code-review@openocd.org (ECDSA)
256 SHA256:A13M5QlnozFOvTllybRZH6vm7iSt0XLxbA48yfc2yfY gerrit-code-review@openocd.org (ECDSA)
256 SHA256:spYMBqEYoAOtK7yZBrcwE8ZpYt6b68Cfh9yEVetvbXg gerrit-code-review@openocd.org (ED25519)
+--[ED25519 256]--+
|=..              |
|+o..   .         |
|*.o   . .        |
|+B . . .         |
|Bo. = o S        |
|Oo.+ + =         |
|oB=.* = . o      |
| =+=.+   + E     |
|. .=o   . o      |
+----[SHA256]-----+
2048 SHA256:0Onrb7/PHjpo6iVZ7xQX2riKN83FJ3KGU0TvI0TaFG4 gerrit-code-review@openocd.zylin.com (RSA)