source: trunk/include/samhain.h@ 400

Last change on this file since 400 was 373, checked in by katerina, 13 years ago

Patch for ticket #265 (inotify). Handling of added subdirectories and file list rescan.

File size: 14.1 KB
RevLine 
[1]1/* SAMHAIN file system integrity testing */
2/* Copyright (C) 1999 Rainer Wichmann */
3/* */
4/* This program is free software; you can redistribute it */
5/* and/or modify */
6/* it under the terms of the GNU General Public License as */
7/* published by */
8/* the Free Software Foundation; either version 2 of the License, or */
9/* (at your option) any later version. */
10/* */
11/* This program is distributed in the hope that it will be useful, */
12/* but WITHOUT ANY WARRANTY; without even the implied warranty of */
13/* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the */
14/* GNU General Public License for more details. */
15/* */
16/* You should have received a copy of the GNU General Public License */
17/* along with this program; if not, write to the Free Software */
18/* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */
19
20#ifndef SAMHAIN_H
21#define SAMHAIN_H
22
23#include <sys/types.h>
24#include "slib.h"
25
26#ifdef SH_ENCRYPT
27#include "rijndael-api-fst.h"
28#endif
29
30/**************************************************
31 *
32 * STANDARD DEFINES
33 *
34 **************************************************/
35
[295]36/* IPv6 */
37#if defined(HAVE_GETNAMEINFO) && defined(HAVE_GETADDRINFO)
38
39#if defined(SH_COMPILE_STATIC) && defined(__linux__)
40#undef USE_IPVX
41#define SH_SOCKMAX 1
42#else
43
44#if defined(USE_IPV4)
45#undef USE_IPVX
46#else
47#define USE_IPVX 1
48#endif
49
50#define SH_SOCKMAX 8
51#endif
52
53#else
54#undef USE_IPVX
55#define SH_SOCKMAX 1
56#endif
57
58/* end IPv6 */
59
[1]60#define REPLACE_OLD
61
[283]62/* Standard buffer sizes.
63 * IPv6 is 8 groups of 4 hex digits seperated by colons.
[1]64 */
[295]65#define SH_IP_BUF 48
[1]66#define SH_MINIBUF 64
67#define SH_BUFSIZE 1024
68#define SH_MAXBUF 4096
69#define SH_PATHBUF 256
[170]70#define SH_MSG_BUF 64512
[1]71
[132]72#define SH_ERRBUF_SIZE 64
[131]73
[170]74/* MAX_PATH_STORE must be >= KEY_LEN
75 */
76#define MAX_PATH_STORE 12287
77
[1]78/* Sizes for arrays (user, group, timestamp).
79 */
[40]80#define SOCKPASS_MAX 14
[149]81#define USER_MAX 20
82#define GROUP_MAX 20
83#define TIM_MAX 32
[1]84
[149]85#define CMODE_SIZE 11
[40]86
[149]87#define ATTRBUF_SIZE 16
88#define ATTRBUF_USED 12
89
[1]90/* The number of bytes in a key,
91 * the number of chars in its hex repesentation,
92 * and the block size of the hash algorithm.
93 */
94#define KEY_BYT 24
95#define KEY_LEN 48
96#define KEY_BLOCK 24
[133]97#define KEYBUF_SIZE (KEY_LEN+1)
[1]98
99/* The length of the compiled-in password.
100 */
101#define PW_LEN 8
102
103#undef GOOD
104#define GOOD 1
105#undef BAD
106#define BAD 0
107#undef ON
108#define ON 1
109#undef OFF
110#define OFF 0
111#undef S_TRUE
112#define S_TRUE 1
113#undef S_FALSE
114#define S_FALSE 0
115
116/* An unsigned integer guaranteed to be 32 bit.
117 */
118#if defined(HAVE_INT_32)
119#define UINT32 unsigned int
120#define SINT32 int
121#elif defined(HAVE_LONG_32)
122#define UINT32 unsigned long
123#define SINT32 long
124#elif defined(HAVE_SHORT_32)
125#define UINT32 unsigned short
126#define SINT32 short
127#endif
128
129#ifdef HAVE_INTTYPES_H
130#include <inttypes.h>
[156]131#endif
[1]132#ifdef HAVE_STDINT_H
133#include <stdint.h>
134#endif
135
[265]136#if !defined(HAVE_UINT16_T)
137#define UINT16 unsigned short
138#else
139#define UINT16 uint16_t
140#endif
141
[156]142#if !defined(HAVE_UINT64_T)
[1]143
144#ifdef HAVE_LONG_LONG_64
145#define UINT64 unsigned long long
146#else
147#ifdef HAVE_LONG_64
148#define UINT64 unsigned long
149#else
[156]150#error "no 64bit type found"
[1]151#endif
152#endif
153
154#else
155#define UINT64 uint64_t
156#endif
157
158
159
160#define UBYTE unsigned char
161
162
163enum {
164 SH_CHECK_NONE = 0,
165 SH_CHECK_INIT = 1,
166 SH_CHECK_CHECK = 2
167};
168
[143]169#define SH_MOD_THREAD 1
170#define SH_MOD_ACTIVE 0
171#define SH_MOD_FAILED -1
[207]172#define SH_MOD_OFFSET 10
[156]173
[114]174/* Flags for file status
175 */
176#define SH_FFLAG_ALLIGNORE (1<<0)
177#define SH_FFLAG_VISITED (1<<1)
178#define SH_FFLAG_CHECKED (1<<3)
179#define SH_FFLAG_REPORTED (1<<3)
[115]180#define SH_FFLAG_SUIDCHK (1<<4)
[93]181
[114]182#define SH_FFLAG_ALLIGNORE_SET(a) (((a) & SH_FFLAG_ALLIGNORE) != 0)
183#define SET_SH_FFLAG_ALLIGNORE(a) ((a) |= SH_FFLAG_ALLIGNORE)
184#define CLEAR_SH_FFLAG_ALLIGNORE(a) ((a) &= ~SH_FFLAG_ALLIGNORE)
185
186#define SH_FFLAG_VISITED_SET(a) (((a) & SH_FFLAG_VISITED) != 0)
187#define SET_SH_FFLAG_VISITED(a) ((a) |= SH_FFLAG_VISITED)
188#define CLEAR_SH_FFLAG_VISITED(a) ((a) &= ~SH_FFLAG_VISITED)
189
190#define SH_FFLAG_CHECKED_SET(a) (((a) & SH_FFLAG_VISITED) != 0)
191#define SET_SH_FFLAG_CHECKED(a) ((a) |= SH_FFLAG_VISITED)
192#define CLEAR_SH_FFLAG_CHECKED(a) ((a) &= ~SH_FFLAG_VISITED)
193
194#define SH_FFLAG_REPORTED_SET(a) (((a) & SH_FFLAG_REPORTED) != 0)
195#define SET_SH_FFLAG_REPORTED(a) ((a) |= SH_FFLAG_REPORTED)
196#define CLEAR_SH_FFLAG_REPORTED(a) ((a) &= ~SH_FFLAG_REPORTED)
197
[115]198#define SH_FFLAG_SUIDCHK_SET(a) (((a) & SH_FFLAG_SUIDCHK) != 0)
199#define SET_SH_FFLAG_SUIDCHK(a) ((a) |= SH_FFLAG_SUIDCHK)
200#define CLEAR_SH_FFLAG_SUIDCHK(a) ((a) &= ~SH_FFLAG_SUIDCHK)
[114]201
[367]202/* Flags for inotify
203 */
204#define SH_INOTIFY_USE (1<<0)
205#define SH_INOTIFY_DOSCAN (1<<1)
206#define SH_INOTIFY_NEEDINIT (1<<2)
[373]207#define SH_INOTIFY_INSCAN (1<<3)
[114]208
[367]209
[1]210/**************************************************
211 *
212 * TYPEDEFS
213 *
214 **************************************************/
215
216enum {
217 SH_LEVEL_READONLY = 1,
218 SH_LEVEL_LOGFILES = 2,
219 SH_LEVEL_LOGGROW = 3,
220 SH_LEVEL_NOIGNORE = 4,
221 SH_LEVEL_ALLIGNORE = 5,
222 SH_LEVEL_ATTRIBUTES = 6,
223 SH_LEVEL_USER0 = 7,
224 SH_LEVEL_USER1 = 8,
[27]225 SH_LEVEL_USER2 = 9,
226 SH_LEVEL_USER3 = 10,
227 SH_LEVEL_USER4 = 11,
228 SH_LEVEL_PRELINK = 12
[1]229};
230
231typedef struct {
232 time_t alarm_interval;
233 time_t alarm_last;
234} sh_timer_t;
235
236typedef struct {
237 char path[SH_PATHBUF];
238 char hash[KEY_LEN+1];
239} sh_sh_df;
240
241typedef struct {
242 char user[USER_MAX];
243 char group[GROUP_MAX];
244 char home[SH_PATHBUF];
245 uid_t uid;
246 gid_t gid;
247} sh_sh_user;
248
249typedef struct {
250 char name[SH_PATHBUF]; /* local hostname */
251 char system[SH_MINIBUF]; /* system */
252 char release[SH_MINIBUF]; /* release */
253 char machine[SH_MINIBUF]; /* machine */
254} sh_sh_local;
255
256typedef struct {
257 char name[SH_PATHBUF];
258 char alt[SH_PATHBUF];
259} sh_sh_remote;
260
261typedef struct {
262 unsigned long bytes_hashed; /* bytes last check */
263 unsigned long bytes_speed; /* bytes/sec last check */
264 unsigned long mail_success; /* mails sent */
265 unsigned long mail_failed; /* mails not sent */
266 time_t time_start; /* start last check */
267 time_t time_check; /* time last check */
268 unsigned long dirs_checked; /* #dirs last check */
269 unsigned long files_checked; /* #files last check */
270} sh_sh_stat;
271
272typedef struct {
273 int exit; /* exit value */
274 int checkSum; /* whether to init/check checksums */
275 int update; /* update db */
276 int opts; /* reading cl options */
[256]277 int started; /* finished with startup stuff */
[1]278 int isdaemon; /* daemon or not */
279 int loop; /* go in loop even if not daemon */
280 int nice; /* desired nicety */
281 int isserver; /* server or not */
282 int islocked; /* BAD if logfile not locked */
283 int smsg; /* GOOD if end message sent */
284 int log_start; /* TRUE if new audit trail */
285 int reportonce; /* TRUE if bad files only once rep.*/
286 int fulldetail; /* TRUE if full details requested */
287 int client_severity; /* TRUE if client severity used */
288 int client_class; /* TRUE if client class used */
289 int audit;
290 unsigned long aud_mask;
[367]291 int hidefile; /* TRUE if file not shown in log */
292 int inotify; /* Flags for inotify */
[1]293} sh_sh_flag;
294
295typedef struct {
296
297 char prg_name[8];
[162]298
299 UINT64 pid;
[1]300
301 sh_sh_df exec;
302 sh_sh_df conf;
303 sh_sh_df data;
304
305 sh_sh_user real;
306 sh_sh_user effective;
307 sh_sh_user run;
308
309 sh_sh_local host;
310
311 sh_sh_remote srvtime;
312 sh_sh_remote srvmail;
313 sh_sh_remote srvexport;
314 sh_sh_remote srvcons;
315 sh_sh_remote srvlog;
316
317 sh_sh_stat statistics;
318 sh_sh_flag flag;
319
320#ifdef SH_STEALTH
321 unsigned long off_data;
322#endif
323
324 sh_timer_t mailNum;
325 sh_timer_t mailTime;
326 sh_timer_t fileCheck;
327
328 int looptime; /* timing for main loop */
329 /*@null@*//*@out@*/ char * timezone;
330} sh_struct;
331
332
333extern volatile int sig_raised;
334extern volatile int sig_urgent;
335extern volatile int sig_debug_switch; /* SIGUSR1 */
336extern volatile int sig_suspend_switch; /* SIGUSR2 */
[143]337extern volatile int sh_global_suspend_flag;
[1]338extern volatile int sig_fresh_trail; /* SIGIOT */
[143]339extern volatile int sh_thread_pause_flag;
[1]340extern volatile int sig_config_read_again; /* SIGHUP */
341extern volatile int sig_terminate; /* SIGQUIT */
342extern volatile int sig_termfast; /* SIGTERM */
343extern volatile int sig_force_check; /* SIGTTOU */
344
345extern long int eintr__result;
346
[20]347extern int sh_argc_store;
348extern char ** sh_argv_store;
349
[1]350#include "sh_calls.h"
351
352
353typedef struct {
[40]354 char sh_sockpass[2*SOCKPASS_MAX+2];
[1]355 char sigkey_old[KEY_LEN+1];
356 char sigkey_new[KEY_LEN+1];
357 char mailkey_old[KEY_LEN+1];
358 char mailkey_new[KEY_LEN+1];
359 char crypt[KEY_LEN+1];
360 char session[KEY_LEN+1];
361 char vernam[KEY_LEN+1];
362 int mlock_failed;
363
364 char pw[PW_LEN];
365
366 char poolv[KEY_BYT];
367 int poolc;
368
369 int rngI;
370 UINT32 rng0[3];
371 UINT32 rng1[3];
372 UINT32 rng2[3];
373
[156]374 UINT32 res_vec[6];
375
[1]376 UINT32 ErrFlag[2];
377
378#ifdef SH_ENCRYPT
379 /*@out@*/ keyInstance keyInstE;
380 /*@out@*/ keyInstance keyInstD;
381#endif
382} sh_key_t;
383
384extern sh_struct sh;
385/*@null@*/ extern sh_key_t *skey;
386
[22]387/**************************************************
388 *
389 * macros
390 *
391 **************************************************/
[1]392
[76]393#if defined(__GNUC__) && (__GNUC__ >= 4)
[149]394#define SH_GNUC_SENTINEL __attribute__((__sentinel__))
[76]395#else
[149]396#define SH_GNUC_SENTINEL
[76]397#endif
398
[149]399#if defined(__GNUC__) && (__GNUC__ >= 3)
400#undef SH_GNUC_PURE
401#define SH_GNUC_PURE __attribute__((pure))
402#undef SH_GNUC_CONST
403#define SH_GNUC_CONST __attribute__((const))
404#undef SH_GNUC_NORETURN
405#define SH_GNUC_NORETURN __attribute__((noreturn))
406#undef SH_GNUC_MALLOC
407#define SH_GNUC_MALLOC __attribute__((malloc))
408#else
409#undef SH_GNUC_PURE
410#define SH_GNUC_PURE
411#undef SH_GNUC_CONST
412#define SH_GNUC_CONST
413#undef SH_GNUC_NORETURN
414#define SH_GNUC_NORETURN
415#undef SH_GNUC_MALLOC
416#define SH_GNUC_MALLOC
417#endif
418
419
[76]420/* The semantics of the built-in are that it is expected that expr == const
421 * for __builtin_expect ((expr), const)
422 */
423#if defined(__GNUC__) && (__GNUC__ > 2) && defined(__OPTIMIZE__)
424#define SH_LIKELY(expr) (__builtin_expect((expr), 1))
425#define SH_UNLIKELY(expr) (__builtin_expect((expr), 0))
426#else
427#define SH_LIKELY(expr) (expr)
428#define SH_UNLIKELY(expr) (expr)
429#endif
430
[22]431/* signal-safe log function
432 */
[170]433int safe_logger (int thesignal, int method, char * details);
434void safe_fatal (const char * details, const char *f, int l);
[22]435
[25]436#define SH_VALIDATE_EQ(a,b) \
[22]437 do { \
[34]438 if ((a) != (b)) safe_fatal(#a " != " #b, FIL__, __LINE__);\
[22]439 } while (0)
440
[25]441#define SH_VALIDATE_NE(a,b) \
442 do { \
[34]443 if ((a) == (b)) safe_fatal(#a " == " #b, FIL__, __LINE__);\
[25]444 } while (0)
[22]445
[68]446#define SH_VALIDATE_GE(a,b) \
447 do { \
448 if ((a) < (b)) safe_fatal(#a " < " #b, FIL__, __LINE__);\
449 } while (0)
450
[1]451#if defined(HAVE_MLOCK) && !defined(HAVE_BROKEN_MLOCK)
452#define MLOCK(a, b) \
453 if ((skey != NULL) && skey->mlock_failed == SL_FALSE){ \
454 (void) sl_set_suid(); \
[19]455 if (sh_unix_mlock(FIL__, __LINE__, a, b) < 0) skey->mlock_failed = SL_TRUE; \
[1]456 (void) sl_unset_suid(); }
457#else
458#define MLOCK(a, b) \
459 ;
460#endif
461
462#if defined(HAVE_MLOCK) && !defined(HAVE_BROKEN_MLOCK)
463#define MUNLOCK(a, b) \
464 if ((skey != NULL) && skey->mlock_failed == SL_FALSE){ \
465 (void) sl_set_suid(); \
466 (void) sh_unix_munlock( a, b );\
467 (void) sl_unset_suid(); }
468#else
469#define MUNLOCK(a, b) \
470 ;
471#endif
472
473#ifdef SH_STEALTH
474void sh_do_encode (char * str, int len);
475#define sh_do_decode sh_do_encode
476#endif
477
478/* #if defined(SCREW_IT_UP)
479 * extern volatile int sh_not_traced;
480 * inline int sh_sigtrap_prepare();
481 * inline int sh_derr();
482 * #endif
483 */
484
485#if defined(SCREW_IT_UP) && (defined(__FreeBSD__) || defined(__linux__)) && defined(__i386__)
486#define BREAKEXIT(expr) \
487 do { \
488 int ixi; \
489 for (ixi = 0; ixi < 8; ++ixi) { \
490 if ((*(volatile unsigned *)((unsigned) expr + ixi) & 0xff) == 0xcc) \
491 _exit(EXIT_FAILURE); \
492 } \
493 } \
494 while (1 == 0)
495#else
496#define BREAKEXIT(expr)
497#endif
498
499
500
501#include "sh_cat.h"
502#include "sh_trace.h"
503#include "sh_mem.h"
504
505#endif
506
507/* CRIT: */
508/* NEW_CLIENT <client> */
509/* BAD_CLIENT <client> -- <details> */
510/* ERR_CLIENT <client> -- <details> */
511
512/* ALERT: */
513/* LOG_KEY samhain|yule <key> */
514/* STARTUP samhain|yule -- user <username> */
515/* EXIT samhain|yule */
516/* GOODSIG <file> <user> */
517/* FP_KEY <fingerprint> */
518/* GOODSIG_DAT <file> <user> */
519/* FP_KEY_DAT <fingerprint> */
520/* TIGER_CFG <file> <checksum> */
521/* TIGER_DAT <file> <checksum> */
522
523/* PANIC -- <details> */
524/* ERROR -- <details> */
525
526/* Policy */
527/* POLICY <code> <file> */
528/* <code> = MISSING || ADDED || NOT_A_DIRECTORY || <policy> */
529
530
531
Note: See TracBrowser for help on using the repository browser.