address.cpp 2.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120
  1. #include <array>
  2. #include "address.h"
  3. using namespace std;
  4. namespace nmrpflash {
  5. namespace {
  6. unsigned ip6_netmask_to_prefix_len(const ip6_addr::bytes_type& a)
  7. {
  8. size_t i = 0;
  9. for (; i < a.size() && a[i] == 0xff; ++i) {
  10. ;
  11. }
  12. if (i < a.size()) {
  13. unsigned r = 0;
  14. for (uint8_t b = a[i]; b & 0x80; b <<= 1) {
  15. ++r;
  16. }
  17. do {
  18. if (((0xff << (8 - r)) & 0xff) != a[i]) {
  19. break;
  20. }
  21. auto k = i + 1;
  22. for (; k < a.size(); ++k) {
  23. if (a[k]) {
  24. break;
  25. }
  26. }
  27. if (k < a.size()) {
  28. break;
  29. }
  30. return i * 8 + r;
  31. } while (false);
  32. throw invalid_argument("discontignuous netmask");
  33. }
  34. return i * 8;
  35. }
  36. }
  37. ip_addr ip_from_sockaddr(gsl::not_null<const sockaddr*> addr)
  38. {
  39. if (addr->sa_family == AF_INET) {
  40. return ip4_from_sockaddr(reinterpret_cast<const sockaddr_in*>(addr.get()));
  41. } else if (addr->sa_family == AF_INET6) {
  42. return ip6_from_sockaddr(reinterpret_cast<const sockaddr_in6*>(addr.get()));
  43. } else {
  44. throw invalid_argument("unsupported address family");
  45. }
  46. }
  47. ip4_addr ip4_from_sockaddr(gsl::not_null<const sockaddr_in*> addr)
  48. {
  49. return ip4_addr(htonl(addr->sin_addr.s_addr));
  50. }
  51. ip6_addr ip6_from_sockaddr(gsl::not_null<const sockaddr_in6*> addr)
  52. {
  53. return ip6_addr(to_array(addr->sin6_addr.s6_addr));
  54. }
  55. ip_net::ip_net(const ip_addr& addr, const ip_addr& netmask)
  56. {
  57. if (addr.is_v4() && netmask.is_v4()) {
  58. m_net = ip4_net(addr.to_v4(), netmask.to_v4());
  59. } else if (addr.is_v6() && netmask.is_v6()) {
  60. m_net = ip6_net(addr.to_v6(), ip6_netmask_to_prefix_len(netmask.to_v6().to_bytes()));
  61. } else {
  62. throw invalid_argument("unsupported address and netmask combination");
  63. }
  64. }
  65. mac_addr::mac_addr(const string& addr)
  66. {
  67. do {
  68. vector<string> parts;
  69. boost::split(parts, addr, boost::algorithm::is_any_of(":-"));
  70. if (parts.size() != length) {
  71. break;
  72. }
  73. for (size_t i = 0; i < length; ++i) {
  74. if (parts[i].size() != 2) {
  75. break;
  76. }
  77. size_t k;
  78. int n = stoi(parts[i], &k, 16);
  79. if (k != 2 || n < 0 || n > 0xff) {
  80. break;
  81. }
  82. m_addr[i] = n & 0xff;
  83. }
  84. return;
  85. } while (false);
  86. throw invalid_argument("invalid MAC address: " + addr);
  87. }
  88. ostream& operator<<(ostream& os, const mac_addr& addr)
  89. {
  90. using boost::algorithm::join;
  91. using boost::adaptors::transformed;
  92. os << join(addr.m_addr | transformed([](uint8_t b) { return to_hex(b); }), ":");
  93. return os;
  94. }
  95. }