4 * Copyright (C) 1995-2001 by Darren Reed.
6 * See the IPFILTER.LICENCE file for details on licencing.
8 #if defined(KERNEL) || defined(_KERNEL)
14 #include <sys/param.h>
15 #if defined(__hpux) && (HPUXREV >= 1111) && !defined(_KERNEL)
16 # include <sys/kern_svcs.h>
18 #include <sys/types.h>
20 #include <sys/errno.h>
31 # include <sys/systm.h>
32 # if !defined(__svr4__) && !defined(__SVR4)
33 # include <sys/mbuf.h>
36 #include <sys/socket.h>
37 #if !defined(__hpux) && !defined(__osf__) && !defined(linux) && !defined(AIX)
38 # include <sys/ioccom.h>
41 # include <sys/filio.h>
42 # include <sys/malloc.h>
44 # include <sys/ioctl.h>
47 #include <netinet/in.h>
48 #include <netinet/in_systm.h>
49 #include <netinet/ip.h>
50 #include <netinet/tcp.h>
55 #include "netinet/ip_compat.h"
56 #include "netinet/ip_fil.h"
57 #include "netinet/ip_state.h"
58 #include "netinet/ip_scan.h"
62 #if defined(__NetBSD__)
63 #include <sys/cdefs.h>
64 __KERNEL_RCSID(0, "$NetBSD$");
66 static const char sccsid
[] = "@(#)ip_state.c 1.8 6/5/96 (C) 1993-2000 Darren Reed";
67 static const char rcsid
[] = "@(#)Id: ip_scan.c,v 2.40.2.10 2007/06/02 21:22:28 darrenr Exp";
71 #ifdef IPFILTER_SCAN /* endif at bottom of file */
74 ipscan_t
*ipsc_list
= NULL
,
76 ipscanstat_t ipsc_stat
;
78 ipfrwlock_t ipsc_rwlock
;
82 # define isalpha(x) (((x) >= 'A' && 'Z' >= (x)) || \
83 ((x) >= 'a' && 'z' >= (x)))
87 int ipsc_add
__P((void *));
88 int ipsc_delete
__P((void *));
89 struct ipscan
*ipsc_lookup
__P((char *));
90 int ipsc_matchstr
__P((sinfo_t
*, char *, int));
91 int ipsc_matchisc
__P((ipscan_t
*, ipstate_t
*, int, int, int *));
92 int ipsc_match
__P((ipstate_t
*));
94 static int ipsc_inited
= 0;
99 RWLOCK_INIT(&ipsc_rwlock
, "ip scan rwlock");
107 if (ipsc_inited
== 1) {
108 RW_DESTROY(&ipsc_rwlock
);
120 KMALLOC(isc
, ipscan_t
*);
124 err
= copyinptr(data
, isc
, sizeof(*isc
));
130 WRITE_ENTER(&ipsc_rwlock
);
132 i
= ipsc_lookup(isc
->ipsc_tag
);
134 RWLOCK_EXIT(&ipsc_rwlock
);
140 ipsc_tail
->ipsc_next
= isc
;
141 isc
->ipsc_pnext
= &ipsc_tail
->ipsc_next
;
146 isc
->ipsc_pnext
= &ipsc_list
;
148 isc
->ipsc_next
= NULL
;
153 isc
->ipsc_active
= 0;
155 ipsc_stat
.iscs_entries
++;
156 RWLOCK_EXIT(&ipsc_rwlock
);
161 int ipsc_delete(data
)
167 err
= copyinptr(data
, &isc
, sizeof(isc
));
171 WRITE_ENTER(&ipsc_rwlock
);
173 i
= ipsc_lookup(isc
.ipsc_tag
);
178 RWLOCK_EXIT(&ipsc_rwlock
);
182 *i
->ipsc_pnext
= i
->ipsc_next
;
184 i
->ipsc_next
->ipsc_pnext
= i
->ipsc_pnext
;
186 if (i
->ipsc_pnext
== &ipsc_list
)
189 ipsc_tail
= *(*i
->ipsc_pnext
)->ipsc_pnext
;
192 ipsc_stat
.iscs_entries
--;
195 RWLOCK_EXIT(&ipsc_rwlock
);
200 struct ipscan
*ipsc_lookup(tag
)
205 for (i
= ipsc_list
; i
; i
= i
->ipsc_next
)
206 if (!strcmp(i
->ipsc_tag
, tag
))
212 int ipsc_attachfr(fr
)
217 if (fr
->fr_isctag
[0]) {
218 READ_ENTER(&ipsc_rwlock
);
219 i
= ipsc_lookup(fr
->fr_isctag
);
221 ATOMIC_INC32(i
->ipsc_fref
);
223 RWLOCK_EXIT(&ipsc_rwlock
);
232 int ipsc_attachis(is
)
238 READ_ENTER(&ipsc_rwlock
);
242 if ((i
!= NULL
) && (i
!= (ipscan_t
*)-1)) {
244 ATOMIC_INC32(i
->ipsc_sref
);
246 is
->is_flags
|= IS_SC_CLIENT
;
248 is
->is_flags
|= IS_SC_MATCHC
;
250 is
->is_flags
|= IS_SC_SERVER
;
252 is
->is_flags
|= IS_SC_MATCHS
;
255 RWLOCK_EXIT(&ipsc_rwlock
);
260 int ipsc_detachfr(fr
)
267 ATOMIC_DEC32(i
->ipsc_fref
);
273 int ipsc_detachis(is
)
278 READ_ENTER(&ipsc_rwlock
);
279 if ((i
= is
->is_isc
) && (i
!= (ipscan_t
*)-1)) {
280 ATOMIC_DEC32(i
->ipsc_sref
);
282 is
->is_flags
&= ~(IS_SC_CLIENT
|IS_SC_SERVER
);
284 RWLOCK_EXIT(&ipsc_rwlock
);
290 * 'string' compare for scanning
292 int ipsc_matchstr(sp
, str
, n
)
304 for (s
= sp
->s_txt
, t
= sp
->s_msk
; i
; i
--, s
++, t
++, up
++)
312 if (!ISALPHA(*up
) || ((*s
& 0x5f) != (*up
& 0x5f)))
323 * Returns 3 if both server and client match, 2 if just server,
326 int ipsc_matchisc(isc
, is
, cl
, sl
, maxm
)
331 int i
, j
, k
, n
, ret
= 0, flags
;
333 flags
= is
->is_flags
;
336 * If we've already matched more than what is on offer, then
337 * assume we have a better match already and forget this one.
340 if (isc
->ipsc_clen
< maxm
[0])
342 if (isc
->ipsc_slen
< maxm
[1])
353 else if (((flags
& (IS_SC_MATCHC
|IS_SC_CLIENT
)) == IS_SC_CLIENT
) &&
354 cl
&& isc
->ipsc_clen
) {
356 n
= MIN(cl
, isc
->ipsc_clen
);
357 if ((n
> 0) && (!maxm
|| (n
>= maxm
[1]))) {
358 if (!ipsc_matchstr(&isc
->ipsc_cl
, is
->is_sbuf
[0], n
)) {
369 else if (((flags
& (IS_SC_MATCHS
|IS_SC_SERVER
)) == IS_SC_SERVER
) &&
370 sl
&& isc
->ipsc_slen
) {
372 n
= MIN(cl
, isc
->ipsc_slen
);
373 if ((n
> 0) && (!maxm
|| (n
>= maxm
[1]))) {
374 if (!ipsc_matchstr(&isc
->ipsc_sl
, is
->is_sbuf
[1], n
)) {
383 if (maxm
&& (ret
== 3)) {
394 int i
, j
, k
, n
, cl
, sl
, maxm
[2];
398 for (cl
= 0, n
= is
->is_smsk
[0]; n
& 1; n
>>= 1)
400 for (sl
= 0, n
= is
->is_smsk
[1]; n
& 1; n
>>= 1)
407 * Known object to scan for.
409 i
= ipsc_matchisc(isc
, is
, cl
, sl
, NULL
);
411 is
->is_flags
|= IS_SC_MATCHC
;
412 is
->is_flags
&= ~IS_SC_CLIENT
;
413 } else if (cl
>= isc
->ipsc_clen
)
414 is
->is_flags
&= ~IS_SC_CLIENT
;
416 is
->is_flags
|= IS_SC_MATCHS
;
417 is
->is_flags
&= ~IS_SC_SERVER
;
418 } else if (sl
>= isc
->ipsc_slen
)
419 is
->is_flags
&= ~IS_SC_SERVER
;
425 for (k
= 0, isc
= ipsc_list
; isc
; isc
= isc
->ipsc_next
) {
426 i
= ipsc_matchisc(isc
, is
, cl
, sl
, maxm
);
429 * We only want to remember the best match
430 * and the number of times we get a best
433 if ((j
== 3) && (i
< 3))
435 if ((i
== 3) && (j
!= 3))
449 * No matches or partial matches, so reset the respective
453 is
->is_flags
&= ~IS_SC_CLIENT
;
456 is
->is_flags
&= ~IS_SC_SERVER
;
459 * If we found the best match, then set flags appropriately.
461 if ((j
== 3) && (k
== 1)) {
462 is
->is_flags
&= ~(IS_SC_SERVER
|IS_SC_CLIENT
);
463 is
->is_flags
|= (IS_SC_MATCHS
|IS_SC_MATCHC
);
468 * If the acknowledged side of a connection has moved past the data in
469 * which we are interested, then reset respective flag.
471 t
= &is
->is_tcp
.ts_data
[0];
472 if (t
->td_end
> is
->is_s0
[0] + 15)
473 is
->is_flags
&= ~IS_SC_CLIENT
;
475 t
= &is
->is_tcp
.ts_data
[1];
476 if (t
->td_end
> is
->is_s0
[1] + 15)
477 is
->is_flags
&= ~IS_SC_SERVER
;
480 * Matching complete ?
483 if ((is
->is_flags
& IS_SC_MATCHALL
) == IS_SC_MATCHALL
) {
484 j
= isc
->ipsc_action
;
485 ipsc_stat
.iscs_acted
++;
486 } else if ((is
->is_isc
!= NULL
) &&
487 ((is
->is_flags
& IS_SC_MATCHALL
) != IS_SC_MATCHALL
) &&
488 !(is
->is_flags
& (IS_SC_CLIENT
|IS_SC_SERVER
))) {
493 ipsc_stat
.iscs_else
++;
500 * If as a result of a successful match we are to
501 * close a connection, change the "keep state" info.
502 * to block packets and generate TCP RST's.
504 is
->is_pass
&= ~FR_RETICMP
;
505 is
->is_pass
|= FR_RETRST
;
516 * check if a packet matches what we're scanning for
518 int ipsc_packet(fin
, is
)
522 int i
, j
, rv
, dlen
, off
, thoff
;
526 rv
= !IP6_EQ(&fin
->fin_fi
.fi_src
, &is
->is_src
);
528 seq
= ntohl(tcp
->th_seq
);
534 * check if this packet has more data that falls within the first
535 * 16 bytes sent in either direction.
539 if ((off
> 15) || (off
< 0))
541 thoff
= TCP_OFF(tcp
) << 2;
542 dlen
= fin
->fin_dlen
- thoff
;
550 j
= 0xffff >> (16 - dlen
);
551 i
= (0xffff & j
) << off
;
553 COPYDATA(*(mb_t
**)fin
->fin_mp
, fin
->fin_plen
- fin
->fin_dlen
+ thoff
,
554 dlen
, (void *)is
->is_sbuf
[rv
] + off
);
556 is
->is_smsk
[rv
] |= i
;
557 for (j
= 0, i
= is
->is_smsk
[rv
]; i
& 1; i
>>= 1)
562 (void) ipsc_match(is
);
565 * There is the potential here for plain text passwords to get
566 * buffered and stored for some time...
568 if (!(is
->is_flags
& IS_SC_CLIENT
))
569 bzero(is
->is_sbuf
[0], sizeof(is
->is_sbuf
[0]));
570 if (!(is
->is_flags
& IS_SC_SERVER
))
571 bzero(is
->is_sbuf
[1], sizeof(is
->is_sbuf
[1]));
577 int fr_scan_ioctl(data
, cmd
, mode
, uid
, ctx
)
589 err
= ipsc_add(data
);
592 err
= ipsc_delete(data
);
595 bcopy((char *)&ipsc_stat
, (char *)&ipscs
, sizeof(ipscs
));
596 ipscs
.iscs_list
= ipsc_list
;
597 err
= BCOPYOUT(&ipscs
, data
, sizeof(ipscs
));
608 #endif /* IPFILTER_SCAN */