/* AppState - application initialisation and behaviour Copyright (C) Lumiera.org 2008, Hermann Vosseler 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. * *****************************************************/ #include "include/lifecycle.h" #include "lumiera/appstate.hpp" #include "lib/lifecycleregistry.hpp" #include "include/error.hpp" #include "common/util.hpp" using util::isnil; using util::cStr; namespace lumiera { Symbol ON_BASIC_INIT ("ON_BASIC_INIT"); Symbol ON_GLOBAL_INIT ("ON_GLOBAL_INIT"); Symbol ON_GLOBAL_SHUTDOWN ("ON_GLOBAL_SHUTDOWN"); /** perform initialisation triggered on first access. * Will execute the ON_BASIC_INIT hook, but under typical * circumstances this is a NOP, because when callbacks are * added to this hook, the AppState singleton instance has * already been created. For this reason, there is special * treatment for the ON_BASIC_INIT in LifecycleHook::add, * causing the provided callbacks to be fired immediately. * (btw, this is nothing to be worried of, because from * client codes POV it just behaves like intended). */ AppState::AppState() : lifecycleHooks_(new LifecycleRegistry) { lifecycleHooks_->execute (ON_BASIC_INIT); // note in most cases a NOP } AppState::~AppState() { } AppState& AppState::instance() // Meyer's singleton { static scoped_ptr theApp_ (0); if (!theApp_) theApp_.reset (new AppState ()); return *theApp_; } void AppState::lifecycle (Symbol event_label) { instance().lifecycleHooks_->execute(event_label); } void AppState::evaluate (lumiera::Option& options) { UNIMPLEMENTED ("evaluate the options and set global application state"); } void AppState::maybeStart (lumiera::Subsys&) { UNIMPLEMENTED ("maybe register and activate the given subsystem"); } typedef AppState::ExitCode ExitCode; ExitCode AppState::maybeWait() { UNIMPLEMENTED ("wait on termination and normal shutdown"); } ExitCode AppState::abort (lumiera::Error& problem) { UNIMPLEMENTED ("clean error shutdown"); } ExitCode AppState::abort () { UNIMPLEMENTED ("emergency shutdown"); } // ==== implementation LifecycleHook class ======= typedef LifecycleRegistry::Hook Callback; LifecycleHook::LifecycleHook (Symbol eventLabel, Callback callbackFun) { add (eventLabel,callbackFun); } void LifecycleHook::add (Symbol eventLabel, Callback callbackFun) { bool isNew = AppState::instance().lifecycleHooks_->enroll (eventLabel,callbackFun); if (isNew && !strcmp(ON_BASIC_INIT, eventLabel)) callbackFun(); // when this code executes, // then per definition we are already post "basic init" // (which happens in the AppState ctor); thus fire it immediately } void trigger (Symbol eventLabel) { AppState::lifecycle (eventLabel); } } // namespace lumiera extern "C" { /* ==== implementation C interface for lifecycle hooks ======= */ extern const char * lumiera_ON_BASIC_INIT = lumiera::ON_BASIC_INIT; extern const char * lumiera_ON_GLOBAL_INIT = lumiera::ON_GLOBAL_INIT; extern const char * lumiera_ON_GLOBAL_SHUTDOWN = lumiera::ON_GLOBAL_SHUTDOWN; void lumiera_LifecycleHook_add (const char* eventLabel, void callbackFun(void)) { lumiera::LifecycleHook (eventLabel, callbackFun); } void lumiera_Lifecycle_trigger (const char* eventLabel) { lumiera::AppState::lifecycle (eventLabel); } }