nmap.cpp 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  1. #include "nmap.h"
  2. #include "hloop.h"
  3. #include "hstring.h"
  4. #include "hsocket.h"
  5. #include "netinet.h"
  6. using namespace hv;
  7. #define MAX_RECVFROM_TIMEOUT 5000 // ms
  8. #define MAX_SENDTO_PERSOCKET 1024
  9. typedef struct nmap_ctx_s {
  10. Nmap* nmap;
  11. int send_cnt;
  12. int recv_cnt;
  13. int up_cnt;
  14. int idle_cnt;
  15. } nmap_ctx_t;
  16. static void on_idle(hidle_t* idle) {
  17. hloop_t* loop = hevent_loop(idle);
  18. nmap_ctx_t* ctx = (nmap_ctx_t*)hloop_userdata(loop);
  19. ctx->idle_cnt++;
  20. if (ctx->idle_cnt == 1) {
  21. // try again?
  22. }
  23. hloop_stop(loop);
  24. }
  25. static void on_timer(htimer_t* timer) {
  26. hloop_t* loop = hevent_loop(timer);
  27. hloop_stop(loop);
  28. }
  29. static void on_recvfrom(hio_t* io, void* buf, int readbytes) {
  30. //printd("on_recv fd=%d readbytes=%d\n", hio_fd(io), readbytes);
  31. /*
  32. char localaddrstr[SOCKADDR_STRLEN] = {0};
  33. char peeraddrstr[SOCKADDR_STRLEN] = {0};
  34. printd("[%s] <=> [%s]\n",
  35. SOCKADDR_STR(hio_localaddr(io), localaddrstr),
  36. SOCKADDR_STR(hio_peeraddr(io), peeraddrstr));
  37. */
  38. hloop_t* loop = hevent_loop(io);
  39. nmap_ctx_t* ctx = (nmap_ctx_t*)hloop_userdata(loop);
  40. if (++ctx->recv_cnt == ctx->send_cnt) {
  41. //hloop_stop(loop);
  42. }
  43. Nmap* nmap = ctx->nmap;
  44. struct sockaddr_in* peeraddr = (struct sockaddr_in*)hio_peeraddr(io);
  45. auto iter = nmap->find(peeraddr->sin_addr.s_addr);
  46. if (iter != nmap->end()) {
  47. if (iter->second == 0) {
  48. iter->second = 1;
  49. if (++ctx->up_cnt == nmap->size()) {
  50. hloop_stop(loop);
  51. }
  52. }
  53. }
  54. }
  55. int nmap_discover(Nmap* nmap) {
  56. hloop_t* loop = hloop_new(0);
  57. uint64_t start_hrtime = hloop_now_hrtime(loop);
  58. nmap_ctx_t ctx;
  59. ctx.nmap = nmap;
  60. ctx.send_cnt = 0;
  61. ctx.recv_cnt = 0;
  62. ctx.up_cnt = 0;
  63. ctx.idle_cnt = 0;
  64. hloop_set_userdata(loop, &ctx);
  65. char recvbuf[128];
  66. // icmp
  67. char sendbuf[44]; // 20IP + 44ICMP = 64
  68. icmp_t* icmp_req = (icmp_t*)sendbuf;
  69. icmp_req->icmp_type = ICMP_ECHO;
  70. icmp_req->icmp_code = 0;
  71. icmp_req->icmp_id = getpid();
  72. for (int i = 0; i < sizeof(sendbuf) - sizeof(icmphdr_t); ++i) {
  73. icmp_req->icmp_data[i] = i;
  74. }
  75. struct sockaddr_in peeraddr;
  76. hio_t* io = NULL;
  77. for (auto iter = nmap->begin(); iter != nmap->end(); ++iter) {
  78. if (iter->second == 1) continue;
  79. if (ctx.send_cnt % MAX_SENDTO_PERSOCKET == 0) {
  80. // socket
  81. int sockfd = socket(AF_INET, SOCK_RAW, IPPROTO_ICMP);
  82. if (sockfd < 0) {
  83. perror("socket");
  84. if (errno == EPERM) {
  85. fprintf(stderr, "please use root or sudo to create a raw socket.\n");
  86. }
  87. return -socket_errno();
  88. }
  89. nonblocking(sockfd);
  90. int len = 425984; // 416K
  91. socklen_t optlen = sizeof(len);
  92. setsockopt(sockfd, SOL_SOCKET, SO_SNDBUF, (const char*)&len, optlen);
  93. io = hio_get(loop, sockfd);
  94. if (io == NULL) return -1;
  95. hio_set_type(io, HIO_TYPE_IP);
  96. struct sockaddr_in localaddr;
  97. socklen_t addrlen = sizeof(localaddr);
  98. memset(&localaddr, 0, addrlen);
  99. localaddr.sin_addr.s_addr = htonl(INADDR_ANY);
  100. hio_set_localaddr(io, (struct sockaddr*)&localaddr, addrlen);
  101. hrecvfrom(loop, sockfd, recvbuf, sizeof(recvbuf), on_recvfrom);
  102. }
  103. icmp_req->icmp_seq = iter->first;
  104. icmp_req->icmp_cksum = 0;
  105. icmp_req->icmp_cksum = checksum((uint8_t*)icmp_req, sizeof(sendbuf));
  106. socklen_t addrlen = sizeof(peeraddr);
  107. memset(&peeraddr, 0, addrlen);
  108. peeraddr.sin_family = AF_INET;
  109. peeraddr.sin_addr.s_addr = iter->first;
  110. hio_set_peeraddr(io, (struct sockaddr*)&peeraddr, addrlen);
  111. hsendto(loop, hio_fd(io), sendbuf, sizeof(sendbuf), NULL);
  112. ++ctx.send_cnt;
  113. }
  114. htimer_add(loop, on_timer, MAX_RECVFROM_TIMEOUT, 1);
  115. hidle_add(loop, on_idle, 3);
  116. hloop_run(loop);
  117. uint64_t end_hrtime = hloop_now_hrtime(loop);
  118. hloop_free(&loop);
  119. // print result
  120. char ip[INET_ADDRSTRLEN];
  121. auto iter = nmap->begin();
  122. while (iter != nmap->end()) {
  123. inet_ntop(AF_INET, (void*)&iter->first, ip, sizeof(ip));
  124. printd("%s\t is %s.\n", ip, iter->second == 0 ? "down" : "up");
  125. ++iter;
  126. }
  127. printd("Nmap done: %lu IP addresses (%d hosts up) scanned in %.2f seconds\n",
  128. nmap->size(), ctx.up_cnt, (end_hrtime-start_hrtime)/1000000.0f);
  129. return ctx.up_cnt;
  130. }
  131. int segment_discover(const char* segment16, Nmap* nmap) {
  132. StringList strlist = split(segment16, '.');
  133. if (strlist.size() != 4) return -1;
  134. uint32_t addr = 0;
  135. uint8_t* p = (uint8_t*)&addr;
  136. p[0] = atoi(strlist[0].c_str());
  137. p[1] = atoi(strlist[1].c_str());
  138. p[3] = 1;
  139. printd("Nmap scan %u.%u.x.1...\n", p[0], p[1]);
  140. nmap->clear();
  141. for (int i = 0; i < 256; ++i) {
  142. p[2] = i;
  143. (*nmap)[addr] = 0;
  144. }
  145. return nmap_discover(nmap);
  146. }
  147. int host_discover(const char* segment24, Nmap* nmap) {
  148. StringList strlist = split(segment24, '.');
  149. if (strlist.size() != 4) return -1;
  150. uint32_t addr = 0;
  151. uint8_t* p = (uint8_t*)&addr;
  152. p[0] = atoi(strlist[0].c_str());
  153. p[1] = atoi(strlist[1].c_str());
  154. p[2] = atoi(strlist[2].c_str());
  155. printd("Nmap scan %u.%u.%u.x...\n", p[0], p[1], p[2]);
  156. // 0,255 reserved
  157. nmap->clear();
  158. for (int i = 1; i < 255; ++i) {
  159. p[3] = i;
  160. (*nmap)[addr] = 0;
  161. }
  162. return nmap_discover(nmap);
  163. }