/* -*-c-*- -------------- mixvm_loop.c : * Implementation of mix vm command loop. * ------------------------------------------------------------------ * Copyright (C) 2000, 2001, 2002, 2004, 2006, 2007 Free Software Foundation, Inc. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * */ #include <stdio.h> #include <string.h> #include <mixlib/mix.h> #include <mixlib/mix_config.h> #include <mixlib/mix_vm.h> #include <mixlib/mix_device.h> #include <mixlib/mix_vm_dump.h> #ifdef MAKE_GUILE #include <mixguile/mixguile.h> #endif #include "mixvm_command.h" #include "mixvm_loop.h" #ifdef HAVE_LIBHISTORY # include <readline/history.h> #else # define add_history(x) ((void)0) #endif #ifdef HAVE_LIBREADLINE # include <readline/readline.h> #else /* !HAVE_LIBREADLINE */ static char * readline (char *prompt) { enum {LINE_LEN = 256}; char *line = g_new (char, LINE_LEN); printf ("%s", prompt); return fgets (line, LINE_LEN, stdin); } #endif /* HAVE_LIBREADLINE */ /* A static variable for holding the line. */ static char *line_read = (char *)NULL; #define PROMPT_LEN 128 static char PROMPT[PROMPT_LEN + 1] = {'M', 'I', 'X', '>', ' '}; static const char *CONFIG_FILE_ = "mixvm.config"; static const char *PROMPT_KEY_ = "Prompt"; /* Read a string, and return a pointer to it. Returns NULL on EOF. */ static char * rl_gets () { /* If the buffer has already been allocated, return the memory to the free pool. */ if (line_read) { g_free (line_read); line_read = (char *)NULL; } /* Get a line from the user. */ line_read = readline ((char *)PROMPT); /* If the line has any text in it, save it on the history. */ if (line_read && *line_read) add_history (line_read); return (line_read); } /* The main command loop of the virtual machine */ static mix_config_t *config_ = NULL; static mix_vm_cmd_dispatcher_t * init_mixvm_ (const gchar *file, gboolean use_emacs) { static const gchar *HISTORY_FILE = "mixvm.history"; static gint HISTORY_SIZE = 100; config_ = mix_config_new (NULL, CONFIG_FILE_); mix_config_set_autosave (config_, TRUE); if (!mix_config_get_history_file (config_)) mix_config_set_history_file (config_, HISTORY_FILE); if (mix_config_get_history_size (config_) == 0) mix_config_set_history_size (config_, HISTORY_SIZE); mix_vmloop_set_prompt (mix_config_get (config_, PROMPT_KEY_)); return mixvm_cmd_init (config_, (char *)file, use_emacs); } void mix_vmloop_set_prompt (const gchar *prompt) { if (prompt) { g_snprintf (PROMPT, PROMPT_LEN, "%s ", prompt); mix_config_update (config_, PROMPT_KEY_, prompt); } } static void loop_ (int argc, char *argv[]) { while ( mixvm_cmd_exec (rl_gets ()) ) ; mix_config_delete (config_); } void mix_vmloop (int argc, char *argv[], gboolean initfile, const gchar *file, gboolean use_emacs) { #ifdef MAKE_GUILE mix_vm_cmd_dispatcher_t *dis = init_mixvm_ (file, use_emacs); mixguile_init (argc, argv, initfile, loop_, dis); #else (void) init_mixvm_ (file, use_emacs); loop_ (argc, argv); #endif } /* run a program and exit */ void mix_vmrun (const gchar *code_file, gboolean dump, gboolean ptime) { gchar *time_cmd = ptime? g_strdup ("stime on") : g_strdup ("stime off"); gchar *run_cmd = g_strdup ("run"); gchar *dump_cmd = dump? g_strdup ("pall") : NULL; gboolean result; init_mixvm_ (code_file, FALSE); result = mixvm_cmd_exec (time_cmd) && mixvm_cmd_exec (run_cmd); if (result && dump) mixvm_cmd_exec (dump_cmd); mix_config_set_autosave (config_, FALSE); mix_config_delete (config_); g_free(time_cmd); g_free(run_cmd); if (dump_cmd) g_free(dump_cmd); }