]> git.saurik.com Git - bison.git/commitdiff
errors: show carets
authorTheophile Ranquet <ranquet@lrde.epita.fr>
Fri, 30 Nov 2012 13:34:56 +0000 (14:34 +0100)
committerTheophile Ranquet <ranquet@lrde.epita.fr>
Wed, 5 Dec 2012 12:46:01 +0000 (13:46 +0100)
* src/locations.c (caret_info): New, persistant information useful
for...
(location_caret): New, print a caret.
(cleanup_caret): Release caret_info cleanly, call it...
* src/main.c (main): Here.
* src/complain.c (error_message): Call location_caret here.

src/complain.c
src/getargs.c
src/gram.c
src/location.c
src/location.h
src/main.c

index b063c6b622b38cb036b03cb1a5b6762f10507487..ede0ccf4f5cb122d601e329dc5eeaafb1a5997d7 100644 (file)
@@ -74,11 +74,15 @@ error_message (location *loc,
   vfprintf (stderr, message, args);
   {
     size_t l = strlen (message);
   vfprintf (stderr, message, args);
   {
     size_t l = strlen (message);
-    if (l < 2 || message[l-2] != ':' || message[l-1] != ' ') {
-      putc ('\n', stderr);
-      fflush (stderr);
-    }
+    if (l < 2 || message[l - 2] != ':' || message[l - 1] != ' ')
+      {
+        putc ('\n', stderr);
+        fflush (stderr);
+        if (loc && feature_flag & feature_caret)
+          location_caret (stderr, *loc);
+      }
   }
   }
+  fflush (stderr);
 }
 
 /** Wrap error_message() with varargs handling. */
 }
 
 /** Wrap error_message() with varargs handling. */
index 1b7c6f1bd1be3f8a64ad6c0ea25369dd43d46c18..c8865bfbd96d2e652b91cefc569b5e63c7076160 100644 (file)
@@ -646,7 +646,7 @@ getargs (int argc, char *argv[])
        exit (EXIT_SUCCESS);
 
       case 'f':
        exit (EXIT_SUCCESS);
 
       case 'f':
-        FLAGS_ARGMATCH (flag, optarg);
+        FLAGS_ARGMATCH (feature, optarg);
         break;
 
       case 'W':
         break;
 
       case 'W':
index d1b3804337bd55a622896ae571453fa696c77c62..5730e596026729c24c2cec1f7bd4f85527d92a8b 100644 (file)
@@ -308,11 +308,16 @@ grammar_rules_useless_report (const char *message)
   for (r = 0; r < nrules ; ++r)
     if (!rules[r].useful)
       {
   for (r = 0; r < nrules ; ++r)
     if (!rules[r].useful)
       {
-        warn_at (rules[r].location, "%s: ", message);
-        if (warnings_flag & warnings_other)
+        if (feature_flag & feature_caret)
+          warn_at (rules[r].location, "%s", message);
+        else
           {
           {
-            rule_print (&rules[r], stderr);
-            fflush (stderr);
+            warn_at (rules[r].location, "%s: ", message);
+            if (warnings_flag & warnings_other)
+              {
+                rule_print (&rules[r], stderr);
+                fflush (stderr);
+              }
           }
       }
 }
           }
       }
 }
index fa1b53cb5611a81a19319c464e2ab094ced5009a..6e5306c62019a7cf5a9576527d0b83d9ac4d881f 100644 (file)
@@ -138,6 +138,89 @@ location_print (FILE *out, location loc)
   return res;
 }
 
   return res;
 }
 
