fix endless-loop-at-shutdown bug now that abstract UI request buffers are used
[ardour.git] / libs / pbd / transmitter.cc
1 /*
2     Copyright (C) 1998-99 Paul Barton-Davis
3
4     This program is free software; you can redistribute it and/or modify
5     it under the terms of the GNU General Public License as published by
6     the Free Software Foundation; either version 2 of the License, or
7     (at your option) any later version.
8
9     This program is distributed in the hope that it will be useful,
10     but WITHOUT ANY WARRANTY; without even the implied warranty of
11     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12     GNU General Public License for more details.
13
14     You should have received a copy of the GNU General Public License
15     along with this program; if not, write to the Free Software
16     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
17
18     $Id$
19 */
20
21 #include <cstdlib>
22 #include <signal.h>
23 #include <string>
24
25 #include "pbd/transmitter.h"
26 #include "pbd/error.h"
27
28 using std::string;
29 using std::ios;
30
31 Transmitter::Transmitter (Channel c)
32 {
33         channel = c;
34         switch (c) {
35         case Error:
36                 send = &error;
37                 break;
38         case Warning:
39                 send = &warning;
40                 break;
41         case Info:
42                 send = &info;
43                 break;
44         case Fatal:
45                 send = &fatal;
46                 break;
47         case Throw:
48                 /* we should never call Transmitter::deliver
49                    for thrown messages (because its overridden in the
50                    class heirarchy). force a segv if we do.
51                 */
52                 send = 0;
53                 break;
54         }
55 }
56
57 void
58 Transmitter::deliver ()
59
60 {
61         string foo;
62
63         /* NOTE: this is just a default action for a Transmitter or a
64            derived class. Any class can override this to produce some
65            other action when deliver() is called.
66         */
67
68         *this << '\0';
69
70         /* send the SigC++ signal */
71
72         foo = str();
73         (*send) (channel, foo.c_str());
74
75         /* XXX when or how can we delete this ? */
76         // delete foo;
77
78         /* return to a pristine state */
79
80         clear ();
81         seekp (0, ios::beg);
82         seekg (0, ios::beg);
83
84         /* do the right thing if this should not return */
85
86         if (does_not_return()) {
87 #ifndef PLATFORM_WINDOWS
88 // TODO !!!! Commented out temporarily (for Windows)
89                 sigset_t mask;
90
91                 sigemptyset (&mask);
92                 sigsuspend (&mask);
93                 /*NOTREACHED*/
94                 exit (1);
95 /* JE - From what I can tell, the above code suspends
96  * program execution until (any) signal occurs. Not
97  * sure at the moment what this achieves, unless it
98  * provides some time for the user to see the message.
99  */
100 #endif
101         }
102 }
103
104 bool
105 Transmitter::does_not_return ()
106
107 {
108         if (channel == Fatal || channel == Throw) {
109                 return true;
110         } else {
111                 return false;
112         }
113 }
114
115
116 extern "C" {
117   void pbd_c_error (const char *str)
118
119   {
120         PBD::error << str << endmsg;
121   }
122 }