Cleanup logging.
[apps/pfixtools.git] / common / common.h
1 /******************************************************************************/
2 /*          pfixtools: a collection of postfix related tools                  */
3 /*          ~~~~~~~~~                                                         */
4 /*  ________________________________________________________________________  */
5 /*                                                                            */
6 /*  Redistribution and use in source and binary forms, with or without        */
7 /*  modification, are permitted provided that the following conditions        */
8 /*  are met:                                                                  */
9 /*                                                                            */
10 /*  1. Redistributions of source code must retain the above copyright         */
11 /*     notice, this list of conditions and the following disclaimer.          */
12 /*  2. Redistributions in binary form must reproduce the above copyright      */
13 /*     notice, this list of conditions and the following disclaimer in the    */
14 /*     documentation and/or other materials provided with the distribution.   */
15 /*  3. The names of its contributors may not be used to endorse or promote    */
16 /*     products derived from this software without specific prior written     */
17 /*     permission.                                                            */
18 /*                                                                            */
19 /*  THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND   */
20 /*  ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE     */
21 /*  IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR        */
22 /*  PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS    */
23 /*  BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR    */
24 /*  CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF      */
25 /*  SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS  */
26 /*  INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN   */
27 /*  CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)   */
28 /*  ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF    */
29 /*  THE POSSIBILITY OF SUCH DAMAGE.                                           */
30 /******************************************************************************/
31
32 /*
33  * Copyright © 2007 Pierre Habouzit
34  * Copyright © 2008 Florent Bruneau
35  */
36
37 #ifndef PFIXTOOLS_COMMON_H
38 #define PFIXTOOLS_COMMON_H
39
40 #include <errno.h>
41 #include <fcntl.h>
42 #include <limits.h>
43 #include <netinet/in.h>
44 #include <signal.h>
45 #include <stdbool.h>
46 #include <stdbool.h>
47 #include <stddef.h>
48 #include <stdint.h>
49 #include <stdio.h>
50 #include <stdlib.h>
51 #include <syslog.h>
52 #include <sys/socket.h>
53 #include <sys/stat.h>
54 #include <sys/types.h>
55 #include <time.h>
56 #include <unistd.h>
57
58 #include "mem.h"
59
60
61 #define __tostr(x)  #x
62 #define STR(x)      __tostr(x)
63
64 typedef int  (*initcall_t)(void);
65 typedef void (*exitcall_t)(void);
66
67 #define __init __attribute__((__used__,__section__(".mad.init")))
68 #define __exit __attribute__((__used__,__section__(".mad.exit")))
69
70 #define module_init(fn)  static __init initcall_t __init_##fn = fn;
71 #define module_exit(fn)  static __exit exitcall_t __exit_##fn = fn;
72
73 #define __log(Level, Fmt, ...)                                    \
74     if (log_level >= Level) {                                     \
75         syslog(Level, Fmt, ##__VA_ARGS__);                        \
76     }
77
78 #define debug(Fmt, ...)  __log(LOG_DEBUG,   Fmt, ##__VA_ARGS__)
79 #define notice(Fmt, ...) __log(LOG_NOTICE,  Fmt, ##__VA_ARGS__)
80 #define info(Fmt, ...)   __log(LOG_INFO,    Fmt, ##__VA_ARGS__)
81 #define warn(Fmt, ...)   __log(LOG_WARNING, Fmt, ##__VA_ARGS__)
82 #define err(Fmt, ...)    __log(LOG_ERR,     Fmt, ##__VA_ARGS__)
83 #define crit(Fmt, ...)   __log(LOG_CRIT,    Fmt, ##__VA_ARGS__)
84 #define alert(Fmt, ...)  __log(LOG_ALERT,   Fmt, ##__VA_ARGS__)
85 #define emerg(Fmt, ...)  __log(LOG_ALERT,   Fmt, ##__VA_ARGS__)
86
87 #define UNIXERR(fun)     err("%s:%d:%s %s: %m",                      \
88                              __FILE__, __LINE__, __func__, fun)
89
90 extern sig_atomic_t sigint;
91 extern sig_atomic_t sighup;
92 extern int          log_level;
93
94 void common_sighandler(int sig);
95
96 int tcp_bind(const struct sockaddr *addr, socklen_t len);
97 int tcp_listen(const struct sockaddr *addr, socklen_t len);
98 int tcp_listen_nonblock(const struct sockaddr *addr, socklen_t len);
99 int accept_nonblock(int fd);
100 int xwrite(int fd, const char *s, size_t l);
101
102 int daemon_detach(void);
103 int drop_privileges(const char *user, const char *group);
104
105 int pidfile_open(const char *name);
106 int pidfile_refresh(void);
107
108 int common_setup(const char* pidfile, bool unsafe, const char* runas_user,
109                  const char* runas_group, bool daemonize);
110
111 #define DECLARE_MAIN                                              \
112     static int main_initialize(void)                              \
113     {                                                             \
114         openlog(DAEMON_NAME, LOG_PID, LOG_MAIL);                  \
115         signal(SIGPIPE, SIG_IGN);                                 \
116         signal(SIGINT,  &common_sighandler);                      \
117         signal(SIGTERM, &common_sighandler);                      \
118         signal(SIGHUP,  &common_sighandler);                      \
119         signal(SIGSEGV, &common_sighandler);                      \
120         info("starting...");                                      \
121         return 0;                                                 \
122     }                                                             \
123                                                                   \
124     static void main_shutdown(void)                               \
125     {                                                             \
126         closelog();                                               \
127     }                                                             \
128                                                                   \
129     module_init(main_initialize);                                 \
130     module_exit(main_shutdown);
131
132 #endif