//===-- Common.cpp --------------------------------------------------------===// // // The KLEE Symbolic Virtual Machine // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #include "Common.h" #include #include #include #include #include #include using namespace klee; FILE* klee::klee_warning_file = NULL; FILE* klee::klee_message_file = NULL; /* 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) { FILE *f = stderr; if (!onlyToFile) { fprintf(f, "KLEE: "); if (pfx) fprintf(f, "%s: ", pfx); vfprintf(f, msg, ap); fprintf(f, "\n"); fflush(f); } if (pfx == NULL) f = klee_message_file; else f = klee_warning_file; if (f) { fprintf(f, "KLEE: "); if (pfx) fprintf(f, "%s: ", pfx); vfprintf(f, msg, ap); fprintf(f, "\n"); fflush(f); } } 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("ERROR", false, msg, ap); va_end(ap); exit(1); } void klee::klee_warning(const char *msg, ...) { va_list ap; va_start(ap, msg); klee_vmessage("WARNING", 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 > keys; std::pair 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("WARNING", false, msg, ap); va_end(ap); } }