1 // Stream based logging mechanism.
3 // Written by Bernie Bright, 1998
5 // Copyright (C) 1998 Bernie Bright - bbright@c031.aone.net.au
7 // This program is free software; you can redistribute it and/or
8 // modify it under the terms of the GNU General Public License as
9 // published by the Free Software Foundation; either version 2 of the
10 // License, or (at your option) any later version.
12 // This program is distributed in the hope that it will be useful, but
13 // WITHOUT ANY WARRANTY; without even the implied warranty of
14 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 // General Public License for more details.
17 // You should have received a copy of the GNU General Public License
18 // along with this program; if not, write to the Free Software
19 // Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
22 // (Log is kept at end of this file)
32 #include <Include/compiler.h>
34 #ifdef FG_HAVE_STD_INCLUDES
38 # include <iostream.h>
39 # include "Include/fg_traits.hxx"
42 #include "debug_types.h"
44 #ifndef FG_HAVE_NATIVE_SGI_COMPILERS
45 FG_USING_STD(streambuf);
46 FG_USING_STD(ostream);
54 // 1. Change output destination. Done.
55 // 2. Make logbuf thread safe.
56 // 3. Read environment for default debugClass and debugPriority.
59 //-----------------------------------------------------------------------------
61 // logbuf is an output-only streambuf with the ability to disable sets of
62 // messages at runtime. Only messages with priority >= logbuf::logPriority
63 // and debugClass == logbuf::logClass are output.
65 class logbuf : public streambuf
69 #ifndef FG_HAVE_STD_INCLUDES
70 typedef char_traits<char> traits_type;
71 typedef char_traits<char>::int_type int_type;
72 typedef char_traits<char>::pos_type pos_type;
73 typedef char_traits<char>::off_type off_type;
75 // logbuf( streambuf* sb ) : sbuf(sb) {}
79 // Is logging enabled?
80 bool enabled() { return logging_enabled; }
82 // Set the logging level of subsequent messages.
83 void set_log_state( fgDebugClass c, fgDebugPriority p );
85 // Set the global logging level.
86 static void set_log_level( fgDebugClass c, fgDebugPriority p );
89 void set_sb( streambuf* sb );
93 inline virtual int sync();
94 int_type overflow( int ch );
95 // int xsputn( const char* s, istreamsize n );
99 // The streambuf used for actual output. Defaults to cerr.rdbuf().
100 static streambuf* sbuf;
102 static bool logging_enabled;
103 static fgDebugClass logClass;
104 static fgDebugPriority logPriority;
109 logbuf( const logbuf& );
110 void operator= ( const logbuf& );
116 #ifdef FG_HAVE_STD_INCLUDES
117 return sbuf->pubsync();
124 logbuf::set_log_state( fgDebugClass c, fgDebugPriority p )
126 logging_enabled = ((c & logClass) != 0 && p >= logPriority);
129 inline logbuf::int_type
130 logbuf::overflow( int c )
132 return logging_enabled ? sbuf->sputc(c) : (EOF == 0 ? 1: 0);
135 //-----------------------------------------------------------------------------
137 // logstream manipulator for setting the log level of a message.
141 loglevel( fgDebugClass c, fgDebugPriority p )
142 : logClass(c), logPriority(p) {}
144 fgDebugClass logClass;
145 fgDebugPriority logPriority;
148 //-----------------------------------------------------------------------------
150 // A helper class that ensures a streambuf and ostream are constructed and
151 // destroyed in the correct order. The streambuf must be created before the
152 // ostream but bases are constructed before members. Thus, making this class
153 // a private base of logstream, declared to the left of ostream, we ensure the
154 // correct order of construction and destruction.
156 struct logstream_base
158 // logstream_base( streambuf* sb ) : lbuf(sb) {}
164 //-----------------------------------------------------------------------------
168 class logstream : private logstream_base, public ostream
171 // The default is to send messages to cerr.
172 logstream( ostream& out )
173 // : logstream_base(out.rdbuf()),
175 ostream(&lbuf) { lbuf.set_sb(out.rdbuf());}
177 void set_output( ostream& out ) { lbuf.set_sb( out.rdbuf() ); }
179 // Set the global log class and priority level.
180 void setLogLevels( fgDebugClass c, fgDebugPriority p );
182 // Output operator to capture the debug level and priority of a message.
183 inline ostream& operator<< ( const loglevel& l );
187 logstream::operator<< ( const loglevel& l )
189 lbuf.set_log_state( l.logClass, l.logPriority );
193 //-----------------------------------------------------------------------------
195 // Return the one and only logstream instance.
196 // We use a function instead of a global object so we are assured that cerr
197 // has been initialised.
202 static logstream logstrm( cerr );
207 # define FG_LOG(C,P,M)
209 # define FG_LOG(C,P,M) fglog() << loglevel(C,P) << M << endl
212 #endif // _LOGSTREAM_H
215 // Revision 1.4 1999/03/02 01:01:47 curt
216 // Tweaks for compiling with native SGI compilers.
218 // Revision 1.3 1999/01/19 20:53:35 curt
219 // Portability updates by Bernie Bright.
221 // Revision 1.2 1998/11/07 19:07:02 curt
222 // Enable release builds using the --without-logging option to the configure
223 // script. Also a couple log message cleanups, plus some C to C++ comment
226 // Revision 1.1 1998/11/06 21:20:42 curt