]> git.tdb.fi Git - libs/core.git/blobdiff - source/core/application.cpp
Nicer formatting of exceptions with multi-line whats
[libs/core.git] / source / core / application.cpp
index 53fb1c3bc6ac6a49582d8aad1ad692414a8e963f..290f20b05239702c40746f9048aba993aa877b21 100644 (file)
@@ -1,17 +1,13 @@
-/* $Id$
-
-This file is part of libmspcore
-Copyright © 2006-2008, 2011  Mikko Rasa, Mikkosoft Productions
-Distributed under the LGPL
-*/
-
-#ifdef WIN32
-#include <windows.h>
-#endif
-#include <signal.h>
-#include <iostream>
+#include <cstring>
 #include <typeinfo>
-#include "../debug/demangle.h"
+#include <signal.h>
+#include <msp/debug/demangle.h>
+#include <msp/debug/errorreporter.h>
+#include <msp/fs/dir.h>
+#include <msp/fs/path.h>
+#include <msp/fs/utils.h>
+#include <msp/io/print.h>
+#include <msp/strings/utils.h>
 #include "application.h"
 #include "getopt.h"
 
@@ -21,37 +17,31 @@ namespace Msp {
 
 Application *Application::app_ = 0;
 Application::Starter *Application::starter_ = 0;
+const char *Application::argv0_ = 0;
+string Application::name_;
 void *Application::data_ = 0;
 
-Application::Application():
+Application::Application(const string &n):
        exit_code(0)
-{ }
+{
+       if(app_)
+               throw logic_error("instance already exists");
 
-/**
-Constructs an instance of the registered application class and runs it.  If the
-application throws a usage_error, a help message is printed.  The GetOpt class
-will throw such exceptions automatically in error conditions.
+       if(!n.empty())
+               name_ = n;
+       else
+               name_ = FS::basename(argv0_);
+}
 
-This function can only be called once.  The global main() function provided by
-the library normally does it automatically at program startup.
-*/
-int Application::run(int argc, char **argv, void *data)
+int Application::run(int argc, char **argv, void *data, void (*created_callback)(void *))
 {
-       static bool called = false;
-       if(called)
-       {
-               cerr<<"Trying to call Application::run_app twice!\n";
-               return 125;
-       }
-       called = true;
-
        if(!starter_)
        {
-               cerr<<"Trying to run with no RegisteredApplication class!\n";
+               IO::cerr.write("Application::run called with no RegisteredApplication class!\n");
                return 126;
        }
 
-       data_ = data;
+       set_startup_info(argv[0], data);
 
        try
        {
@@ -61,11 +51,13 @@ int Application::run(int argc, char **argv, void *data)
                }
                catch(const usage_error &e)
                {
-                       cerr<<e.what()<<'\n';
-                       cerr<<e.help()<<'\n';
+                       IO::print(IO::cerr, "%s\n%s\n", e.what(), e.help());
                        return 1;
                }
 
+               if(created_callback)
+                       created_callback(data);
+
                int result = app_->main();
                Application *a = app_;
                app_ = 0;
@@ -74,25 +66,58 @@ int Application::run(int argc, char **argv, void *data)
        }
        catch(const exception &e)
        {
+               bool handled = false;
+               if(const Debug::ErrorReporter *er = Debug::ErrorReporter::get_current())
+                       handled = er->report_uncaught_exception(e);
+
+               if(!handled)
+               {
+                       IO::print(IO::cerr, "An uncaught exception occurred.\n");
+                       IO::print(IO::cerr, "  type:   %s\n", Debug::demangle(typeid(e).name()));
+                       vector<string> lines = split(e.what(), '\n');
+                       if(lines.size()<2)
+                               IO::print(IO::cerr, "  what(): %s\n", e.what());
+                       else
+                       {
+                               IO::print(IO::cerr, "  what(): %s\n", lines.front());
+                               for(vector<string>::const_iterator i=lines.begin(); ++i!=lines.end(); )
+                                       IO::print(IO::cerr, "          %s\n", *i);
+                       }
+               }
+
                delete app_;
+               app_ = 0;
+
+               return 124;
+       }
+}
+
+void Application::set_startup_info(const char *argv0, void *data)
+{
+       if(argv0_)
+               throw logic_error("startup info already set");
 
-#ifdef WIN32
-               string msg = Debug::demangle(typeid(e).name())+":\n"+e.what();
-               MessageBoxA(0, msg.c_str(), "Uncaught exception", MB_OK|MB_ICONERROR);
+       static FS::Path exe;
+
+       if(!argv0 || !*argv0)
+       {
+#ifdef _WIN32
+               argv0 = "application.exe";
 #else
-               cerr<<"An uncaught exception occurred.\n";
-               cerr<<"  type:   "<<Debug::demangle(typeid(e).name())<<'\n';
-               cerr<<"  what(): "<<e.what()<<'\n';
+               argv0 = "./application";
 #endif
-
-               return 124;
        }
+
+       bool has_slash = strchr(argv0, FS::DIRSEP);
+       if(!has_slash)
+               exe = FS::path_lookup(argv0);
+       if(exe.empty())
+               exe = FS::realpath(argv0);
+
+       argv0_ = exe.c_str();
+       data_ = data;
 }
 
-/**
-Default main loop.  Calls tick() repeatedly until exit() is called.  A custom
-main loop should monitor the done member variable and return exit_code.
-*/
 int Application::main()
 {
        done = false;
@@ -102,26 +127,17 @@ int Application::main()
        return exit_code;
 }
 
-/**
-Sets the specified signal to be delivered to the sighandler member function.
-*/
 void Application::catch_signal(int s)
 {
        signal(s, &sighandler_);
 }
 
-/**
-Causes the application to exit gracefully with the given exit code.
-*/
 void Application::exit(int c)
 {
        done = true;
        exit_code = c;
 }
 
-/**
-Static wrapper function to call a member function of the Application instance.
-*/
 void Application::sighandler_(int s)
 {
        app_->sighandler(s);