+
+/* Persistant data used by location_caret to avoid reopening and rereading the
+   same file all over for each error.  */
+struct caret_info
+{
+  FILE* source;
+  size_t line;
+  size_t offset;
+};
+
+static struct caret_info caret_info = { NULL, 1, 0 };
+
+/* Free any allocated ressources and close any open file handles that are
+   left-over by the usage of location_caret.  */
+void
+cleanup_caret ()
+{
+  if (caret_info.source)
+    fclose (caret_info.source);
+}
+
+/* Output to OUT the line and caret corresponding to location LOC.  */
+void
+location_caret (FILE *out, location loc)
+{
+  /* FIXME: find a way to support X-file locations, and only open once each
+     file. That would make the procedure future-proof.  */
+  if (! (caret_info.source
+         || (caret_info.source = fopen (loc.start.file, "r")))
+      || loc.start.column == -1 || loc.start.line == -1)
+    return;
+
+  /* If the line we want to quote is seekable (the same line as the previous
+     location), just seek it. If it was before, we lost track of it, so
+     return to the start of file.  */
+  if (caret_info.line <= loc.start.line)
+    fseek (caret_info.source, caret_info.offset, SEEK_SET);
+  else
+    {
+      caret_info.line = 1;
+      caret_info.offset = 0;
+      fseek (caret_info.source, caret_info.offset, SEEK_SET);
+    }
+
+  /* Advance to the line's position, keeping track of the offset.  */
+  {
+    int i;
+    for (i = caret_info.line; i < loc.start.line; caret_info.offset++)
+      if (fgetc (caret_info.source) == '\n')
+        ++i;
+  }
+  caret_info.line = loc.start.line;
+
+  /* Read the actual line.  Don't update the offset, so that we keep a pointer
+     to the start of the line.  */
+  {
+    ssize_t len = 0;
+    char *buf = NULL;
+    if ((len = getline (&buf, (size_t*) &len, caret_info.source)) != -1)
+      {
+        /* The caret of a multiline location ends with the first line.  */
+        int end = loc.start.line != loc.end.line ? len : loc.end.column;
+
+        if (len)
+          {
+            int i = loc.start.column;
+            /* Quote the file, indent by a single column.  */
+            fputc (' ', out);
+            fwrite (buf, 1, len, out);
+
+            /* Print the caret, with the same indent as above.  */
+            fputc (' ', out);
+            fprintf (out, "%*s", loc.start.column - 1, "");
+            do {
+              fputc ('^', out);
+            } while (++i < end);
+          }
+        fputc ('\n', out);
+        free (buf);
+      }
+  }
+}
+
 void
 boundary_set_from_string (boundary *bound, char *loc_str)
 {
 void
 boundary_set_from_string (boundary *bound, char *loc_str)
 {
index 5ebb92e30e5272aa8e4e6fe3f950d55acdea35f6..c1859aeb57f9b325a3b53535bbc57f46c8042b97 100644 (file)
@@ -102,6 +102,13 @@ void location_compute (location *loc,
    characters.  */
 unsigned location_print (FILE *out, location loc);
 
    characters.  */
 unsigned location_print (FILE *out, location loc);
 
+/* Free any allocated ressources and close any open file handles that are
+   left-over by the usage of location_caret.  */
+void cleanup_caret (void);
+
+/* Output to OUT the line and caret corresponding to location LOC.  */
+void location_caret (FILE *out, location loc);
+
 /* Return -1, 0, 1, depending whether a is before, equal, or
    after b.  */
 static inline int
 /* Return -1, 0, 1, depending whether a is before, equal, or
    after b.  */
 static inline int
index 0396b0f1be425f59b08c7e62dd3b9c9b5b5e410b..184d789dd7a7cf2a640017f569dd8c48392062c2 100644 (file)
@@ -216,5 +216,7 @@ main (int argc, char *argv[])
   timevar_stop (TV_TOTAL);
   timevar_print (stderr);
 
   timevar_stop (TV_TOTAL);
   timevar_print (stderr);
 
+  cleanup_caret ();
+
   return complaint_issued ? EXIT_FAILURE : EXIT_SUCCESS;
 }
   return complaint_issued ? EXIT_FAILURE : EXIT_SUCCESS;
 }