nmap.cpp 4.8 KB

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