22 cmd_handler_t function;
26 /* The list of cmds */
27 static cmd_t *cmd_list = NULL;
30 #define ALIAS_NAME_MAX 32
31 typedef struct cmd_alias_s
33 char name[ALIAS_NAME_MAX];
35 struct cmd_alias_s *next;
38 /* The list of aliases */
39 static cmd_alias_t *cmd_alias_list = NULL;
42 /* add a new console command */
43 void cmd_addcommand(char *cmd_name, cmd_handler_t cmd_func)
47 Assert(cmd_name != NULL);
49 for (cmd = cmd_list; cmd; cmd = cmd->next) {
50 if (!stricmp(cmd_name, cmd->name))
53 con_printf(CON_NORMAL, "command %s already exists, not adding\n", cmd_name);
58 /* create command, insert at front of list */
59 MALLOC(cmd, cmd_t, 1);
61 cmd->function = cmd_func;
63 con_printf(CON_DEBUG, "cmd_addcommand: added %s\n", cmd->name);
68 typedef struct cmd_queue_s
71 struct cmd_queue_s *next;
74 /* The list of commands to be executed */
75 static cmd_queue_t *cmd_queue_head = NULL;
76 static cmd_queue_t *cmd_queue_tail = NULL;
79 void cvar_cmd_set(int argc, char **argv);
82 /* execute a parsed command */
83 void cmd_execute(int argc, char **argv)
88 for (cmd = cmd_list; cmd; cmd = cmd->next) {
89 if (!stricmp(argv[0], cmd->name)) {
90 con_printf(CON_DEBUG, "cmd_execute: executing %s\n", argv[0]);
91 return cmd->function(argc, argv);
95 for (alias = cmd_alias_list; alias; alias = alias->next) {
96 if (!stricmp(argv[0], alias->name)) {
97 con_printf(CON_DEBUG, "cmd_execute: pushing alias \"%s\": %s\n", alias->name, alias->value);
98 cmd_insert(alias->value);
104 { // set value of cvar
105 char *new_argv[argc+1];
109 for (i = 0; i < argc; i++)
110 new_argv[i+1] = argv[i];
111 cvar_cmd_set(argc + 1, new_argv);
116 /* Parse an input string */
117 void cmd_parse(char *input)
119 char buffer[CMD_MAX_LENGTH];
120 char *tokens[CMD_MAX_TOKENS];
124 Assert(input != NULL);
126 /* Strip leading spaces */
127 for (i=0; isspace(input[i]); i++) ;
128 strncpy( buffer, &input[i], CMD_MAX_LENGTH );
130 //printf("lead strip \"%s\"\n",buffer);
131 l = (int)strlen(buffer);
132 /* If command is empty, give up */
135 /* Strip trailing spaces */
136 for (i=l-1; i>0 && isspace(buffer[i]); i--) ;
138 //printf("trail strip \"%s\"\n",buffer);
140 /* Split into tokens */
141 l = (int)strlen(buffer);
145 for (i=1; i<l; i++) {
146 if (buffer[i] == '"') {
147 tokens[num_tokens - 1] = &buffer[++i];
148 while (i < l && buffer[i] != '"')
153 if (isspace(buffer[i]) || buffer[i] == '=') {
155 while (isspace(buffer[i+1]) && (i+1 < l)) i++;
156 tokens[num_tokens++] = &buffer[i+1];
160 /* Check for matching commands */
161 cmd_execute(num_tokens, tokens);
165 int cmd_queue_wait = 0;
167 int cmd_queue_process(void)
171 while (!cmd_queue_wait && cmd_queue_head) {
172 cmd = cmd_queue_head;
173 cmd_queue_head = cmd_queue_head->next;
175 cmd_queue_tail = NULL;
177 con_printf(CON_DEBUG, "cmd_queue_process: processing %s\n", cmd->command_line);
178 cmd_parse(cmd->command_line); // Note, this may change the queue
180 d_free(cmd->command_line);
184 if (cmd_queue_wait > 0) {
186 if (Function_mode == FMODE_GAME) {
187 con_printf(CON_DEBUG, "cmd_queue_process: waiting\n");
196 /* execute until there are no commands left */
197 void cmd_queue_flush(void)
199 while (cmd_queue_process()) {
204 /* Add some commands to the queue to be executed */
205 void cmd_enqueue(int insert, char *input)
207 cmd_queue_t *new, *head, *tail;
208 char output[CMD_MAX_LENGTH];
211 Assert(input != NULL);
218 /* Strip leading spaces */
219 while(isspace(*input) || *input == ';')
222 /* If command is empty, give up */
226 /* Find the end of this line (\n, ;, or nul) */
233 } else if ( *input == '\n' || (!quoted && *input == ';') ) {
237 } while ((*optr++ = *input++));
240 /* make a new queue item, add it to list */
241 MALLOC(new, cmd_queue_t, 1);
242 new->command_line = d_strdup(output);
251 con_printf(CON_DEBUG, "cmd_enqueue: adding %s\n", output);
255 /* add our list to the head of the main list */
257 tail->next = cmd_queue_head;
259 cmd_queue_tail = tail;
261 cmd_queue_head = head;
262 con_printf(CON_DEBUG, "cmd_enqueue: added to front of list\n");
264 /* add our list to the tail of the main list */
266 cmd_queue_head = head;
268 cmd_queue_tail->next = head;
270 cmd_queue_tail = tail;
271 con_printf(CON_DEBUG, "cmd_enqueue: added to back of list\n");
275 void cmd_enqueuef(int insert, char *fmt, ...)
278 char buf[CMD_MAX_LENGTH];
280 va_start (arglist, fmt);
281 vsnprintf (buf, CMD_MAX_LENGTH, fmt, arglist);
284 cmd_enqueue(insert, buf);
288 /* Attempt to autocomplete an input string */
289 char *cmd_complete(char *input)
294 int len = (int)strlen(input);
299 for (ptr = cmd_list; ptr != NULL; ptr = ptr->next)
300 if (!strnicmp(input, ptr->name, len))
303 for (aptr = cmd_alias_list; aptr != NULL; aptr = aptr->next)
304 if (!strnicmp(input, aptr->name, len))
307 return cvar_complete(input);
312 void cmd_alias(int argc, char **argv)
315 char buf[CMD_MAX_LENGTH] = "";
318 if (argc == 2 && !stricmp(argv[1], "-h")) {
319 con_printf(CON_NORMAL, "%s <name> <commands>\n", argv[0]);
320 con_printf(CON_NORMAL, " define <name> as an alias for <commands>\n");
321 con_printf(CON_NORMAL, "%s <name>\n", argv[0]);
322 con_printf(CON_NORMAL, " show the current definition of <name>\n");
323 con_printf(CON_NORMAL, "%s\n", argv[0]);
324 con_printf(CON_NORMAL, " show all defined aliases\n");
329 con_printf(CON_NORMAL, "aliases:\n");
330 for (alias = cmd_alias_list; alias; alias = alias->next)
331 con_printf(CON_NORMAL, "%s: %s\n", alias->name, alias->value);
336 for (alias = cmd_alias_list; alias; alias = alias->next)
337 if (!stricmp(argv[1], alias->name)) {
338 con_printf(CON_NORMAL, "%s: %s\n", alias->name, alias->value);
342 con_printf(CON_NORMAL, "alias: %s not found\n", argv[1]);
346 for (i = 2; i < argc; i++) {
348 strncat(buf, " ", CMD_MAX_LENGTH);
349 strncat(buf, argv[i], CMD_MAX_LENGTH);
352 for (alias = cmd_alias_list; alias; alias = alias->next) {
353 if (!stricmp(argv[1], alias->name)) {
354 d_free(alias->value);
355 alias->value = d_strdup(buf);
360 MALLOC(alias, cmd_alias_t, 1);
361 strncpy(alias->name, argv[1], ALIAS_NAME_MAX);
362 alias->value = d_strdup(buf);
363 alias->next = cmd_alias_list;
364 cmd_alias_list = alias;
369 void cmd_unalias(int argc, char **argv)
371 cmd_alias_t *alias, *prev_alias = NULL;
373 if (argc != 2 || (argc == 2 && !stricmp(argv[1], "-h"))) {
374 con_printf(CON_NORMAL, "%s <name>\n", argv[0]);
375 con_printf(CON_NORMAL, " undefine the alias <name>\n");
379 for (alias = cmd_alias_list; alias ; alias = alias->next) {
380 if (!stricmp(argv[1], alias->name))
386 con_printf(CON_NORMAL, "alias: %s not found\n", argv[1]);
391 prev_alias->next = alias->next;
393 cmd_alias_list = alias->next;
395 d_free(alias->value);
400 /* echo to console */
401 void cmd_echo(int argc, char **argv)
403 char buf[CMD_MAX_LENGTH] = "";
406 if (argc == 2 && !stricmp(argv[1], "-h")) {
407 con_printf(CON_NORMAL, "usage: %s [text]\n", argv[0]);
408 con_printf(CON_NORMAL, " write <text> to the console\n");
413 for (i = 1; i < argc; i++) {
415 strncat(buf, " ", CMD_MAX_LENGTH);
416 strncat(buf, argv[i], CMD_MAX_LENGTH);
418 con_printf(CON_NORMAL, "%s\n", buf);
422 void cmd_exec(int argc, char **argv) {
423 cmd_queue_t *new, *head, *tail;
425 char line[CMD_MAX_LENGTH] = "";
427 if (argc != 2 || (argc == 2 && !stricmp(argv[1], "-h"))) {
428 con_printf(CON_NORMAL, "usage: %s <file>\n", argv[0]);
429 con_printf(CON_NORMAL, " execute <file>\n");
436 f = PHYSFSX_openReadBuffered(argv[1]);
438 con_printf(CON_CRITICAL, "exec: %s not found\n", argv[1]);
441 while (PHYSFSX_gets(f, line)) {
442 /* make a new queue item, add it to list */
443 MALLOC(new, cmd_queue_t, 1);
444 new->command_line = d_strdup(line);
453 con_printf(CON_DEBUG, "cmd_exec: adding %s\n", line);
457 /* add our list to the head of the main list */
459 tail->next = cmd_queue_head;
461 cmd_queue_tail = tail;
463 cmd_queue_head = head;
464 con_printf(CON_DEBUG, "cmd_exec: added to front of list\n");
469 void cmd_help(int argc, char **argv)
473 if (argc > 2 || (argc == 2 && !stricmp(argv[1], "-h"))) {
474 con_printf(CON_NORMAL, "usage: %s [command]\n", argv[0]);
475 con_printf(CON_NORMAL, " get help for <command>, or list all commands if not specified.\n");
481 con_printf(CON_NORMAL, "Available commands:\n");
482 for (cmd = cmd_list; cmd; cmd = cmd->next) {
483 con_printf(CON_NORMAL, " %s\n", cmd->name);
489 cmd_insertf("%s -h", argv[1]);
494 void cmd_wait(int argc, char **argv)
496 if (argc > 2 || (argc == 2 && !stricmp(argv[1], "-h"))) {
497 con_printf(CON_NORMAL, "usage: %s [n]\n", argv[0]);
498 con_printf(CON_NORMAL, " stop processing commands, resume in <n> cycles (default 1)\n");
506 cmd_queue_wait = atoi(argv[1]);
517 p = ((cmd_t *)p)->next;
523 d_free(((cmd_alias_t *)p)->value);
525 p = ((cmd_alias_t *)p)->next;
533 cmd_addcommand("alias", cmd_alias);
534 cmd_addcommand("unalias", cmd_unalias);
535 cmd_addcommand("echo", cmd_echo);
536 cmd_addcommand("exec", cmd_exec);
537 cmd_addcommand("help", cmd_help);
538 cmd_addcommand("wait", cmd_wait);