/*******************************************************************************/ /* Copyright (C) 2008 Jonathan Moore Liles */ /* */ /* 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; see the file COPYING. If not,write to the Free Software */ /* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ /*******************************************************************************/ /* Master class for journaling. */ #pragma once #include <stdio.h> #include <stdlib.h> #include <string.h> #include <assert.h> #include <map> #include <string> #include <queue> // #include "types.h" typedef void (progress_func)( int, void * ); typedef void (snapshot_func)( void * ); typedef void (dirty_func)( int, void * ); class Log_Entry; class Loggable; typedef Loggable *(create_func)(Log_Entry &, unsigned int id); #define LOG_REGISTER_CREATE( class ) \ Loggable::register_create( #class, & class ::create ) #define LOG_NAME_FUNC( class ) \ virtual const char *class_name ( void ) const { return #class ; } #define LOG_CREATE_FUNC( class ) \ static Loggable * \ create ( Log_Entry &e, unsigned int id ) \ { \ class *r = new class; \ r->update_id( id ); \ r->set( e ); \ return (Loggable *)r; \ } \ LOG_NAME_FUNC( class ) #define LOG_NOT_LOGGABLE_FUNC( class ) \ virtual const char *class_name ( void ) const { return #class ; } class Logger; class Loggable { struct log_pair { Loggable * loggable; Log_Entry * unjournaled_state; }; static bool _readonly; static FILE *_fp; static unsigned int _log_id; static int _level; static off_t _undo_offset; static std::map <unsigned int, Loggable::log_pair > _loggables; static std::map <std::string, create_func*> _class_map; static std::queue <char *> _transaction; static progress_func *_progress_callback; static void *_progress_callback_arg; static snapshot_func *_snapshot_callback; static void *_snapshot_callback_arg; static dirty_func *_dirty_callback; static void *_dirty_callback_arg; private: static unsigned int _relative_id; #ifndef NDEBUG static bool _snapshotting; static int _snapshot_count; mutable int _num_log_creates; mutable int _num_snapshot; mutable int _num_snapshot_creates; #endif unsigned int _id; Log_Entry *_old_state; int _nest; static int _dirty; /* count of changes */ static void ensure_size ( size_t n ); void log_print ( const Log_Entry *o, const Log_Entry *n ) const; static void log ( const char *fmt, ... ); static void flush ( void ); void init ( bool loggable=true ) { // _new_state #ifndef NDEBUG _num_log_creates = 0; _num_snapshot = 0; _num_snapshot_creates = 0; #endif _old_state = NULL; _nest = 0; if ( loggable ) { _id = ++_log_id; _loggables[ _id ].loggable = this; } else _id = 0; } /* not implemented */ const Loggable & operator= ( const Loggable &rhs ); void record_unjournaled ( void ) const; static bool load_unjournaled_state ( void ); static bool replay ( FILE *fp ); static void signal_dirty ( int v ) { if ( _dirty_callback ) _dirty_callback( v, _dirty_callback_arg ); } static void set_dirty ( void ) { signal_dirty( ++_dirty ); } static void clear_dirty ( void ) { signal_dirty( _dirty = 0 ); } public: static bool readonly ( void ) { return _readonly; } static bool replay ( const char *name ); static bool snapshot( FILE * fp ); static bool snapshot( const char *name ); static void snapshot_callback ( snapshot_func *p, void *arg ) { _snapshot_callback = p; _snapshot_callback_arg = arg; } static void progress_callback ( progress_func *p, void *arg ) { _progress_callback = p; _progress_callback_arg = arg;} static void dirty_callback ( dirty_func *p, void *arg ) { _dirty_callback = p; _dirty_callback_arg = arg;} static const char *escape ( const char *s ); unsigned int id ( void ) const { return _id; } static bool save_unjournaled_state ( void ); static bool open ( const char *filename ); static bool close ( void ); static void undo ( void ); static void compact ( void ); static void block_start ( void ); static void block_end ( void ); static Loggable * find ( unsigned int id ); Loggable ( bool loggable=true ) { init( loggable ); } void update_id ( unsigned int id ); virtual ~Loggable ( ); static void register_create ( const char *name, create_func *func ) { _class_map[ std::string( name ) ] = func; } /* log messages for journal */ virtual void get ( Log_Entry &e ) const = 0; virtual void get_unjournaled ( Log_Entry & ) const { /* implementation optional */ } virtual void set ( Log_Entry &e ) = 0; virtual const char *class_name ( void ) const = 0; virtual void log_children ( void ) const { return; } static void begin_relative_id_mode ( void ); static void end_relative_id_mode ( void ); static bool do_this ( const char *s, bool reverse ); static int dirty ( void ) { return _dirty; } void log_create ( void ) const; protected: void log_start ( void ); void log_end ( void ); void log_destroy ( void ) const; /* leaf subclasses *must* call log_create() at the end of their copy contructors */ Loggable ( const Loggable & ) { init( true ); } public: friend class Logger; }; class Logger { Loggable *_this; Logger ( ) {} /* not permitted */ Logger ( const Logger &rhs ); const Logger & operator= ( const Logger &rhs ); public: Logger ( Loggable *l ) : _this( l ) { _this->log_start(); } ~Logger ( ) { _this->log_end(); } void hold ( void ) { _this->_nest++; } void release ( void ) { _this->_nest--; assert( _this->_nest ); } }; #include "Log_Entry.H"