Created
April 7, 2011 09:17
-
-
Save guillaumerose/907400 to your computer and use it in GitHub Desktop.
Premier exercice
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
#include <stdio.h> | |
#include <stdlib.h> | |
#include <unistd.h> | |
#include <netinet/in.h> | |
#include <linux/types.h> | |
#include <linux/netfilter.h> /* for NF_ACCEPT */ | |
#include <linux/ip.h> | |
#include <linux/tcp.h> | |
#include <libnetfilter_queue/libnetfilter_queue.h> | |
int open = 0; | |
int look_for_tcp_flags(unsigned char* dgram, unsigned int datalen){ | |
struct iphdr * iphdrs = (struct iphdr *) dgram; | |
if (datalen < sizeof(struct iphdr) +sizeof(struct tcphdr)){ | |
return 0; | |
} | |
printf("open = %d\n", open); | |
printf("source = %s ", inet_ntoa(*((struct in_addr *)&iphdrs->saddr))); | |
printf("destination = %s ", inet_ntoa(*((struct in_addr *)&iphdrs->daddr))); | |
/* check IP version */ | |
if (iphdrs->version == 4){ | |
if ( iphdrs->protocol == IPPROTO_TCP){ | |
struct tcphdr * tcphdrs=(struct tcphdr *) (dgram+4*iphdrs->ihl); | |
printf("port source = %d ", ntohs(tcphdrs->source)); | |
printf("port dest = %d ", ntohs(tcphdrs->dest)); | |
if (ntohs(tcphdrs->dest) == 100) { | |
open = 1; | |
} | |
} | |
} | |
return 0; | |
} | |
/* returns packet id */ | |
static u_int32_t print_pkt (struct nfq_data *tb) | |
{ | |
int id = 0; | |
struct nfqnl_msg_packet_hdr *ph; | |
struct nfqnl_msg_packet_hw *hwph; | |
u_int32_t mark,ifi; | |
int ret; | |
unsigned char *data; | |
ph = nfq_get_msg_packet_hdr(tb); | |
if (ph) { | |
id = ntohl(ph->packet_id); | |
printf("hw_protocol=0x%04x hook=%u id=%u ", | |
ntohs(ph->hw_protocol), ph->hook, id); | |
} | |
hwph = nfq_get_packet_hw(tb); | |
if (hwph) { | |
int i, hlen = ntohs(hwph->hw_addrlen); | |
printf("hw_src_addr="); | |
for (i = 0; i < hlen-1; i++) | |
printf("%02x:", hwph->hw_addr[i]); | |
printf("%02x ", hwph->hw_addr[hlen-1]); | |
} | |
mark = nfq_get_nfmark(tb); | |
if (mark) | |
printf("mark=%u ", mark); | |
ifi = nfq_get_indev(tb); | |
if (ifi) | |
printf("indev=%u ", ifi); | |
ifi = nfq_get_outdev(tb); | |
if (ifi) | |
printf("outdev=%u ", ifi); | |
ifi = nfq_get_physindev(tb); | |
if (ifi) | |
printf("physindev=%u ", ifi); | |
ifi = nfq_get_physoutdev(tb); | |
if (ifi) | |
printf("physoutdev=%u ", ifi); | |
ret = nfq_get_payload(tb, &data); | |
if (ret >= 0) | |
printf("payload_len=%d ", ret); | |
look_for_tcp_flags((unsigned char *) data, ret); | |
fputc('\n', stdout); | |
return id; | |
} | |
static int cb(struct nfq_q_handle *qh, struct nfgenmsg *nfmsg, | |
struct nfq_data *nfa, void *data) | |
{ | |
u_int32_t id = print_pkt(nfa); | |
printf("entering callback\n"); | |
u_int32_t verdict; | |
if (open == 0) { | |
verdict = NF_DROP; | |
} | |
else { | |
verdict = NF_ACCEPT; | |
} | |
return nfq_set_verdict(qh, id, verdict, 0, NULL); | |
} | |
int main(int argc, char **argv) | |
{ | |
struct nfq_handle *h; | |
struct nfq_q_handle *qh; | |
struct nfnl_handle *nh; | |
int fd; | |
int rv; | |
char buf[4096] __attribute__ ((aligned)); | |
printf("opening library handle\n"); | |
h = nfq_open(); | |
if (!h) { | |
fprintf(stderr, "error during nfq_open()\n"); | |
exit(1); | |
} | |
printf("unbinding existing nf_queue handler for AF_INET (if any)\n"); | |
if (nfq_unbind_pf(h, AF_INET) < 0) { | |
fprintf(stderr, "error during nfq_unbind_pf()\n"); | |
exit(1); | |
} | |
printf("binding nfnetlink_queue as nf_queue handler for AF_INET\n"); | |
if (nfq_bind_pf(h, AF_INET) < 0) { | |
fprintf(stderr, "error during nfq_bind_pf()\n"); | |
exit(1); | |
} | |
printf("binding this socket to queue '0'\n"); | |
qh = nfq_create_queue(h, 0, &cb, NULL); | |
if (!qh) { | |
fprintf(stderr, "error during nfq_create_queue()\n"); | |
exit(1); | |
} | |
printf("setting copy_packet mode\n"); | |
if (nfq_set_mode(qh, NFQNL_COPY_PACKET, 0xffff) < 0) { | |
fprintf(stderr, "can't set packet_copy mode\n"); | |
exit(1); | |
} | |
fd = nfq_fd(h); | |
while ((rv = recv(fd, buf, sizeof(buf), 0)) && rv >= 0) { | |
printf("pkt received\n"); | |
nfq_handle_packet(h, buf, rv); | |
} | |
printf("unbinding from queue 0\n"); | |
nfq_destroy_queue(qh); | |
#ifdef INSANE | |
/* normally, applications SHOULD NOT issue this command, since | |
* it detaches other programs/sockets from AF_INET, too ! */ | |
printf("unbinding from AF_INET\n"); | |
nfq_unbind_pf(h, AF_INET); | |
#endif | |
printf("closing library handle\n"); | |
nfq_close(h); | |
exit(0); | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment