emailrelay/src/glib/glogoutput.cpp
Graeme Walker ead43a7739 v0.9.1
2001-09-08 12:00:00 +00:00

179 lines
4.0 KiB
C++

//
// Copyright (C) 2001 Graeme Walker <graeme_walker@users.sourceforge.net>
//
// This program is free software; you can redistribute it and/or
// modify it under the terms of the GNU General Public License
// as published by the Free Software Foundation; either
// version 2 of the License, or (at your option) any later
// version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program; if not, write to the Free Software
// Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
//
// ===
//
// glogoutput.cpp
//
#include "gdef.h"
#include "glogoutput.h"
#include <cstdlib>
#include <cstring>
G::LogOutput *G::LogOutput::m_this = NULL ;
G::LogOutput::LogOutput( bool enabled , bool verbose ) :
m_enabled(enabled) ,
m_verbose(verbose) ,
m_syslog(false)
{
if( m_this == NULL )
m_this = this ;
}
G::LogOutput::~LogOutput()
{
if( m_this == this )
m_this = NULL ;
}
bool G::LogOutput::enable( bool enabled )
{
bool was_enabled = m_enabled ;
m_enabled = enabled ;
return was_enabled ;
}
//static
void G::LogOutput::itoa( char *out , unsigned int n )
{
n = n % 1000000U ;
if( n == 0U )
{
out[0] = '0' ;
out[1] = '\0' ;
}
else
{
char buffer[15] ;
char *p = buffer + sizeof(buffer) - 1 ;
*p-- = '\0' ;
for( ; n > 0U ; --p )
{
*p = '0' + (n % 10U) ;
n /= 10U ;
}
::strcpy( out , p+1 ) ;
}
}
//static
void G::LogOutput::output( G::Log::Severity severity , const char *text )
{
if( m_this != NULL )
{
if( severity != G::Log::s_Debug || m_this->m_verbose )
{
m_this->rawOutput( severity , text ? text : "" ) ;
if( text && text[0] && text[std::strlen(text)-1] != '\n' )
m_this->rawOutput( severity , "\n" ) ;
}
}
}
//static
void G::LogOutput::output( G::Log::Severity severity , const char *file, unsigned line, const char *text )
{
file = file ? file : "" ;
text = text ? text : "" ;
// no-op if disabled
if( m_this == NULL || !m_this->m_enabled )
return ;
char buffer[500] ;
buffer[0] = '\0' ;
if( severity == G::Log::s_Debug )
fileAndLine( buffer , sizeof(buffer) , file , line ) ;
std::strncat( buffer + std::strlen(buffer) , text , sizeof(buffer) - 1 - std::strlen(buffer) ) ;
output( severity , buffer ) ;
}
G::LogOutput *G::LogOutput::instance()
{
return m_this ;
}
void G::LogOutput::onAssert()
{
// no-op
}
//static
void G::LogOutput::fileAndLine( char *buffer , size_t size , const char *file , int line )
{
const char *forward = ::strrchr( file , '/' ) ;
const char *back = ::strrchr( file , '\\' ) ;
const char *last = forward > back ? forward : back ;
const char *basename = last ? (last+1) : file ;
std::strncat( buffer+std::strlen(buffer) , basename , size-std::strlen(buffer)-1 ) ;
std::strncat( buffer+std::strlen(buffer) , "(" , size-std::strlen(buffer)-1 ) ;
char b[15] ;
itoa( b , line ) ;
std::strncat( buffer+std::strlen(buffer) , b , size-std::strlen(buffer)-1 ) ;
std::strncat( buffer+std::strlen(buffer) , "): " , size-std::strlen(buffer)-1 ) ;
}
void G::LogOutput::assertion( const char *file , unsigned line , bool test , const char *test_string )
{
if( !test )
{
char buffer[100] ;
std::strcpy( buffer , "Assertion error: " ) ;
size_t size = sizeof(buffer) - 10 ; // -10 for luck
if( file )
{
fileAndLine( buffer , size , file , line ) ;
}
if( test_string )
{
std::strncat( buffer+std::strlen(buffer) , test_string , size-std::strlen(buffer)-1);
}
if( instance() )
{
// forward to derived classes -- these
// overrides may safely re-enter this method --
// all code in this class is re-entrant
//
instance()->onAssert() ;
}
output( G::Log::s_Assertion , buffer ) ;
halt() ;
}
}
//static
void G::LogOutput::halt()
{
abort() ;
}
void G::LogOutput::syslog( SyslogFacility facility )
{
m_syslog = true ;
m_facility = facility ;
}