mod_tcp_chaff.c 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177
  1. /*
  2. * mod_tcp_chaff.c
  3. *
  4. * Copyright (c) 2001 Dug Song <dugsong@monkey.org>
  5. *
  6. * $Id: mod_tcp_chaff.c 2000 2008-04-27 06:17:35Z aturner $
  7. */
  8. #include "config.h"
  9. #include <stdio.h>
  10. #include <stdlib.h>
  11. #include <string.h>
  12. #include "pkt.h"
  13. #include "mod.h"
  14. #include "randutil.h"
  15. #define CHAFF_TYPE_CKSUM 1
  16. #define CHAFF_TYPE_NULL 2
  17. #define CHAFF_TYPE_PAWS 3
  18. #define CHAFF_TYPE_REXMIT 4
  19. #define CHAFF_TYPE_SEQ 5
  20. #define CHAFF_TYPE_SYN 6
  21. #define CHAFF_TYPE_TTL 7
  22. struct tcp_chaff_data {
  23. rand_t *rnd;
  24. int type;
  25. int ttl;
  26. };
  27. void *
  28. tcp_chaff_close(void *d)
  29. {
  30. struct tcp_chaff_data *data = (struct tcp_chaff_data *)d;
  31. if (data != NULL) {
  32. rand_close(data->rnd);
  33. free(data);
  34. }
  35. return (NULL);
  36. }
  37. void *
  38. tcp_chaff_open(int argc, char *argv[])
  39. {
  40. struct tcp_chaff_data *data;
  41. if (argc < 2)
  42. return (NULL);
  43. if ((data = calloc(1, sizeof(*data))) == NULL)
  44. return (NULL);
  45. data->rnd = rand_open();
  46. if (strcasecmp(argv[1], "cksum") == 0)
  47. data->type = CHAFF_TYPE_CKSUM;
  48. else if (strcasecmp(argv[1], "null") == 0)
  49. data->type = CHAFF_TYPE_NULL;
  50. else if (strcasecmp(argv[1], "paws") == 0)
  51. data->type = CHAFF_TYPE_PAWS;
  52. else if (strcasecmp(argv[1], "rexmit") == 0)
  53. data->type = CHAFF_TYPE_REXMIT;
  54. else if (strcasecmp(argv[1], "seq") == 0)
  55. data->type = CHAFF_TYPE_SEQ;
  56. else if (strcasecmp(argv[1], "syn") == 0)
  57. data->type = CHAFF_TYPE_SYN;
  58. else if ((data->ttl = atoi(argv[1])) > 0 && data->ttl < 256)
  59. data->type = CHAFF_TYPE_TTL;
  60. else
  61. return (tcp_chaff_close(data));
  62. return (data);
  63. }
  64. int
  65. tcp_chaff_apply(void *d, struct pktq *pktq)
  66. {
  67. struct tcp_chaff_data *data = (struct tcp_chaff_data *)d;
  68. struct pkt *pkt, *new, *next;
  69. struct tcp_opt opt;
  70. int i;
  71. for (pkt = TAILQ_FIRST(pktq); pkt != TAILQ_END(pktq); pkt = next) {
  72. next = TAILQ_NEXT(pkt, pkt_next);
  73. if (pkt->pkt_ip == NULL || pkt->pkt_ip->ip_p != IP_PROTO_TCP ||
  74. pkt->pkt_tcp == NULL || pkt->pkt_tcp_data == NULL ||
  75. (pkt->pkt_tcp->th_flags & TH_ACK) == 0)
  76. continue;
  77. new = pkt_dup(pkt);
  78. rand_strset(data->rnd, new->pkt_tcp_data, new->pkt_end -
  79. new->pkt_tcp_data + 1);
  80. switch (data->type) {
  81. case CHAFF_TYPE_CKSUM:
  82. ip_checksum(new->pkt_ip,
  83. new->pkt_ip_data - new->pkt_eth_data);
  84. new->pkt_tcp->th_sum = rand_uint16(data->rnd);
  85. break;
  86. case CHAFF_TYPE_NULL:
  87. new->pkt_tcp->th_flags = 0;
  88. ip_checksum(new->pkt_ip, new->pkt_end -
  89. new->pkt_eth_data);
  90. break;
  91. case CHAFF_TYPE_PAWS:
  92. /* Delete any existing TCP options. */
  93. i = (new->pkt_tcp->th_off << 2) - TCP_HDR_LEN;
  94. new->pkt_tcp->th_off = 5;
  95. new->pkt_end -= i;
  96. new->pkt_ip->ip_len = htons(new->pkt_end -
  97. new->pkt_eth_data);
  98. /* Insert initial timestamp, for PAWS elimination. */
  99. opt.opt_type = TCP_OPT_TIMESTAMP;
  100. opt.opt_len = TCP_OPT_LEN + 8;
  101. opt.opt_data.timestamp[0] = 0;
  102. opt.opt_data.timestamp[1] = 0;
  103. if ((i = ip_add_option(new->pkt_ip,
  104. PKT_BUF_LEN - ETH_HDR_LEN,
  105. IP_PROTO_TCP, &opt, opt.opt_len)) < 0) {
  106. pkt_free(new);
  107. continue;
  108. }
  109. new->pkt_end += i;
  110. ip_checksum(new->pkt_ip, new->pkt_end -
  111. new->pkt_eth_data);
  112. pkt_decorate(new);
  113. break;
  114. case CHAFF_TYPE_REXMIT:
  115. new->pkt_ts.tv_usec = 1;
  116. ip_checksum(new->pkt_ip, new->pkt_end -
  117. new->pkt_eth_data);
  118. break;
  119. case CHAFF_TYPE_SEQ:
  120. /* XXX - dunno recv window? */
  121. new->pkt_tcp->th_seq = htonl(666);
  122. new->pkt_tcp->th_ack = htonl(666);
  123. ip_checksum(new->pkt_ip, new->pkt_end -
  124. new->pkt_eth_data);
  125. break;
  126. case CHAFF_TYPE_SYN:
  127. new->pkt_tcp->th_flags = TH_SYN;
  128. new->pkt_tcp->th_seq = rand_uint32(data->rnd);
  129. new->pkt_tcp->th_ack = 0;
  130. new->pkt_end = new->pkt_tcp_data;
  131. new->pkt_tcp_data = NULL;
  132. new->pkt_ip->ip_len = htons(new->pkt_end -
  133. new->pkt_eth_data);
  134. ip_checksum(new->pkt_ip, new->pkt_end -
  135. new->pkt_eth_data);
  136. break;
  137. case CHAFF_TYPE_TTL:
  138. new->pkt_ip->ip_ttl = data->ttl;
  139. ip_checksum(new->pkt_ip, new->pkt_end -
  140. new->pkt_eth_data);
  141. break;
  142. }
  143. /* Minimal random reordering. */
  144. if ((new->pkt_tcp->th_sum & 1) == 0)
  145. TAILQ_INSERT_BEFORE(pkt, new, pkt_next);
  146. else
  147. TAILQ_INSERT_AFTER(pktq, pkt, new, pkt_next);
  148. }
  149. return (0);
  150. }
  151. struct mod mod_tcp_chaff = {
  152. "tcp_chaff", /* name */
  153. "tcp_chaff cksum|null|paws|rexmit|seq|syn|<ttl>", /* usage */
  154. tcp_chaff_open, /* open */
  155. tcp_chaff_apply, /* apply */
  156. tcp_chaff_close /* close */
  157. };