about summary refs log tree commit diff homepage
path: root/lib/Support
diff options
context:
space:
mode:
authorMartin Nowack <martin.nowack@gmail.com>2015-08-10 08:59:00 +0200
committerDan Liew <daniel.liew@imperial.ac.uk>2015-12-17 19:30:14 +0000
commit771cdf39d9c1e142269e2cafc4365d0d68e05f9a (patch)
treee4d0c7a9eb9a5aee2ed350d77625aec89cc05ed5 /lib/Support
parentd8f9c929f69cdfb739f03b6512035071c02108c9 (diff)
downloadklee-771cdf39d9c1e142269e2cafc4365d0d68e05f9a.tar.gz
Refactoring: Moving klee_warning/_error functions to ErrorHandling in Support directory
Diffstat (limited to 'lib/Support')
-rw-r--r--lib/Support/ErrorHandling.cpp166
1 files changed, 166 insertions, 0 deletions
diff --git a/lib/Support/ErrorHandling.cpp b/lib/Support/ErrorHandling.cpp
new file mode 100644
index 00000000..7ca223e5
--- /dev/null
+++ b/lib/Support/ErrorHandling.cpp
@@ -0,0 +1,166 @@
+//===-- ErrorHandling.cpp -------------------------------------------------===//
+//
+//                     The KLEE Symbolic Virtual Machine
+//
+// This file is distributed under the University of Illinois Open Source
+// License. See LICENSE.TXT for details.
+//
+//===----------------------------------------------------------------------===//
+
+#include "klee/Internal/Support/ErrorHandling.h"
+#include "llvm/ADT/StringRef.h"
+#include "llvm/Support/raw_ostream.h"
+
+#include <stdlib.h>
+#include <stdio.h>
+#include <stdarg.h>
+#include <assert.h>
+#include <string.h>
+
+#include <set>
+
+using namespace klee;
+
+FILE *klee::klee_warning_file = NULL;
+FILE *klee::klee_message_file = NULL;
+
+static const char *warningPrefix = "WARNING";
+static const char *warningOncePrefix = "WARNING ONCE";
+static const char *errorPrefix = "ERROR";
+static const char *notePrefix = "NOTE";
+
+static bool shouldSetColor(const char *pfx, const char *msg,
+                           const char *prefixToSearchFor) {
+  if (pfx && strcmp(pfx, prefixToSearchFor) == 0)
+    return true;
+
+  if (llvm::StringRef(msg).startswith(prefixToSearchFor))
+    return true;
+
+  return false;
+}
+
+static void klee_vfmessage(FILE *fp, const char *pfx, const char *msg,
+                           va_list ap) {
+  if (!fp)
+    return;
+
+  llvm::raw_fd_ostream fdos(fileno(fp), /*shouldClose=*/false,
+                            /*unbuffered=*/true);
+  bool modifyConsoleColor = fdos.is_displayed() && (fp == stderr);
+
+  if (modifyConsoleColor) {
+
+    // Warnings
+    if (shouldSetColor(pfx, msg, warningPrefix))
+      fdos.changeColor(llvm::raw_ostream::MAGENTA,
+                       /*bold=*/false,
+                       /*bg=*/false);
+
+    // Once warning
+    if (shouldSetColor(pfx, msg, warningOncePrefix))
+      fdos.changeColor(llvm::raw_ostream::MAGENTA,
+                       /*bold=*/true,
+                       /*bg=*/false);
+
+    // Errors
+    if (shouldSetColor(pfx, msg, errorPrefix))
+      fdos.changeColor(llvm::raw_ostream::RED,
+                       /*bold=*/true,
+                       /*bg=*/false);
+
+    // Notes
+    if (shouldSetColor(pfx, msg, notePrefix))
+      fdos.changeColor(llvm::raw_ostream::WHITE,
+                       /*bold=*/true,
+                       /*bg=*/false);
+  }
+
+  fdos << "KLEE: ";
+  if (pfx)
+    fdos << pfx << ": ";
+
+  // FIXME: Can't use fdos here because we need to print
+  // a variable number of arguments and do substitution
+  vfprintf(fp, msg, ap);
+  fflush(fp);
+
+  fdos << "\n";
+
+  if (modifyConsoleColor)
+    fdos.resetColor();
+
+  fdos.flush();
+}
+
+/* Prints a message/warning.
+
+   If pfx is NULL, this is a regular message, and it's sent to
+   klee_message_file (messages.txt).  Otherwise, it is sent to
+   klee_warning_file (warnings.txt).
+
+   Iff onlyToFile is false, the message is also printed on stderr.
+*/
+static void klee_vmessage(const char *pfx, bool onlyToFile, const char *msg,
+                          va_list ap) {
+  if (!onlyToFile) {
+    va_list ap2;
+    va_copy(ap2, ap);
+    klee_vfmessage(stderr, pfx, msg, ap2);
+    va_end(ap2);
+  }
+
+  klee_vfmessage(pfx ? klee_warning_file : klee_message_file, pfx, msg, ap);
+}
+
+void klee::klee_message(const char *msg, ...) {
+  va_list ap;
+  va_start(ap, msg);
+  klee_vmessage(NULL, false, msg, ap);
+  va_end(ap);
+}
+
+/* Message to be written only to file */
+void klee::klee_message_to_file(const char *msg, ...) {
+  va_list ap;
+  va_start(ap, msg);
+  klee_vmessage(NULL, true, msg, ap);
+  va_end(ap);
+}
+
+void klee::klee_error(const char *msg, ...) {
+  va_list ap;
+  va_start(ap, msg);
+  klee_vmessage(errorPrefix, false, msg, ap);
+  va_end(ap);
+  exit(1);
+}
+
+void klee::klee_warning(const char *msg, ...) {
+  va_list ap;
+  va_start(ap, msg);
+  klee_vmessage(warningPrefix, false, msg, ap);
+  va_end(ap);
+}
+
+/* Prints a warning once per message. */
+void klee::klee_warning_once(const void *id, const char *msg, ...) {
+  static std::set<std::pair<const void *, const char *> > keys;
+  std::pair<const void *, const char *> key;
+
+  /* "calling external" messages contain the actual arguments with
+     which we called the external function, so we need to ignore them
+     when computing the key. */
+  if (strncmp(msg, "calling external", strlen("calling external")) != 0)
+    key = std::make_pair(id, msg);
+  else
+    key = std::make_pair(id, "calling external");
+
+  if (!keys.count(key)) {
+    keys.insert(key);
+    va_list ap;
+    va_start(ap, msg);
+    klee_vmessage(warningOncePrefix, false, msg, ap);
+    va_end(ap);
+  }
+}