2003-07-15 18:19:29 +00:00
|
|
|
// The contents of this file are subject to the BOINC Public License
|
2003-06-11 22:42:49 +00:00
|
|
|
// Version 1.0 (the "License"); you may not use this file except in
|
|
|
|
// compliance with the License. You may obtain a copy of the License at
|
2003-07-15 18:19:29 +00:00
|
|
|
// http://boinc.berkeley.edu/license_1.0.txt
|
2004-07-13 13:54:09 +00:00
|
|
|
//
|
2003-06-11 22:42:49 +00:00
|
|
|
// Software distributed under the License is distributed on an "AS IS"
|
|
|
|
// basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See the
|
|
|
|
// License for the specific language governing rights and limitations
|
2004-07-13 13:54:09 +00:00
|
|
|
// under the License.
|
|
|
|
//
|
|
|
|
// The Original Code is the Berkeley Open Infrastructure for Network Computing.
|
|
|
|
//
|
2003-06-11 22:42:49 +00:00
|
|
|
// The Initial Developer of the Original Code is the SETI@home project.
|
2003-07-02 20:57:59 +00:00
|
|
|
// Portions created by the SETI@home project are Copyright (C) 2002
|
2004-07-13 13:54:09 +00:00
|
|
|
// University of California at Berkeley. All Rights Reserved.
|
|
|
|
//
|
2003-06-11 22:42:49 +00:00
|
|
|
// Contributor(s):
|
|
|
|
//
|
|
|
|
|
2004-03-04 11:41:43 +00:00
|
|
|
#ifdef _WIN32
|
2004-06-16 23:29:48 +00:00
|
|
|
#include "boinc_win.h"
|
2004-03-04 11:41:43 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef _WIN32
|
2004-07-13 13:54:09 +00:00
|
|
|
#include <cassert>
|
|
|
|
#include <cstring>
|
2003-06-11 22:42:49 +00:00
|
|
|
#include <string>
|
2004-03-04 11:41:43 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "util.h"
|
2004-04-08 08:15:23 +00:00
|
|
|
#include "msg_log.h"
|
2003-06-11 22:42:49 +00:00
|
|
|
|
2004-04-30 23:18:56 +00:00
|
|
|
#ifdef _USING_FCGI_
|
|
|
|
#include "fcgi_stdio.h"
|
|
|
|
#endif
|
|
|
|
|
2004-06-30 22:16:26 +00:00
|
|
|
using std::string;
|
|
|
|
|
2004-04-08 08:15:23 +00:00
|
|
|
// MSG_LOG is a base class for writing messages not intended for the end user.
|
|
|
|
// This includes all server messages and client debugging messages.
|
|
|
|
// SCHED_MSG_LOG (in sched/sched_msg_log.C) decides which scheduler messages
|
|
|
|
// to print and formats the "kind" keyword;
|
|
|
|
// CLIENT_MSG_LOG does the same thing for client debugging output.
|
2003-07-02 20:57:59 +00:00
|
|
|
//
|
2004-04-08 08:15:23 +00:00
|
|
|
// MSG_LOG has an "indent_level" state for how many spaces to indent output.
|
|
|
|
// This corresponds in general to the function-call recursion level.
|
|
|
|
// Call MSG_LOG::enter_level() to increase by 1 level
|
|
|
|
// and leave_level() to decrease by 1 level.
|
|
|
|
// The SCOPE_MSG_LOG class takes care of calling leave_level() for you.
|
|
|
|
// Create a SCOPE_MSG_LOG object on the stack at the beginning of a function;
|
|
|
|
// it will increment the level by 1 on construction,
|
|
|
|
// and decrement the level by 1 on destruction at end of scope.
|
|
|
|
// This way you don't have to worry about decrementing
|
|
|
|
// before mid-function returns, exceptions, etc.
|
2003-07-02 20:57:59 +00:00
|
|
|
|
|
|
|
// Each [v]printf* function prints the timestamp, the formatted KIND string,
|
2004-04-08 08:15:23 +00:00
|
|
|
// indentation level, then the specified string.
|
|
|
|
// The string to print can be a one-line string (including the trailing \n),
|
|
|
|
// a multi-line string (it's broken up into lines
|
|
|
|
// to get the prefix on each line), or a file (also broken up into lines).
|
2003-07-02 20:57:59 +00:00
|
|
|
|
2004-04-08 08:15:23 +00:00
|
|
|
// Scheduler functions should use "sched_messages" which is an instance of
|
|
|
|
// SCHED_MSG_LOG. Client functions should use "client_messages",
|
|
|
|
// which is an instance of CLIENT_MSG_LOG.
|
2003-07-02 20:57:59 +00:00
|
|
|
|
2004-04-08 08:15:23 +00:00
|
|
|
// See sched/sched_msg_log.C and client/client_msg_log.C for those classes.
|
2003-07-02 20:57:59 +00:00
|
|
|
|
2004-04-08 08:15:23 +00:00
|
|
|
MSG_LOG::MSG_LOG(FILE* output_) {
|
2003-06-11 22:42:49 +00:00
|
|
|
output = output_;
|
|
|
|
indent_level = 0;
|
|
|
|
spaces[0] = 0;
|
|
|
|
strcpy(spaces+1, " ");
|
|
|
|
}
|
|
|
|
|
2004-04-08 08:15:23 +00:00
|
|
|
void MSG_LOG::enter_level(int diff) {
|
2003-06-11 22:42:49 +00:00
|
|
|
assert (indent_level >= 0);
|
|
|
|
spaces[indent_level] = ' ';
|
|
|
|
indent_level += diff*2;
|
|
|
|
spaces[indent_level] = 0;
|
|
|
|
assert (indent_level >= 0);
|
|
|
|
}
|
|
|
|
|
2004-04-08 08:15:23 +00:00
|
|
|
void MSG_LOG::vprintf(int kind, const char* format, va_list va) {
|
2004-04-01 23:19:13 +00:00
|
|
|
const char* now_timestamp = time_to_string(time(0));
|
2003-06-11 22:42:49 +00:00
|
|
|
if (!v_message_wanted(kind)) return;
|
2004-04-01 23:19:13 +00:00
|
|
|
fprintf(output, "%s [%s]%s ", now_timestamp, v_format_kind(kind), spaces);
|
2003-06-11 22:42:49 +00:00
|
|
|
vfprintf(output, format, va);
|
|
|
|
}
|
|
|
|
|
|
|
|
// break a multi-line string into lines (so that we show prefix on each line)
|
2004-04-08 08:15:23 +00:00
|
|
|
void MSG_LOG::vprintf_multiline(
|
2004-04-01 23:19:13 +00:00
|
|
|
int kind, const char* str, const char* prefix_format, va_list va
|
|
|
|
) {
|
2003-06-11 22:42:49 +00:00
|
|
|
if (!v_message_wanted(kind)) return;
|
|
|
|
if (str == NULL) return;
|
|
|
|
|
|
|
|
char sprefix[256] = "";
|
|
|
|
if (prefix_format) {
|
|
|
|
vsprintf(sprefix, prefix_format, va);
|
|
|
|
}
|
2004-04-01 23:19:13 +00:00
|
|
|
const char* now_timestamp = time_to_string(time(0));
|
2003-06-11 22:42:49 +00:00
|
|
|
const char* skind = v_format_kind(kind);
|
|
|
|
|
|
|
|
string line;
|
|
|
|
while (*str) {
|
|
|
|
if (*str == '\n') {
|
|
|
|
fprintf(output, "%s [%s]%s %s%s\n", now_timestamp, skind, spaces, sprefix, line.c_str());
|
|
|
|
line.erase();
|
|
|
|
} else {
|
|
|
|
line += *str;
|
|
|
|
}
|
|
|
|
++str;
|
|
|
|
}
|
|
|
|
if (!line.empty()) {
|
|
|
|
fprintf(output, "%s %s[%s] %s%s\n", now_timestamp, spaces, skind, sprefix, line.c_str());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-04-08 08:15:23 +00:00
|
|
|
void MSG_LOG::vprintf_file(
|
2004-04-01 23:19:13 +00:00
|
|
|
int kind, const char* filename, const char* prefix_format, va_list va
|
|
|
|
) {
|
2003-06-11 22:42:49 +00:00
|
|
|
if (!v_message_wanted(kind)) return;
|
|
|
|
|
|
|
|
char sprefix[256] = "";
|
|
|
|
if (prefix_format) {
|
|
|
|
vsprintf(sprefix, prefix_format, va);
|
|
|
|
}
|
2004-04-01 23:19:13 +00:00
|
|
|
const char* now_timestamp = time_to_string(time(0));
|
2003-06-11 22:42:49 +00:00
|
|
|
const char* skind = v_format_kind(kind);
|
|
|
|
|
2004-04-30 23:18:56 +00:00
|
|
|
FILE* f = fopen(filename, "r");
|
2003-06-11 22:42:49 +00:00
|
|
|
if (!f) return;
|
2004-04-30 23:18:56 +00:00
|
|
|
char buf[256];
|
2003-06-11 22:42:49 +00:00
|
|
|
|
2004-04-30 23:18:56 +00:00
|
|
|
while (fgets(buf, 256, f)) {
|
|
|
|
fprintf(output, "%s [%s]%s %s%s\n", now_timestamp, skind, spaces, sprefix, buf);
|
2003-06-11 22:42:49 +00:00
|
|
|
}
|
2004-04-30 23:18:56 +00:00
|
|
|
fclose(f);
|
2003-06-11 22:42:49 +00:00
|
|
|
}
|
|
|
|
|
2004-04-08 08:15:23 +00:00
|
|
|
void MSG_LOG::printf(int kind, const char* format, ...) {
|
2003-06-11 22:42:49 +00:00
|
|
|
va_list va;
|
|
|
|
va_start(va, format);
|
|
|
|
vprintf(kind, format, va);
|
|
|
|
va_end(va);
|
|
|
|
}
|
|
|
|
|
2004-04-08 08:15:23 +00:00
|
|
|
void MSG_LOG::printf_multiline(
|
2004-04-01 23:19:13 +00:00
|
|
|
int kind, const char* str, const char* prefix_format, ...
|
|
|
|
) {
|
2003-06-11 22:42:49 +00:00
|
|
|
va_list va;
|
|
|
|
va_start(va, prefix_format);
|
|
|
|
vprintf_multiline(kind, str, prefix_format, va);
|
|
|
|
va_end(va);
|
|
|
|
}
|
|
|
|
|
2004-04-08 08:15:23 +00:00
|
|
|
void MSG_LOG::printf_file(
|
2004-04-01 23:19:13 +00:00
|
|
|
int kind, const char* filename, const char* prefix_format, ...
|
|
|
|
) {
|
2003-06-11 22:42:49 +00:00
|
|
|
va_list va;
|
|
|
|
va_start(va, prefix_format);
|
|
|
|
vprintf_file(kind, filename, prefix_format, va);
|
|
|
|
va_end(va);
|
|
|
|
}
|
|
|
|
|
2004-04-08 08:15:23 +00:00
|
|
|
// These SCOPE_MSG_LOG functions are utility functions that call their
|
|
|
|
// corresponding MSG_LOG functions with the same name, passing the KIND that
|
|
|
|
// was specified on creation of the SCOPE_MSG_LOG object.
|
2003-07-02 20:57:59 +00:00
|
|
|
|
2004-04-08 08:15:23 +00:00
|
|
|
void SCOPE_MSG_LOG::printf(const char* format, ...) {
|
2003-06-11 22:42:49 +00:00
|
|
|
va_list va;
|
|
|
|
va_start(va, format);
|
|
|
|
messages.vprintf(kind, format, va);
|
|
|
|
va_end(va);
|
|
|
|
}
|
|
|
|
|
2004-04-08 08:15:23 +00:00
|
|
|
void SCOPE_MSG_LOG::printf_multiline(
|
2004-04-01 23:19:13 +00:00
|
|
|
const char* str, const char* prefix_format, ...
|
|
|
|
) {
|
2003-06-11 22:42:49 +00:00
|
|
|
va_list va;
|
|
|
|
va_start(va, prefix_format);
|
|
|
|
messages.vprintf_multiline(kind, str, prefix_format, va);
|
|
|
|
va_end(va);
|
|
|
|
}
|
|
|
|
|
2004-04-08 08:15:23 +00:00
|
|
|
void SCOPE_MSG_LOG::printf_file(
|
2004-04-01 23:19:13 +00:00
|
|
|
const char* filename, const char* prefix_format, ...
|
|
|
|
) {
|
2003-06-11 22:42:49 +00:00
|
|
|
va_list va;
|
|
|
|
va_start(va, prefix_format);
|
|
|
|
messages.vprintf_file(kind, filename, prefix_format, va);
|
|
|
|
va_end(va);
|
|
|
|
}
|
2004-12-08 00:40:19 +00:00
|
|
|
|
|
|
|
#ifdef __GNUC__
|
|
|
|
static volatile const char __attribute__((unused)) *BOINCrcsid="$Id$";
|
|
|
|
#else
|
|
|
|
static volatile const char *BOINCrcsid="$Id$";
|
|
|
|
#endif
|