Diff for /freem/src/mdebug.c between versions 1.8 and 1.13

version 1.8, 2025/04/30 20:03:09 version 1.13, 2025/05/20 18:07:41
Line 24 Line 24
  *   along with FreeM.  If not, see <https://www.gnu.org/licenses/>.   *   along with FreeM.  If not, see <https://www.gnu.org/licenses/>.
  *   *
  *   $Log$   *   $Log$
    *   Revision 1.13  2025/05/20 18:07:41  snw
    *   Add completion to debugger
    *
    *   Revision 1.12  2025/05/01 21:02:31  snw
    *   Documentation updates
    *
    *   Revision 1.11  2025/05/01 17:02:30  snw
    *   Further debugging improvements
    *
    *   Revision 1.10  2025/05/01 04:45:15  snw
    *   Improve backtraces
    *
    *   Revision 1.9  2025/05/01 03:56:29  snw
    *   -m
    *
  *   Revision 1.8  2025/04/30 20:03:09  snw   *   Revision 1.8  2025/04/30 20:03:09  snw
  *   Work on entryref parser   *   Work on entryref parser
  *   *
Line 86  extern int read_history (); Line 101  extern int read_history ();
 #include "freem.h"  #include "freem.h"
 #include "mref.h"  #include "mref.h"
   
   #ifdef HAVE_LIBREADLINE
   char *dbg_commands[] = {
       "backtrace",
       "continue",
       "examine",
       "exit",
       "halt",
       "next",
       "quit",
       "step",
       "trace",
       "watch",
       NULL
   };
   
   char **dbg_completion(const char *, int, int);
   char *dbg_generator(const char *, int);
   
   char **dbg_completion(const char *text, int start, int end)
   {
       if (start > 0) return NULL;
       rl_attempted_completion_over = 1;
       return rl_completion_matches (text, dbg_generator);
   }
   
   char *dbg_generator(const char *text, int state)
   {
       static int list_index, len;
       char *name;
   
       if (!state) {
           list_index = 0;
           len = strlen(text);
       }
   
       while ((name = dbg_commands[list_index++])) {
           if (strncmp (name, text, len) == 0) {
               return strdup (name);
           }
       }
   
       return NULL;
   }
   #endif
   
 dbg_watch dbg_watchlist[MAXWATCH];    /* list of watchpoints */  dbg_watch dbg_watchlist[MAXWATCH];    /* list of watchpoints */
 short dbg_enable_watch;               /* 0 = watches disabled, 1 = watches enabled */  short dbg_enable_watch;               /* 0 = watches disabled, 1 = watches enabled */
 int dbg_pending_watches;  int dbg_pending_watches;
