123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496 |
- /*
- * Forge Socket Banner Grab Copyright 2013 Regents of the University of Michigan
- *
- * Licensed under the Apache License, Version 2.0 (the "License"); you may not
- * use this file except in compliance with the License. You may obtain a copy
- * of the License at http://www.apache.org/licenses/LICENSE-2.0
- */
- #include <sys/socket.h>
- #include <netinet/in.h>
- #include <arpa/inet.h>
- #include "logger.h"
- #include <event.h>
- #include <event2/bufferevent_ssl.h>
- #include <getopt.h>
- #include <stdio.h>
- #include <stdlib.h>
- #include <string.h>
- #include <errno.h>
- #include <ulimit.h>
- #include "forge_socket.h"
- #define MAX_BANNER_LEN 1024
- #define BASE64_ALPHABET "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"
- struct config {
- int read_timeout; // how long to wait once connected for the banner (seconds)
- int current_running;
- int max_concurrent;
- struct event_base *base;
- struct bufferevent *stdin_bev;
- int stdin_closed;
- enum {FORMAT_HEX, FORMAT_BASE64, FORMAT_ASCII} format;
- char *send_str;
- long send_str_size;
- struct stats_st {
- int init_connected_hosts; // Number of hosts we have even tried to connect to
- int connected_hosts; // # hosts that picked up
- int conn_timed_out; // # hosts that timed out during connection
- int read_timed_out; // # hosts that connected, but sent no data (banner)
- int timed_out; // # hosts that timed out at all (conn_timed_out+read_timed_out)?
- int completed_hosts; // # hosts that presented a banner
- } stats;
- };
- struct state {
- struct config *conf;
- uint32_t src_ip;
- uint32_t dst_ip;
- uint16_t sport;
- uint16_t dport;
- uint32_t seq;
- uint32_t seq_ack;
- enum {CONNECTING, CONNECTED, RECEIVED} state;
- };
- void stdin_readcb(struct bufferevent *bev, void *arg);
- void print_status(evutil_socket_t fd, short events, void *arg)
- {
- struct event *ev;
- struct config *conf = arg;
- struct event_base *base = conf->base;
- struct timeval status_timeout = {1, 0};
- ev = evtimer_new(base, print_status, conf);
- evtimer_add(ev, &status_timeout);
- (void)fd; (void)events;
- log_info("forge-socket", "(%d/%d in use) - Totals: %d inited, %d connected, %d conn timeout, %d read timeout %d completed",
- conf->current_running, conf->max_concurrent,
- conf->stats.init_connected_hosts,
- conf->stats.connected_hosts, conf->stats.conn_timed_out,
- conf->stats.read_timed_out, conf->stats.completed_hosts);
- }
- void decrement_cur_running(struct state *st)
- {
- struct config *conf = st->conf;
- conf->current_running--;
- log_debug("forge-socket", "done, down to %d",
- conf->current_running);
- if (evbuffer_get_length(bufferevent_get_input(conf->stdin_bev)) > 0) {
- stdin_readcb(conf->stdin_bev, conf);
- }
- free(st);
- if (conf->stdin_closed && conf->current_running == 0) {
- // Done
- log_info("forge-socket", "done");
- print_status(0, 0, conf);
- exit(0);
- }
- }
- void event_cb(struct bufferevent *bev, short events, void *arg)
- {
- struct state *st = arg;
- struct config *conf = st->conf;
- struct in_addr addr;
- addr.s_addr = st->src_ip;
- if (events & BEV_EVENT_CONNECTED) {
- log_error("forge-socket", "%s connected - wat?", inet_ntoa(addr));
-
- } else {
- if (st->state == CONNECTED) {
- // Print out that we just didn't receive data
- printf("%s X\n", inet_ntoa(addr));
- fflush(stdout);
- conf->stats.read_timed_out++;
- } else {
- conf->stats.conn_timed_out++;
- }
- log_debug("forge-socket", "%s bailing..", inet_ntoa(addr));
- bufferevent_free(bev);
- conf->stats.timed_out++;
- decrement_cur_running(st);
- }
- }
- // Grab these bytes, and close the connection.
- // Even if we don't need to read any bytes,
- // we have to have this so that libevent thinks we have
- // a read event, so that it can timeout TCP connects
- // (as a read timeout)
- void read_cb(struct bufferevent *bev, void *arg)
- {
- struct evbuffer *in = bufferevent_get_input(bev);
- struct state *st = arg;
- size_t len = evbuffer_get_length(in);
- struct in_addr addr;
- addr.s_addr = st->src_ip;
- log_debug("forge-socket", "read_cb for %s", inet_ntoa(addr));
- if (len > MAX_BANNER_LEN) {
- len = MAX_BANNER_LEN;
- }
- if (len > 0) {
- // Grab the banner
- unsigned int i;
- unsigned char *buf = malloc(len+1);
- st->state = RECEIVED;
- if (!buf) {
- log_fatal("forge-socket", "cannot alloc %d byte buf", len+1);
- return;
- }
- evbuffer_remove(in, buf, len);
-
- printf("%s ", inet_ntoa(addr));
- if (st->conf->format == FORMAT_ASCII) {
- // Ascii
- buf[len] = '\0';
- printf("%s\n", buf);
- } else if (st->conf->format == FORMAT_HEX) {
- // Hex output
- for (i=0; i<len; i++) {
- printf("%02x", buf[i]);
- }
- printf("\n");
- } else if (st->conf->format == FORMAT_BASE64) {
- // Base64
- int i=0;
- char out[4] = {0,0,0,0};
- while (i < len) {
- uint32_t value = 0;
- value += (i < len) ? buf[i++] << 16 : 0;
- value += (i < len) ? buf[i++] << 8 : 0;
- value += (i < len) ? buf[i++] : 0;
- out[0] = BASE64_ALPHABET[(value >> 18) & 0x3F];
- out[1] = BASE64_ALPHABET[(value >> 12) & 0x3F];
- out[2] = BASE64_ALPHABET[(value >> 6) & 0x3F];
- out[3] = BASE64_ALPHABET[(value ) & 0x3F];
- if (i < len) {
- printf("%c%c%c%c", out[0], out[1], out[2], out[3]);
- }
- }
- if (len > 0) {
- switch (len % 3) {
- case 1:
- out[2] = '=';
- case 2:
- out[3] = '=';
- default:
- break;
- }
- printf("%c%c%c%c\n", out[0], out[1], out[2], out[3]);
- }
- }
- fflush(stdout);
- free(buf);
- st->conf->stats.completed_hosts++;
- }
- bufferevent_free(bev);
- decrement_cur_running(st);
- }
- int set_sock_state(int sock, struct tcp_state *st)
- {
- struct sockaddr_in sin;
- sin.sin_family = AF_INET;
- sin.sin_addr.s_addr = st->src_ip;
- sin.sin_port = st->sport;
- int value = 1;
- if (setsockopt(sock, SOL_SOCKET, SO_REUSEADDR, &value, sizeof(value)) < 0) {
- perror("setsockopt SO_REUSEADDR");
- return -1;
- }
- if (setsockopt(sock, SOL_IP, IP_TRANSPARENT, &value, sizeof(value)) < 0) {
- perror("setsockopt IP_TRANSPARENT");
- return -1;
- }
- if (bind(sock, (struct sockaddr *)&sin, sizeof(sin)) < 0) {
- perror("bind");
- return -1;
- }
- if (setsockopt(sock, IPPROTO_TCP, TCP_STATE, st, sizeof(struct tcp_state)) < 0) {
- perror("setsockopt TCP_STATE");
- return -1;
- }
- return 0;
- }
- void grab_banner(struct state *st)
- {
- struct sockaddr_in addr;
- struct bufferevent *bev;
- struct timeval read_to = {st->conf->read_timeout, 0};
- struct tcp_state tcp_st;
- int sock = socket(AF_INET, SOCK_FORGE, 0);
- addr.sin_addr.s_addr = st->src_ip;
- if (sock < 0) {
- perror("SOCK_FORGE socket");
- log_fatal("forge_socket", "(did you insmod forge_socket.ko?)");
- return;
- }
- memset(&tcp_st, 0, sizeof(tcp_st));
-
- // These need to be in network order for forge socket"
- tcp_st.src_ip = st->dst_ip;
- tcp_st.dst_ip = st->src_ip;
- tcp_st.sport = htons(st->dport);
- tcp_st.dport = htons(st->sport);
- // This should be in ???
- tcp_st.seq = st->seq_ack;
- tcp_st.ack = (st->seq + 1);
- tcp_st.snd_wnd = 0x1000;
- tcp_st.rcv_wnd = 0x1000;
-
- tcp_st.snd_una = tcp_st.seq;
- st->state = CONNECTING;
- st->conf->stats.init_connected_hosts++;
- // consider this a non-blocking, but completed "connect()". heh.
- if (set_sock_state(sock, &tcp_st) != 0) {
- log_error("forge_socket", "set_sock_state failed\n");
- decrement_cur_running(st);
- return;
- }
- evutil_make_socket_nonblocking(sock);
- bev = bufferevent_socket_new(st->conf->base, sock, BEV_OPT_CLOSE_ON_FREE);
- bufferevent_set_timeouts(bev, &read_to, &read_to);
- bufferevent_setcb(bev, read_cb, NULL, event_cb, st);
- bufferevent_enable(bev, EV_READ);
- // Send data
- if (st->conf->send_str) {
- struct evbuffer *evout = bufferevent_get_output(bev);
- // HACK!!! TODO: make some messy parser that replaces ${IP} with IP etc
- // and allow null characters
- evbuffer_add_printf(evout, st->conf->send_str,
- inet_ntoa(addr.sin_addr), inet_ntoa(addr.sin_addr),
- inet_ntoa(addr.sin_addr), inet_ntoa(addr.sin_addr));
- log_trace("forge-socket", "sent str to %s", inet_ntoa(addr.sin_addr));
- }
- // Update state/stats
- st->state = CONNECTED;
- st->conf->stats.connected_hosts++;
- log_trace("forge-socket", "go %s go! read a byte!!", inet_ntoa(addr.sin_addr));
- }
- void stdin_eventcb(struct bufferevent *bev, short events, void *ptr) {
- struct config *conf = ptr;
- if (events & BEV_EVENT_EOF) {
- log_debug("forge-socket",
- "received EOF; quitting after buffer empties");
- conf->stdin_closed = 1;
- if (conf->current_running == 0) {
- log_info("forge-socket", "done");
- print_status(0, 0, conf);
- exit(0);
- }
- }
- }
- void stdin_readcb(struct bufferevent *bev, void *arg)
- {
- struct evbuffer *in = bufferevent_get_input(bev);
- struct config *conf = arg;
- log_debug("forge-socket", "stdin cb %d < %d ?",
- conf->current_running, conf->max_concurrent);
- while (conf->current_running < conf->max_concurrent &&
- evbuffer_get_length(in) > 0) {
- size_t line_len;
- char *line = evbuffer_readln(in, &line_len, EVBUFFER_EOL_LF);
- struct state *st;
- if (!line)
- break;
- log_debug("forge-socket", "line: '%s'", line);
- //synack, 77.176.116.205, 141.212.121.125, 443, 49588, 3628826326, 3441755636, 0, 0,2013-08-11 19:16:05.799
- char synack[12];
- char srcip[INET_ADDRSTRLEN], dstip[INET_ADDRSTRLEN];
- uint32_t seq, seq_ack;
- uint16_t sport, dport;
- int cooldown, repeat=1;
- int ret = sscanf(line, "%11[^,], %15[^,], %15[^,], %hu, %hu, %u, %u, %d, %d,%*s",
- synack, srcip, dstip, &sport, &dport, &seq, &seq_ack, &cooldown, &repeat);
- log_trace("forge-socket", "%d '%s' sip: '%s', dip: '%s', sport: %d, dport: %d, seq: %d, seq_ack: %d",
- ret, synack, srcip, dstip, sport, dport, seq, seq_ack);
- if (ret==9 && !repeat && strcmp(synack, "synack") == 0) {
- st = malloc(sizeof(*st));
- st->conf = conf;
- st->src_ip = inet_addr(srcip);
- st->dst_ip = inet_addr(dstip);
- st->sport = sport;
- st->dport = dport;
- st->seq = seq;
- st->seq_ack = seq_ack;
-
- conf->current_running++;
- grab_banner(st);
- }
- }
- }
- int main(int argc, char *argv[])
- {
- struct event_base *base;
- struct event *status_timer;
- struct timeval status_timeout = {1, 0};
- int c;
- struct option long_options[] = {
- {"concurrent", required_argument, 0, 'c'},
- {"read-timeout", required_argument, 0, 'r'},
- {"verbosity", required_argument, 0, 'v'},
- {"format", no_argument, 0, 'f'},
- {"data", required_argument, 0, 'd'},
- {0, 0, 0, 0} };
- struct config conf;
- int ret;
- FILE *fp;
- log_init(stderr, LOG_INFO, 1, "forge-socket");
- ret = ulimit(4, 1000000); // Allow us to open 1 million fds (instead of 1024)
- if (ret < 0) {
- log_fatal("forge-socket", "cannot set ulimit");
- perror("ulimit");
- exit(1);
- }
- base = event_base_new();
- conf.base = base;
- // buffer stdin as an event
- conf.stdin_bev = bufferevent_socket_new(base, 0, BEV_OPT_DEFER_CALLBACKS);
- bufferevent_setcb(conf.stdin_bev, stdin_readcb, NULL, stdin_eventcb, &conf);
- bufferevent_enable(conf.stdin_bev, EV_READ);
- // Status timer
- status_timer = evtimer_new(base, print_status, &conf);
- evtimer_add(status_timer, &status_timeout);
- // Defaults
- conf.max_concurrent = 1;
- conf.current_running = 0;
- memset(&conf.stats, 0, sizeof(conf.stats));
- conf.read_timeout = 4;
- conf.stdin_closed = 0;
- conf.format = FORMAT_BASE64;
- conf.send_str = NULL;
- // Parse command line args
- while (1) {
- int option_index = 0;
- c = getopt_long(argc, argv, "c:t:r:v:f:d:",
- long_options, &option_index);
- if (c < 0) {
- break;
- }
- switch (c) {
- case 'c':
- conf.max_concurrent = atoi(optarg);
- break;
- case 'r':
- conf.read_timeout = atoi(optarg);
- break;
- case 'v':
- if (atoi(optarg) >= 0 && atoi(optarg) <= 5) {
- log_init(stderr, atoi(optarg), 1, "forge-socket");
- }
- break;
- case 'f':
- if (strcmp(optarg, "hex") == 0) {
- conf.format = FORMAT_HEX;
- } else if (strcmp(optarg, "base64") == 0) {
- conf.format = FORMAT_BASE64;
- } else if (strcmp(optarg, "ascii") == 0) {
- conf.format = FORMAT_ASCII;
- } else {
- log_fatal("forge-socket", "Unknown format '%s'; use 'hex', 'base64', or 'ascii'",
- optarg);
- }
- break;
- case 'd':
- fp = fopen(optarg, "r");
- if (!fp) {
- log_error("forge-socket", "Could not open send data file '%s':", optarg);
- perror("fopen");
- exit(-1);
- }
- fseek(fp, 0L, SEEK_END);
- conf.send_str_size = ftell(fp);
- fseek(fp, 0L, SEEK_SET);
- //assert(conf.send_str_size < 10000); // jumbo frames?
- conf.send_str = malloc(conf.send_str_size+1);
- if (!conf.send_str) {
- log_fatal("forge-socket", "Could not malloc %d bytes", conf.send_str_size+1);
- }
- if (fread(conf.send_str, conf.send_str_size, 1, fp) != 1) {
- log_fatal("forge-socket", "Couldn't read from send data file '%s':", optarg);
- }
- conf.send_str[conf.send_str_size] = '\0';
- fclose(fp);
- break;
- case '?':
- printf("Usage:\n");
- printf("\t%s [-c max_concurrency] [-r read_timeout] \n\t"
- "[-v verbosity=0-5] [-d send_data_file] [-f ascii|hex|base64]\n", argv[0]);
- exit(1);
- default:
- log_info("forge-socket", "hmmm..");
- break;
- }
- }
-
- log_info("forge-socket", "Using max_concurrency %d, %d s read timeout",
- conf.max_concurrent, conf.read_timeout);
- event_base_dispatch(base);
- return 0;
- }
|