⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 ip_conntrack_ftp.c

📁 Linux内核源代码 为压缩文件 是<<Linux内核>>一书中的源代码
💻 C
字号:
/* FTP extension for IP connection tracking. */#include <linux/module.h>#include <linux/netfilter.h>#include <linux/ip.h>#include <net/checksum.h>#include <net/tcp.h>#include <linux/netfilter_ipv4/lockhelp.h>#include <linux/netfilter_ipv4/ip_conntrack_helper.h>#include <linux/netfilter_ipv4/ip_conntrack_ftp.h>DECLARE_LOCK(ip_ftp_lock);struct module *ip_conntrack_ftp = THIS_MODULE;#define SERVER_STRING "227 Entering Passive Mode ("#define CLIENT_STRING "PORT "#if 0#define DEBUGP printk#else#define DEBUGP(format, args...)#endifstatic struct {	const char *pattern;	size_t plen;	char term;} search[2] = {	[IP_CT_FTP_PORT] { CLIENT_STRING, sizeof(CLIENT_STRING) - 1, '\r' },	[IP_CT_FTP_PASV] { SERVER_STRING, sizeof(SERVER_STRING) - 1, ')' }};/* Returns 0, or length of numbers */static int try_number(const char *data, size_t dlen, u_int32_t array[6],		      char term){	u_int32_t i, len;	/* Keep data pointing at next char. */	for (i = 0, len = 0; len < dlen; len++, data++) {		if (*data >= '0' && *data <= '9') {			array[i] = array[i]*10 + *data - '0';		}		else if (*data == ',')			i++;		else {			/* Unexpected character; true if it's the			   terminator and we're finished. */			if (*data == term && i == 5)				return len;			DEBUGP("Char %u (got %u nums) `%u' unexpected\n",			       len, i, *data);			return 0;		}	}	return 0;}/* Return 1 for match, 0 for accept, -1 for partial. */static int find_pattern(const char *data, size_t dlen,			const char *pattern, size_t plen,			char term,			unsigned int *numoff,			unsigned int *numlen,			u_int32_t array[6]){	if (dlen == 0)		return 0;	if (dlen < plen) {		/* Short packet: try for partial? */		if (strnicmp(data, pattern, dlen) == 0)			return -1;		else return 0;	}	if (strnicmp(data, pattern, plen) != 0) {#if 0		size_t i;		DEBUGP("ftp: string mismatch\n");		for (i = 0; i < plen; i++) {			DEBUGFTP("ftp:char %u `%c'(%u) vs `%c'(%u)\n",				 i, data[i], data[i],				 pattern[i], pattern[i]);		}#endif		return 0;	}	*numoff = plen;	*numlen = try_number(data + plen, dlen - plen, array, term);	if (!*numlen)		return -1;	return 1;}/* FIXME: This should be in userspace.  Later. */static int help(const struct iphdr *iph, size_t len,		struct ip_conntrack *ct,		enum ip_conntrack_info ctinfo){	/* tcplen not negative guaranteed by ip_conntrack_tcp.c */	struct tcphdr *tcph = (void *)iph + iph->ihl * 4;	const char *data = (const char *)tcph + tcph->doff * 4;	unsigned int tcplen = len - iph->ihl * 4;	unsigned int datalen = tcplen - tcph->doff * 4;	u_int32_t old_seq_aft_nl;	int old_seq_aft_nl_set;	u_int32_t array[6] = { 0 };	int dir = CTINFO2DIR(ctinfo);	unsigned int matchlen, matchoff;	struct ip_conntrack_tuple t, mask;	struct ip_ct_ftp *info = &ct->help.ct_ftp_info;	/* Until there's been traffic both ways, don't look in packets. */	if (ctinfo != IP_CT_ESTABLISHED	    && ctinfo != IP_CT_ESTABLISHED+IP_CT_IS_REPLY) {		DEBUGP("ftp: Conntrackinfo = %u\n", ctinfo);		return NF_ACCEPT;	}	/* Not whole TCP header? */	if (tcplen < sizeof(struct tcphdr) || tcplen < tcph->doff*4) {		DEBUGP("ftp: tcplen = %u\n", (unsigned)tcplen);		return NF_ACCEPT;	}	/* Checksum invalid?  Ignore. */	/* FIXME: Source route IP option packets --RR */	if (tcp_v4_check(tcph, tcplen, iph->saddr, iph->daddr,			 csum_partial((char *)tcph, tcplen, 0))) {		DEBUGP("ftp_help: bad csum: %p %u %u.%u.%u.%u %u.%u.%u.%u\n",		       tcph, tcplen, NIPQUAD(iph->saddr),		       NIPQUAD(iph->daddr));		return NF_ACCEPT;	}	LOCK_BH(&ip_ftp_lock);	old_seq_aft_nl_set = info->seq_aft_nl_set[dir];	old_seq_aft_nl = info->seq_aft_nl[dir];	DEBUGP("conntrack_ftp: datalen %u\n", datalen);	if ((datalen > 0) && (data[datalen-1] == '\n')) {		DEBUGP("conntrack_ftp: datalen %u ends in \\n\n", datalen);		if (!old_seq_aft_nl_set		    || after(ntohl(tcph->seq) + datalen, old_seq_aft_nl)) {			DEBUGP("conntrack_ftp: updating nl to %u\n",			       ntohl(tcph->seq) + datalen);			info->seq_aft_nl[dir] = ntohl(tcph->seq) + datalen;			info->seq_aft_nl_set[dir] = 1;		}	}	UNLOCK_BH(&ip_ftp_lock);	if(!old_seq_aft_nl_set ||			(ntohl(tcph->seq) != old_seq_aft_nl)) {		DEBUGP("ip_conntrack_ftp_help: wrong seq pos %s(%u)\n",		       old_seq_aft_nl_set ? "":"(UNSET) ", old_seq_aft_nl);		return NF_ACCEPT;	}	switch (find_pattern(data, datalen,			     search[dir].pattern,			     search[dir].plen, search[dir].term,			     &matchoff, &matchlen,			     array)) {	case -1: /* partial */		/* We don't usually drop packets.  After all, this is		   connection tracking, not packet filtering.		   However, it is neccessary for accurate tracking in		   this case. */		if (net_ratelimit())			printk("conntrack_ftp: partial %u+%u\n",			       ntohl(tcph->seq), datalen);		return NF_DROP;	case 0: /* no match */		DEBUGP("ip_conntrack_ftp_help: no match\n");		return NF_ACCEPT;	}	DEBUGP("conntrack_ftp: match `%.*s' (%u bytes at %u)\n",	       (int)matchlen, data + matchoff,	       matchlen, ntohl(tcph->seq) + matchoff);	/* Update the ftp info */	LOCK_BH(&ip_ftp_lock);	if (htonl((array[0] << 24) | (array[1] << 16) | (array[2] << 8) | array[3])	    == ct->tuplehash[dir].tuple.src.ip) {		info->is_ftp = 1;		info->seq = ntohl(tcph->seq) + matchoff;		info->len = matchlen;		info->ftptype = dir;		info->port = array[4] << 8 | array[5];	} else {		/* Enrico Scholz's passive FTP to partially RNAT'd ftp		   server: it really wants us to connect to a		   different IP address.  Simply don't record it for		   NAT. */		DEBUGP("conntrack_ftp: NOT RECORDING: %u,%u,%u,%u != %u.%u.%u.%u\n",		       array[0], array[1], array[2], array[3],		       NIPQUAD(ct->tuplehash[dir].tuple.src.ip));	}	t = ((struct ip_conntrack_tuple)		{ { ct->tuplehash[!dir].tuple.src.ip,		    { 0 } },		  { htonl((array[0] << 24) | (array[1] << 16)			  | (array[2] << 8) | array[3]),		    { htons(array[4] << 8 | array[5]) },		    IPPROTO_TCP }});	mask = ((struct ip_conntrack_tuple)		{ { 0xFFFFFFFF, { 0 } },		  { 0xFFFFFFFF, { 0xFFFF }, 0xFFFF }});	/* Ignore failure; should only happen with NAT */	ip_conntrack_expect_related(ct, &t, &mask, NULL);	UNLOCK_BH(&ip_ftp_lock);	return NF_ACCEPT;}static struct ip_conntrack_helper ftp = { { NULL, NULL },					  { { 0, { __constant_htons(21) } },					    { 0, { 0 }, IPPROTO_TCP } },					  { { 0, { 0xFFFF } },					    { 0, { 0 }, 0xFFFF } },					  help };static int __init init(void){	return ip_conntrack_helper_register(&ftp);}static void __exit fini(void){	ip_conntrack_helper_unregister(&ftp);}EXPORT_SYMBOL(ip_ftp_lock);EXPORT_SYMBOL(ip_conntrack_ftp);module_init(init);module_exit(fini);

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -