]>
git.saurik.com Git - apt.git/blob - apt-pkg/contrib/error.cc
1 // -*- mode: cpp; mode: fold -*-
3 /* ######################################################################
5 Global Error Class - Global error mechanism
7 We use a simple STL vector to store each error record. A PendingFlag
8 is kept which indicates when the vector contains a Sever error.
10 This source is placed in the Public Domain, do with it what you will
11 It was originally written by Jason Gunthorpe.
13 ##################################################################### */
15 // Include Files /*{{{*/
16 #include <apt-pkg/error.h>
29 // Global Error Object /*{{{*/
30 /* If the implementation supports posix threads then the accessor function
31 is compiled to be thread safe otherwise a non-safe version is used. A
32 Per-Thread error object is maintained in much the same manner as libc
34 #if defined(_POSIX_THREADS) && defined(HAVE_PTHREAD)
37 static pthread_key_t ErrorKey
;
38 static void ErrorDestroy(void *Obj
) {delete (GlobalError
*)Obj
;};
39 static void KeyAlloc() {pthread_key_create(&ErrorKey
,ErrorDestroy
);};
41 GlobalError
*_GetErrorObj() {
42 static pthread_once_t Once
= PTHREAD_ONCE_INIT
;
43 pthread_once(&Once
,KeyAlloc
);
45 void *Res
= pthread_getspecific(ErrorKey
);
47 pthread_setspecific(ErrorKey
,Res
= new GlobalError
);
48 return (GlobalError
*)Res
;
51 GlobalError
*_GetErrorObj() {
52 static GlobalError
*Obj
= new GlobalError
;
57 // GlobalError::GlobalError - Constructor /*{{{*/
58 GlobalError::GlobalError() : PendingFlag(false) {}
60 // GlobalError::FatalE - Get part of the error string from errno /*{{{*/
61 bool GlobalError::FatalE(const char *Function
,const char *Description
,...) {
63 va_start(args
,Description
);
64 return InsertErrno(FATAL
, Function
, Description
, args
);
67 // GlobalError::Errno - Get part of the error string from errno /*{{{*/
68 bool GlobalError::Errno(const char *Function
,const char *Description
,...) {
70 va_start(args
,Description
);
71 return InsertErrno(ERROR
, Function
, Description
, args
);
74 // GlobalError::WarningE - Get part of the warning string from errno /*{{{*/
75 bool GlobalError::WarningE(const char *Function
,const char *Description
,...) {
77 va_start(args
,Description
);
78 return InsertErrno(WARNING
, Function
, Description
, args
);
81 // GlobalError::NoticeE - Get part of the notice string from errno /*{{{*/
82 bool GlobalError::NoticeE(const char *Function
,const char *Description
,...) {
84 va_start(args
,Description
);
85 return InsertErrno(NOTICE
, Function
, Description
, args
);
88 // GlobalError::DebugE - Get part of the debug string from errno /*{{{*/
89 bool GlobalError::DebugE(const char *Function
,const char *Description
,...) {
91 va_start(args
,Description
);
92 return InsertErrno(DEBUG
, Function
, Description
, args
);
95 // GlobalError::InsertErrno - formats an error message with the errno /*{{{*/
96 bool GlobalError::InsertErrno(MsgType type
, const char* Function
,
97 const char* Description
, va_list const &args
) {
99 vsnprintf(S
,sizeof(S
),Description
,args
);
100 snprintf(S
+ strlen(S
),sizeof(S
) - strlen(S
),
101 " - %s (%i: %s)", Function
, errno
, strerror(errno
));
102 return Insert(type
, S
, args
);
105 // GlobalError::Fatal - Add a fatal error to the list /*{{{*/
106 bool GlobalError::Fatal(const char *Description
,...) {
108 va_start(args
,Description
);
109 return Insert(FATAL
, Description
, args
);
112 // GlobalError::Error - Add an error to the list /*{{{*/
113 bool GlobalError::Error(const char *Description
,...) {
115 va_start(args
,Description
);
116 return Insert(ERROR
, Description
, args
);
119 // GlobalError::Warning - Add a warning to the list /*{{{*/
120 bool GlobalError::Warning(const char *Description
,...) {
122 va_start(args
,Description
);
123 return Insert(WARNING
, Description
, args
);
126 // GlobalError::Notice - Add a notice to the list /*{{{*/
127 bool GlobalError::Notice(const char *Description
,...)
130 va_start(args
,Description
);
131 return Insert(NOTICE
, Description
, args
);
134 // GlobalError::Debug - Add a debug to the list /*{{{*/
135 bool GlobalError::Debug(const char *Description
,...)
138 va_start(args
,Description
);
139 return Insert(DEBUG
, Description
, args
);
142 // GlobalError::Insert - Insert a new item at the end /*{{{*/
143 bool GlobalError::Insert(MsgType type
, const char* Description
,
144 va_list const &args
) {
146 vsnprintf(S
,sizeof(S
),Description
,args
);
148 Item
const m(S
, type
);
149 Messages
.push_back(m
);
151 if (type
== ERROR
|| type
== FATAL
)
154 if (type
== FATAL
|| type
== DEBUG
)
155 std::clog
<< m
<< std::endl
;
160 // GlobalError::PopMessage - Pulls a single message out /*{{{*/
161 bool GlobalError::PopMessage(std::string
&Text
) {
162 if (Messages
.empty() == true)
165 Item
const msg
= Messages
.front();
166 Messages
.pop_front();
168 bool const Ret
= (msg
.Type
== ERROR
|| msg
.Type
== FATAL
);
170 if (PendingFlag
== false || Ret
== false)
173 // check if another error message is pending
174 for (std::list
<Item
>::const_iterator m
= Messages
.begin();
175 m
!= Messages
.end(); m
++)
176 if (m
->Type
== ERROR
|| m
->Type
== FATAL
)
183 // GlobalError::DumpErrors - Dump all of the errors/warns to cerr /*{{{*/
184 void GlobalError::DumpErrors(std::ostream
&out
, MsgType
const &trashhold
,
185 bool const &mergeStack
) {
186 if (mergeStack
== true)
187 for (std::list
<MsgStack
>::const_reverse_iterator s
= Stacks
.rbegin();
188 s
!= Stacks
.rend(); ++s
)
189 Messages
.insert(Messages
.begin(), s
->Messages
.begin(), s
->Messages
.end());
191 for (std::list
<Item
>::const_iterator m
= Messages
.begin();
192 m
!= Messages
.end(); m
++)
193 if (m
->Type
>= trashhold
)
194 out
<< (*m
) << std::endl
;
198 // GlobalError::Discard - Discard /*{{{*/
199 void GlobalError::Discard() {
204 // GlobalError::empty - does our error list include anything? /*{{{*/
205 bool GlobalError::empty(MsgType
const &trashhold
) const {
206 if (PendingFlag
== true)
209 if (Messages
.empty() == true)
212 for (std::list
<Item
>::const_iterator m
= Messages
.begin();
213 m
!= Messages
.end(); m
++)
214 if (m
->Type
>= trashhold
)
220 // GlobalError::PushToStack /*{{{*/
221 void GlobalError::PushToStack() {
222 MsgStack
pack(Messages
, PendingFlag
);
223 Stacks
.push_back(pack
);
227 // GlobalError::RevertToStack /*{{{*/
228 void GlobalError::RevertToStack() {
230 MsgStack pack
= Stacks
.back();
231 Messages
= pack
.Messages
;
232 PendingFlag
= pack
.PendingFlag
;
236 // GlobalError::MergeWithStack /*{{{*/
237 void GlobalError::MergeWithStack() {
238 MsgStack pack
= Stacks
.back();
239 Messages
.insert(Messages
.begin(), pack
.Messages
.begin(), pack
.Messages
.end());
240 PendingFlag
= PendingFlag
|| pack
.PendingFlag
;