Line 107  void dbg_init (void) Line 167  void dbg_init (void)
   
 }  }
   
 int debugger (int entry_mode, char *curcode)  int debugger (int entry_mode, char *entryref)
 {  {
           
 #if defined(HAVE_LIBREADLINE)      #if defined(HAVE_LIBREADLINE)    
Line 116  int debugger (int entry_mode, char *curc Line 176  int debugger (int entry_mode, char *curc
           
     char *dbg_buf;      char *dbg_buf;
     char dbg_prompt[256];      char dbg_prompt[256];
       int dbg_argc;
     HIST_ENTRY **dbg_hist;      HIST_ENTRY **dbg_hist;
     int dbg_hist_idx;      int dbg_hist_idx;
     HIST_ENTRY *dbg_hist_ent;      HIST_ENTRY *dbg_hist_ent;
     char rouname_buf[256];      char rouname_buf[256];
     char *savptr;      char *savptr;
     char *dbg_cmd;      char *dbg_cmd;
       char *dbarg;
     int do_prompt = FALSE;      int do_prompt = FALSE;
       char cur_code[512];
       char tbuf[512];
       char tbuf2[512];
     register int i;      register int i;
           register int j;
   
       rl_attempted_completion_function = dbg_completion;
     set_io (UNIX);      set_io (UNIX);
           
     if (first_entry) {      if (first_entry) {
Line 176  int debugger (int entry_mode, char *curc Line 243  int debugger (int entry_mode, char *curc
             stcpy (rouname_buf, rou_name);              stcpy (rouname_buf, rou_name);
         }          }
         else {          else {
             char tbuf[256];  
               
             getraddress (tbuf, nstx);              getraddress (tbuf, nstx);
             stcpy (rouname_buf, &(tbuf[3]));              stcpy (rouname_buf, &(tbuf[3]));
         }          }
         stcnv_m2c (rouname_buf);          stcnv_m2c (rouname_buf);
                   
         snprintf (dbg_prompt, sizeof (dbg_prompt) - 1, "%s $STACK=%d [DEBUG]> ", rouname_buf, nstx);          if (stepmode != STEPMODE_NONE) {
               getraddress (tbuf, nstx);
               stcnv_m2c (tbuf);
               strcpy (tbuf2, &(tbuf[3]));
               routine_get_line (tbuf2, cur_code);
   
         if ((stepmode == STEPMODE_LINE) || (stepmode == STEPMODE_CMD)) {              printf ("%s\n", cur_code);
             char codbuf[512];  
             stcpy (codbuf, curcode);  
             stcnv_m2c (codbuf);  
               
             printf ("%s\n", codbuf);  
         }          }
           
           snprintf (dbg_prompt, sizeof (dbg_prompt) - 1, "%s $STACK=%d [DEBUG]> ", rouname_buf, nstx);
           if (dbg_enable_watch && dbg_pending_watches) dbg_dump_watchlist ();
         dbg_buf = readline (dbg_prompt);          dbg_buf = readline (dbg_prompt);
           
         if (!dbg_buf) continue;          if (!dbg_buf) continue;
           if (strlen (dbg_buf) < 1) continue;
   
           dbg_argc = 1;
           for (i = 0; i < strlen (dbg_buf); i++) {
               if (dbg_buf[i] == ' ') dbg_argc++;
           }
           dbarg = dbg_buf;
         savptr = dbg_buf;          savptr = dbg_buf;
         dbg_cmd = strtok_r (dbg_buf, " ", &savptr);          dbg_cmd = strtok_r (dbg_buf, " ", &savptr);
           
           dbarg += strlen (dbg_cmd) + 1;
   
         if ((strcmp (dbg_cmd, "exit") == 0) || (strcmp (dbg_cmd, "quit") == 0)) {          if ((strcmp (dbg_cmd, "exit") == 0) || (strcmp (dbg_cmd, "quit") == 0)) {
             first_entry = TRUE;              first_entry = TRUE;
             stepmode = STEPMODE_NONE;              stepmode = STEPMODE_NONE;
Line 206  int debugger (int entry_mode, char *curc Line 281  int debugger (int entry_mode, char *curc
             set_io (MUMPS);              set_io (MUMPS);
             return FALSE;              return FALSE;
         }          }
           else if ((strcmp (dbg_cmd, "examine") == 0) || (strcmp (dbg_cmd, "e") == 0)) {
               char *glvn;
               char key[STRLEN];
               char res[STRLEN];
               
               if (dbg_argc < 2) {
                   printf ("debug:  syntax error\n");
               }
               else {
                   glvn = dbarg;
                   name_to_key (key, glvn, STRLEN - 1);
                   if (key[0] != '^') {
                       symtab (get_sym, key, res);
                   }
                   else {
                       if (key[1] != '$') {
                           global (get_sym, key, res);
                       }
                       else {
                           ssvn (get_sym, key, res);
                       }
                   }
   
                   if (merr () != OK) {
                       if (merr () == M6) {
                           printf ("examine:  local variable %s is not defined\n", glvn);
                           merr_clear ();
                       }
                       else if (merr () == M7) {
                           printf ("examine:  global variable %s is not defined\n", glvn);
                           merr_clear ();
                       }
                       else {
                           printf ("examine:  error retrieving %s\n", glvn);
                           merr_clear ();
                       }
                   }                
                   else {
                       stcnv_m2c (res);
                       printf ("%s=\"%s\"\n", glvn, res);
                   }
   
               }
           }
           else if ((strcmp (dbg_cmd, "trace") == 0) || (strcmp (dbg_cmd, "t") == 0)) {
               if (trace_mode == 0) {                
                   trace_mode = 1;
                   printf ("debug:  trace on\n");
               }
               else {
                   trace_mode = 0;
                   printf ("debug:  trace off\n");
               }
               ssvn_job_update ();
           }
         else if ((strcmp (dbg_cmd, "step") == 0) || (strcmp (dbg_cmd, "s") == 0)) {          else if ((strcmp (dbg_cmd, "step") == 0) || (strcmp (dbg_cmd, "s") == 0)) {
             stepmode = STEPMODE_CMD;              stepmode = STEPMODE_CMD;
             return TRUE;              return TRUE;
Line 219  int debugger (int entry_mode, char *curc Line 349  int debugger (int entry_mode, char *curc
             return TRUE;              return TRUE;
         }          }
         else if ((strcmp (dbg_cmd, "backtrace") == 0) || (strcmp (dbg_cmd, "bt") == 0)) {          else if ((strcmp (dbg_cmd, "backtrace") == 0) || (strcmp (dbg_cmd, "bt") == 0)) {
             char tmpbuf[256];              char tmpbuf[1024];
             char ecbuf[256];              char ecbuf[256];
             char lref[256];              char m_cmd[10];
             char bt_mcode[256];              char lref[1024];
               char bt_mcode[1024];
                           
             printf ("%-10s%s\n", "$STACK", "ENTRY");              printf ("%-10s%-10s%s\n", "$STACK", "COMMAND", "PLACE");
             printf ("%-10s%s\n", "======", "=====");              printf ("%-10s%-10s%s\n", "======", "=======", "=====");
                           
                           
             for (i = 1; i <= nstx; i++) getraddress (callerr[i], i);              for (i = 1; i <= nstx; i++) getraddress (callerr[i], i);
Line 233  int debugger (int entry_mode, char *curc Line 364  int debugger (int entry_mode, char *curc
                                   
                 stcpy (tmpbuf, callerr[i]);                  stcpy (tmpbuf, callerr[i]);
                 stcnv_m2c (tmpbuf);                  stcnv_m2c (tmpbuf);
                   strcpy (ecbuf, &(tmpbuf[1]));
   
                   for (j = 0; j < strlen (ecbuf); j++) {
                       if (ecbuf[j] == ')') {
                           ecbuf[j] = '\0';
                           break;
                       }
                   }
   
                   switch (ecbuf[0]) {
                       case 'F':
                           sprintf (m_cmd, "FOR");
                           break;
                       case 'D':
                           sprintf (m_cmd, "DO");
                           break;
                   }
                               
                                   
                 printf ("%-10d%s\n", i, &(tmpbuf[3]));                  printf ("%-10d%-10s%s\n", i, m_cmd, &(tmpbuf[3]));
                 stcpy (lref, &(tmpbuf[3]));                  stcpy (lref, &(tmpbuf[3]));
                 stcnv_m2c (lref);                  stcnv_m2c (lref);
                 if (routine_get_line (lref, bt_mcode) != NULL) {                  if (routine_get_line (lref, bt_mcode) != NULL) {
                     stcnv_m2c (bt_mcode);                      stcnv_m2c (bt_mcode);
                     printf ("\t%s\n", bt_mcode);                      printf ("%-20s%s\n", " ", bt_mcode);
                 }                  }
             }              }
               stcpy (ecbuf, etrap);
               stcnv_m2c (ecbuf);
               printf ("ETRAP:\t%s\n", ecbuf);
               stcpy (ecbuf, ztrap[nstx]);
               stcnv_m2c (ecbuf);
               printf ("ZTRAP:\t%s\n", ecbuf);
               
           }
           else if ((strcmp (dbg_cmd, "halt") == 0) || (strcmp (dbg_cmd, "h") == 0)) {
               printf ("debug:  forcing halt\n");          
               exit (0);
           }
           else if ((strcmp (dbg_cmd, "watch") == 0) || (strcmp (dbg_cmd, "w") == 0)) {
               char watchmode;
               char *glvn;
               
               if (dbg_argc == 1) {
                   switch (dbg_enable_watch) {
                       case 0:
                           dbg_enable_watch = 1;
                           printf ("debug:  watchpoints enabled\n");
                           break;
                       case 1:
                           dbg_enable_watch = 0;
                           printf ("debug:  watchpoints disabled\n");
                           break;
                   }
               }
               else {
                   watchmode = dbarg[0];
                   glvn = &(dbarg[1]);
                   
                   switch (watchmode) {
                       case '+':
                           dbg_add_watch (glvn);
                           break;
                       case '-':
                           dbg_remove_watch (glvn);
                           break;
                       case '?':
                           dbg_dump_watch (glvn);
                           break;
                       default:
                           printf ("debug:  watch mode must be +, -, or ?\n");
                           break;
                   }
   
                   set_io (UNIX);
               }               
         }          }
         else {          else {
             printf ("DEBUG:  invalid command\r\n");              printf ("DEBUG:  invalid command\r\n");
Line 286  dbg_watch *dbg_add_watch (char *varnam) Line 484  dbg_watch *dbg_add_watch (char *varnam)
         return NULL;          return NULL;
     }      }
   
     if ((dbg_watchlist[index].varnam = (char *) malloc (256 * sizeof (char))) == NULL) {      if ((dbg_watchlist[index].varnam = (char *) calloc (256, sizeof (char))) == NULL) {
         set_io (UNIX);          set_io (UNIX);
         fprintf (stderr, "Could not allocate memory for the new watchlist entry.\n");          fprintf (stderr, "Could not allocate memory for the new watchlist entry.\n");
         set_io (MUMPS);          set_io (MUMPS);
Line 298  dbg_watch *dbg_add_watch (char *varnam) Line 496  dbg_watch *dbg_add_watch (char *varnam)
     dbg_watchlist[index].chgct = 0;      dbg_watchlist[index].chgct = 0;
   
     set_io (UNIX);      set_io (UNIX);
     fprintf (stderr, "Added '%s' to the watchlist.\n", dbg_get_watch_name (varnam));      fprintf (stderr, "Added '%s' to the watchlist.\n", varnam);
     set_io (MUMPS);      set_io (MUMPS);
   
     return NULL;      return NULL;
Line 310  void dbg_dump_watchlist (void) Line 508  void dbg_dump_watchlist (void)
         register int i;          register int i;
   
         for (i = 0; i < MAXWATCH; i++) {          for (i = 0; i < MAXWATCH; i++) {
                 if (dbg_watchlist[i].firect) {              if (dbg_watchlist[i].firect) {
                         dbg_dump_watch (dbg_watchlist[i].varnam);                  dbg_dump_watch (dbg_watchlist[i].varnam);
                 }              }
         }          }
   
         dbg_pending_watches = 0;          dbg_pending_watches = 0;
Line 423  char *dbg_get_watch_name (char *varnam) Line 621  char *dbg_get_watch_name (char *varnam)
   
 void dbg_fire_watch (char *varnam) {  void dbg_fire_watch (char *varnam) {
   
         dbg_watch *w;      dbg_watch *w;
       
         if ((w = dbg_find_watch (varnam)) == NULL) {      if ((w = dbg_find_watch (varnam)) == NULL) {
                 return;          return;
         }      }
       
         w->chgct++;      w->chgct++;
         w->firect++;      w->firect++;
         dbg_pending_watches++;      dbg_pending_watches++;
               
 }  }

Removed from v.1.8  
changed lines
  Added in v.1.13


FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>