nmap.cpp 5.5 KB

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