13 # define va_copy(DEST,SRC) __va_copy((DEST),(SRC))
15 # ifdef HAVE_VA_LIST_AS_ARRAY
16 # define va_copy(DEST,SRC) (*(DEST) = *(SRC))
18 # define va_copy(DEST,SRC) ((DEST) = (SRC))
23 /*@access rpmlogRec @*/
28 static /*@only@*/ /*@null@*/ rpmlogRec recs = NULL;
31 * Wrapper to free(3), hides const compilation noise, permit NULL, return NULL.
32 * @param p memory to free
35 /*@unused@*/ static inline /*@null@*/ void *
36 _free(/*@only@*/ /*@null@*/ /*@out@*/ const void * p) /*@modifies p@*/
38 if (p != NULL) free((void *)p);
42 int rpmlogGetNrecs(void)
49 if (recs != NULL && nrecs > 0)
50 return recs[nrecs-1].code;
55 const char * rpmlogMessage(void)
57 if (recs != NULL && nrecs > 0)
58 return recs[nrecs-1].message;
59 return _("(no error)");
63 void rpmlogPrint(FILE *f)
71 for (i = 0; i < nrecs; i++) {
72 rpmlogRec rec = recs + i;
73 if (rec->message && *rec->message)
74 fprintf(f, " %s", rec->message);
79 void rpmlogClose (void)
80 /*@globals recs, nrecs @*/
81 /*@modifies recs, nrecs @*/
86 for (i = 0; i < nrecs; i++) {
87 rpmlogRec rec = recs + i;
88 rec->message = _free(rec->message);
94 void rpmlogOpen (/*@unused@*/ const char *ident, /*@unused@*/ int option,
95 /*@unused@*/ int facility)
100 static unsigned rpmlogMask = RPMLOG_UPTO( RPMLOG_NOTICE );
104 static /*@unused@*/ unsigned rpmlogFacility = RPMLOG_USER;
107 int rpmlogSetMask (int mask)
108 /*@globals rpmlogMask @*/
109 /*@modifies rpmlogMask @*/
111 int omask = rpmlogMask;
118 static /*@null@*/ rpmlogCallback _rpmlogCallback = NULL;
120 rpmlogCallback rpmlogSetCallback(rpmlogCallback cb)
121 /*@globals _rpmlogCallback @*/
122 /*@modifies _rpmlogCallback @*/
124 rpmlogCallback ocb = _rpmlogCallback;
125 _rpmlogCallback = cb;
129 /*@unchecked@*/ /*@null@*/
130 static FILE * _stdlog = NULL;
132 FILE * rpmlogSetFile(FILE * fp)
133 /*@globals _stdlog @*/
134 /*@modifies _stdlog @*/
136 FILE * ofp = _stdlog;
141 /*@-readonlytrans@*/ /* FIX: double indirection. */
142 /*@observer@*/ /*@unchecked@*/
143 static char *rpmlogMsgPrefix[] = {
144 N_("fatal error: "),/*!< RPMLOG_EMERG */
145 N_("fatal error: "),/*!< RPMLOG_ALERT */
146 N_("fatal error: "),/*!< RPMLOG_CRIT */
147 N_("error: "), /*!< RPMLOG_ERR */
148 N_("warning: "), /*!< RPMLOG_WARNING */
149 "", /*!< RPMLOG_NOTICE */
150 "", /*!< RPMLOG_INFO */
151 "D: ", /*!< RPMLOG_DEBUG */
155 #if !defined(HAVE_VSNPRINTF)
156 static inline int vsnprintf(char * buf, /*@unused@*/ int nb,
157 const char * fmt, va_list ap)
159 return vsprintf(buf, fmt, ap);
164 /*@-compmempass@*/ /* FIX: rpmlogMsgPrefix[] dependent, not unqualified */
165 /*@-nullstate@*/ /* FIX: rpmlogMsgPrefix[] may be NULL */
166 static void vrpmlog (unsigned code, const char *fmt, va_list ap)
167 /*@globals nrecs, recs, internalState @*/
168 /*@modifies nrecs, recs, internalState @*/
170 unsigned pri = RPMLOG_PRI(code);
171 unsigned mask = RPMLOG_MASK(pri);
173 /*@unused@*/ unsigned fac = RPMLOG_FAC(code);
176 int msgnb = BUFSIZ, nb;
177 FILE * msgout = (_stdlog ? _stdlog : stderr);
179 if ((mask & rpmlogMask) == 0)
183 msgbuf = xmalloc(msgnb);
186 /* Allocate a sufficently large buffer for output. */
189 /*@-sysunrecog -usedef@*/ va_copy(apc, ap); /*@=sysunrecog =usedef@*/
190 nb = vsnprintf(msgbuf, msgnb, fmt, apc);
191 if (nb > -1 && nb < msgnb)
193 if (nb > -1) /* glibc 2.1 (and later) */
197 msgbuf = xrealloc(msgbuf, msgnb);
202 msgbuf[msgnb - 1] = '\0';
205 /* Save copy of all messages at warning (or below == "more important"). */
207 if (pri <= RPMLOG_WARNING) {
210 recs = xmalloc((nrecs+2) * sizeof(*recs));
212 recs = xrealloc(recs, (nrecs+2) * sizeof(*recs));
213 recs[nrecs].code = code;
214 recs[nrecs].message = msg = xrealloc(msgbuf, strlen(msgbuf)+1);
215 msgbuf = NULL; /* XXX don't free at exit. */
216 recs[nrecs+1].code = 0;
217 recs[nrecs+1].message = NULL;
220 if (_rpmlogCallback) {
221 /*@-noeffectuncon@*/ /* FIX: useless callback */
224 return; /* XXX Preserve legacy rpmError behavior. */
230 /* rpmMessage behavior */
235 msgout = (_stdlog ? _stdlog : stdout);
241 case RPMLOG_ERR: /* XXX Legacy rpmError behavior used stdout w/o prefix. */
247 if (rpmlogMsgPrefix[pri] && *rpmlogMsgPrefix[pri])
248 (void) fputs(_(rpmlogMsgPrefix[pri]), msgout);
250 (void) fputs(msg, msgout);
251 (void) fflush(msgout);
252 msgbuf = _free(msgbuf);
253 if (pri <= RPMLOG_CRIT)
256 /*@=compmempass =nullstate@*/
259 void rpmlog (int code, const char *fmt, ...)
264 /*@-internalglobs@*/ /* FIX: shrug */
265 vrpmlog(code, fmt, ap);
270 int rpmErrorCode(void)
275 const char * rpmErrorString(void)
277 return rpmlogMessage();
280 rpmlogCallback rpmErrorSetCallback(rpmlogCallback cb)
282 return rpmlogSetCallback(cb);