ointerface.cpp 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642
  1. /*====================================================================*
  2. *
  3. * Copyright (c) 2013 Qualcomm Atheros, Inc.
  4. *
  5. * All rights reserved.
  6. *
  7. *====================================================================*/
  8. /*====================================================================*
  9. *
  10. * ointerface.cpp - ointerface class definition;
  11. *
  12. * host interface information;
  13. *
  14. * Contributor(s):
  15. * Charles Maier <charles.maier@intellon.com>
  16. *
  17. *--------------------------------------------------------------------*/
  18. #ifndef oINTERFACE_SOURCE
  19. #define oINTERFACE_SOURCE
  20. /*====================================================================*
  21. * system header files;
  22. *--------------------------------------------------------------------*/
  23. #include <iostream>
  24. #include <cstring>
  25. #include <cstdlib>
  26. #include <cerrno>
  27. /*====================================================================*
  28. * system header files (environment dependent);
  29. *--------------------------------------------------------------------*/
  30. #if defined (__linux__)
  31. # include <ifaddrs.h>
  32. # include <sys/ioctl.h>
  33. # include <sys/socket.h>
  34. # include <netinet/in.h>
  35. # include <linux/if_packet.h>
  36. #elif defined (__APPLE__)
  37. # include <sys/types.h>
  38. # include <ifaddrs.h>
  39. # include <net/if_dl.h>
  40. # include <net/if_types.h>
  41. # include <fcntl.h>
  42. # include <sys/ioctl.h>
  43. # include <net/bpf.h>
  44. #elif defined (__OpenBSD__)
  45. # include <ifaddrs.h>
  46. # include <net/if_dl.h>
  47. # include <net/if_types.h>
  48. #elif defined (WINPCAP)
  49. # include <pcap.h>
  50. # include <Packet32.h>
  51. # include <ntddndis.h>
  52. #else
  53. #error "Unknown environment."
  54. #endif
  55. /*====================================================================*
  56. * custom header files;
  57. *--------------------------------------------------------------------*/
  58. #include "../classes/ointerface.hpp"
  59. #include "../classes/omemory.hpp"
  60. #include "../classes/oerror.hpp"
  61. /*====================================================================*
  62. *
  63. * unsigned Index (void) const;
  64. *
  65. * return the index of the selected interface; on some systems, the
  66. * index is sequential; on others, it is not;
  67. *
  68. *--------------------------------------------------------------------*/
  69. unsigned ointerface::Index (void) const
  70. {
  71. return (this->mifindex);
  72. }
  73. /*====================================================================*
  74. *
  75. * char const * Name (void) const;
  76. *
  77. * return the NUL terminated interface name string;
  78. *
  79. *--------------------------------------------------------------------*/
  80. char const * ointerface::Name (void) const
  81. {
  82. return (this->mifname);
  83. }
  84. /*====================================================================*
  85. *
  86. * char const * Description (void) const;
  87. *
  88. * return the NUL terminated interface text string; on some systems
  89. * this may be identical to the name string; on other it may not be;
  90. *
  91. *--------------------------------------------------------------------*/
  92. char const * ointerface::Description (void) const
  93. {
  94. return (this->miftext);
  95. }
  96. /*====================================================================*
  97. *
  98. * byte const * HardwareAddress (void) const;
  99. *
  100. * return the NUL terminated internet address string; the string is
  101. * a sequence of hex octets seperated by colons;
  102. *
  103. *--------------------------------------------------------------------*/
  104. byte const * ointerface::HardwareAddress (void) const
  105. {
  106. return (this->mhwaddr);
  107. }
  108. /*====================================================================*
  109. *
  110. * byte const * InternetAddress (void) const;
  111. *
  112. * return the NUL terminated hardware address string; the string is
  113. * a sequence of decimal octets seperated by periods;
  114. *
  115. *--------------------------------------------------------------------*/
  116. byte const * ointerface::InternetAddress (void) const
  117. {
  118. return (this->mipaddr);
  119. }
  120. /*====================================================================*
  121. *
  122. * char const * HardwareAddressString (void) const;
  123. *
  124. * return the NUL terminated internet address string; the string is
  125. * a sequence of hex octets seperated by colons;
  126. *
  127. *--------------------------------------------------------------------*/
  128. char const * ointerface::HardwareAddressString (void) const
  129. {
  130. return (this->mhwstring);
  131. }
  132. /*====================================================================*
  133. *
  134. * char const * InternetAddressString (void) const;
  135. *
  136. * return the NUL terminated hardware address string; the string is
  137. * a sequence of decimal octets seperated by periods;
  138. *
  139. *--------------------------------------------------------------------*/
  140. char const * ointerface::InternetAddressString (void) const
  141. {
  142. return (this->mipstring);
  143. }
  144. /*====================================================================*
  145. *
  146. * ointerface & ExportHardwareAddress (void * memory);
  147. *
  148. * copy the stored hardware address to external memory; this is a
  149. * binary copy operation; return the object instance reference;
  150. *
  151. *--------------------------------------------------------------------*/
  152. ointerface & ointerface::ExportHardwareAddress (void * memory)
  153. {
  154. std::memcpy (memory, this->mhwaddr, sizeof (this->mhwaddr));
  155. return (*this);
  156. }
  157. /*====================================================================*
  158. *
  159. * ointerface & ExportInternetAddress (void * memory);
  160. *
  161. * copy the stored internet address to external memory; this is a
  162. * binary copy operation; return the object instance reference;
  163. *
  164. *--------------------------------------------------------------------*/
  165. ointerface & ointerface::ExportInternetAddress (void * memory)
  166. {
  167. std::memcpy (memory, this->mipaddr, sizeof (this->mipaddr));
  168. return (*this);
  169. }
  170. /*====================================================================*
  171. *
  172. * ointerface & ointerface::Description (char const * string);
  173. *
  174. * replace interface text string and return the object instance
  175. * reference; this string may change whenever the index or name
  176. * are changed but changing the text string does not affect the
  177. * other properties;
  178. *
  179. *--------------------------------------------------------------------*/
  180. ointerface & ointerface::Description (char const * string)
  181. {
  182. omemory::memtext (string, this->miftext, sizeof (this->miftext));
  183. return (*this);
  184. }
  185. /*====================================================================*
  186. *
  187. * ointerface & Print ();
  188. *
  189. * print interface index, hardware address, internet address, name
  190. * and text on stdout; return the object instance reference;
  191. *
  192. *--------------------------------------------------------------------*/
  193. ointerface & ointerface::Print ()
  194. {
  195. std::cout << this->Index () << " ";
  196. std::cout << this->HardwareAddressString () << " ";
  197. std::cout << this->InternetAddressString () << " ";
  198. std::cout << this->Name () << " ";
  199. std::cout << this->Description () << std::endl;
  200. return (*this);
  201. }
  202. /*====================================================================*
  203. *
  204. * bool Disabled () const;
  205. *
  206. * return true if the hardware address is all zeros or all ones;
  207. *
  208. *--------------------------------------------------------------------*/
  209. bool ointerface::Disabled () const
  210. {
  211. byte sample [ETHER_ADDR_LEN];
  212. std::memset (sample, 0x00, sizeof (sample));
  213. if (!std::memcmp (sample, this->mhwaddr, sizeof (sample)))
  214. {
  215. return (true);
  216. }
  217. std::memset (sample, 0xFF, sizeof (sample));
  218. if (!std::memcmp (sample, this->mhwaddr, sizeof (sample)))
  219. {
  220. return (true);
  221. }
  222. return (false);
  223. }
  224. /*====================================================================*
  225. *
  226. * ointerface & ointerface::lookup ();
  227. *
  228. * lookup hardware and internet addresses once the interface index
  229. * and name are known;
  230. *
  231. * two methods are provided for Linux; the first one encountered by
  232. * the compiler will be used; some Linux systems may not support the
  233. * getifaddrs function or may not implement it consistently;
  234. *
  235. *--------------------------------------------------------------------*/
  236. ointerface & ointerface::lookup ()
  237. {
  238. #if defined (__linux__) && 0
  239. /*
  240. * The preferred method for Linux at this time;
  241. */
  242. int fd;
  243. if ((fd = std::socket (AF_INET, SOCK_DGRAM, 0)) != -1)
  244. {
  245. struct ifreq ifreq;
  246. std::memcpy (ifreq.ifr_name, this->mifname, sizeof (ifreq.ifr_name));
  247. if (ioctl (fd, SIOCGIFHWADDR, &ifreq) != -1)
  248. {
  249. std::memcpy (this->mhwaddr, ifreq.ifr_ifru.ifru_hwaddr.sa_data, sizeof (this->mhwaddr));
  250. }
  251. if (ioctl (fd, SIOCGIFADDR, &ifreq) != -1)
  252. {
  253. struct sockaddr_in * sockaddr_in = (struct sockaddr_in *)(&ifreq.ifr_ifru.ifru_addr);
  254. std::memcpy (this->mipaddr, &sockaddr_in->sin_addr.s_addr, sizeof (this->mipaddr));
  255. }
  256. std::close (fd);
  257. }
  258. #elif defined (__linux__) || defined (__APPLE__) || defined (__OpenBSD__)
  259. struct ifaddrs * ifaddrs;
  260. if (getifaddrs (&ifaddrs) != -1)
  261. {
  262. struct ifaddrs * ifaddr;
  263. for (ifaddr = ifaddrs; ifaddr; ifaddr = ifaddr->ifa_next)
  264. {
  265. if (std::strcmp (this->mifname, ifaddr->ifa_name))
  266. {
  267. continue;
  268. }
  269. if (!ifaddr->ifa_addr)
  270. {
  271. continue;
  272. }
  273. if (ifaddr->ifa_addr->sa_family == AF_INET)
  274. {
  275. struct sockaddr_in * sockaddr_in = (struct sockaddr_in *) (ifaddr->ifa_addr);
  276. struct in_addr * in_addr = (struct in_addr *)(&sockaddr_in->sin_addr);
  277. std::memcpy (this->mipaddr, &in_addr->s_addr, sizeof (this->mipaddr));
  278. }
  279. #if defined (__linux__)
  280. if (ifaddr->ifa_addr->sa_family == AF_PACKET)
  281. {
  282. struct sockaddr_ll * sockaddr_ll = (struct sockaddr_ll *) (ifaddr->ifa_addr);
  283. std::memcpy (this->mhwaddr, sockaddr_ll->sll_addr, sizeof (this->mhwaddr));
  284. }
  285. #elif defined (__APPLE__) || defined (__OpenBSD__)
  286. if (ifaddr->ifa_addr->sa_family == AF_LINK)
  287. {
  288. struct sockaddr_dl * sockaddr_dl = (struct sockaddr_dl *) (ifaddr->ifa_addr);
  289. if (sockaddr_dl->sdl_type == IFT_ETHER)
  290. {
  291. std::memcpy (this->mhwaddr, LLADDR (sockaddr_dl), sizeof (this->mhwaddr));
  292. }
  293. }
  294. #else
  295. #error "Abandon all hope!"
  296. #endif
  297. }
  298. freeifaddrs (ifaddrs);
  299. }
  300. #elif defined (WINPCAP)
  301. ointerface::pcap_gethwaddr ();
  302. ointerface::pcap_getipaddr ();
  303. #else
  304. #error "Unknown environment."
  305. #endif
  306. omemory::hexdecode (this->mhwaddr, sizeof (this->mhwaddr), this->mhwstring, sizeof (this->mhwstring));
  307. omemory::decdecode (this->mipaddr, sizeof (this->mipaddr), this->mipstring, sizeof (this->mipstring));
  308. return (* this);
  309. }
  310. /*====================================================================*
  311. *
  312. * unsigned pcap_nametoindex (char const * name) const;
  313. *
  314. * WinPcap version of POSIX if_nametoindex function; return error
  315. * in non-pcap environments; this method is a temporary solution;
  316. *
  317. * Microsoft plans to support the if_nametoindex function on Vista
  318. * and future releases;
  319. *
  320. * see The Open Group Base Specifications Issue 6 IEEE Std 1003.1,
  321. * 2004 Edition for a description of this method;
  322. *
  323. *--------------------------------------------------------------------*/
  324. unsigned ointerface::pcap_nametoindex (char const * name) const
  325. {
  326. #if defined (WINPCAP)
  327. char buffer [PCAP_ERRBUF_SIZE];
  328. pcap_if_t * devices = (pcap_if_t *)(0);
  329. pcap_if_t * device;
  330. if (pcap_findalldevs (&devices, buffer) != -1)
  331. {
  332. unsigned index = 1;
  333. for (device = devices; device; device = device->next)
  334. {
  335. if (std::strcmp (name, device->name))
  336. {
  337. index++;
  338. continue;
  339. }
  340. pcap_freealldevs (devices);
  341. return (index);
  342. }
  343. pcap_freealldevs (devices);
  344. }
  345. #endif
  346. errno = ENXIO;
  347. return (0);
  348. }
  349. /*====================================================================*
  350. *
  351. * char * pcap_indextoname (unsigned ifindex, char * ifname) const;
  352. *
  353. * WinPcap version of POSIX if_indextoname function; return error
  354. * in non-pcap enviroements; this method is a temporary solution;
  355. *
  356. * Microsoft plans to support the if_indextoname function on Vista
  357. * and future releases;
  358. *
  359. * see The Open Group Base Specifications Issue 6 IEEE Std 1003.1,
  360. * 2004 Edition for a description of this method;
  361. *
  362. *--------------------------------------------------------------------*/
  363. char * ointerface::pcap_indextoname (unsigned ifindex, char * ifname) const
  364. {
  365. #if defined (WINPCAP)
  366. char buffer [PCAP_ERRBUF_SIZE];
  367. pcap_if_t * devices = (pcap_if_t *)(0);
  368. pcap_if_t * device;
  369. if ((ifindex--) && (pcap_findalldevs (&devices, buffer) != -1))
  370. {
  371. for (device = devices; device; device = device->next)
  372. {
  373. if (!ifindex--)
  374. {
  375. omemory::memtext (device->name, ifname, IF_NAMESIZE);
  376. pcap_freealldevs (devices);
  377. return (ifname);
  378. }
  379. }
  380. pcap_freealldevs (devices);
  381. }
  382. #endif
  383. errno = ENXIO;
  384. return ((char *)(0));
  385. }
  386. /*====================================================================*
  387. *
  388. * void pcap_gethwaddr ();
  389. *
  390. * update the hardware address based on interface name in winpcap
  391. * environment only; it has no effect in non-pcap environments;
  392. *
  393. * this method is a temporary solution that may become permanent
  394. * since there is no POSIX function defined for this;
  395. *
  396. *--------------------------------------------------------------------*/
  397. void ointerface::pcap_gethwaddr ()
  398. {
  399. #if defined (WINPCAP)
  400. LPADAPTER adapter = PacketOpenAdapter ((PCHAR)(this->mifname));
  401. PPACKET_OID_DATA data = (PPACKET_OID_DATA)(std::malloc (ETHER_ADDR_LEN + sizeof (PACKET_OID_DATA)));
  402. if (!data)
  403. {
  404. oerror::error (1, 0, "Can't allocate packet: %s", this->mifname);
  405. }
  406. data->Oid = OID_802_3_CURRENT_ADDRESS;
  407. data->Length = ETHER_ADDR_LEN;
  408. if ((!adapter) || (adapter->hFile == INVALID_HANDLE_VALUE))
  409. {
  410. oerror::error (1, 0, "Can't access interface: %s", this->mifname);
  411. }
  412. std::memset (this->mhwaddr, 0, ETHER_ADDR_LEN);
  413. if (PacketRequest (adapter, FALSE, data))
  414. {
  415. std::memcpy (this->mhwaddr, data->Data, data->Length);
  416. }
  417. PacketCloseAdapter (adapter);
  418. std::free (data);
  419. #endif
  420. return;
  421. }
  422. /*====================================================================*
  423. *
  424. * void pcap_getipaddr ();
  425. *
  426. * update the internet address based on interface name in winpcap
  427. * environment only; it has no effect in non-pcap environments;
  428. *
  429. *--------------------------------------------------------------------*/
  430. void ointerface::pcap_getipaddr ()
  431. {
  432. #if defined (WINPCAP)
  433. char buffer [PCAP_ERRBUF_SIZE];
  434. pcap_if_t * devices = (pcap_if_t *)(0);
  435. pcap_if_t * device;
  436. if (pcap_findalldevs (&devices, buffer) == -1)
  437. {
  438. oerror::error (1, errno, "Can't enumerate interfaces");
  439. }
  440. for (device = devices; device; device = device->next)
  441. {
  442. if (std::strcmp (this->mifname, device->name))
  443. {
  444. continue;
  445. }
  446. std::memcpy (this->miftext, device->description, std::strlen (device->description));
  447. if (device->addresses)
  448. {
  449. struct pcap_addr * pcap_addr = device->addresses;
  450. struct sockaddr_in * sockaddr_in = (struct sockaddr_in *)(pcap_addr->addr);
  451. struct in_addr * in_addr = (struct in_addr *)(&sockaddr_in->sin_addr);
  452. std::memcpy (this->mipaddr, &in_addr->s_addr, sizeof (this->mipaddr));
  453. }
  454. break;
  455. }
  456. pcap_freealldevs (devices);
  457. #endif
  458. return;
  459. }
  460. /*====================================================================*
  461. *
  462. * ointerface (unsigned ifindex)
  463. *
  464. * instantiate this instance with a given ifindex;
  465. *
  466. *--------------------------------------------------------------------*/
  467. ointerface::ointerface (unsigned ifindex)
  468. {
  469. this->mifindex = ifindex;
  470. std::memset (this->mhwaddr, 0, sizeof (this->mhwaddr));
  471. std::memset (this->mipaddr, 0, sizeof (this->mipaddr));
  472. omemory::hexdecode (this->mhwaddr, sizeof (this->mhwaddr), this->mhwstring, sizeof (this->mhwstring));
  473. omemory::decdecode (this->mipaddr, sizeof (this->mipaddr), this->mipstring, sizeof (this->mipstring));
  474. omemory::memtext ("none", this->mifname, sizeof (this->mifname));
  475. omemory::memtext ("none", this->miftext, sizeof (this->miftext));
  476. #if defined (WINPCAP)
  477. if (!ointerface::pcap_indextoname (this->mifindex, this->mifname))
  478. #else
  479. if (!if_indextoname (this->mifindex, this->mifname))
  480. #endif
  481. {
  482. oerror::error (1, errno, "%d", this->mifindex);
  483. }
  484. omemory::memtext (this->mifname, this->miftext, sizeof (this->miftext));
  485. ointerface::lookup ();
  486. return;
  487. }
  488. /*====================================================================*
  489. *
  490. * ointerface (char const * ifname)
  491. *
  492. * instantiate this instance with a given ifname;
  493. *
  494. *--------------------------------------------------------------------*/
  495. ointerface::ointerface (char const * ifname)
  496. {
  497. std::memset (this->mhwaddr, 0, sizeof (this->mhwaddr));
  498. std::memset (this->mipaddr, 0, sizeof (this->mipaddr));
  499. omemory::hexdecode (this->mhwaddr, sizeof (this->mhwaddr), this->mhwstring, sizeof (this->mhwstring));
  500. omemory::decdecode (this->mipaddr, sizeof (this->mipaddr), this->mipstring, sizeof (this->mipstring));
  501. omemory::memtext ("none", this->mifname, sizeof (this->mifname));
  502. omemory::memtext ("none", this->miftext, sizeof (this->miftext));
  503. #if defined (WINPCAP)
  504. this->mifindex = ointerface::pcap_nametoindex (ifname);
  505. #else
  506. this->mifindex = if_nametoindex (ifname);
  507. #endif
  508. if (this->mifindex)
  509. {
  510. omemory::memtext (ifname, this->mifname, sizeof (this->mifname));
  511. omemory::memtext (ifname, this->miftext, sizeof (this->miftext));
  512. ointerface::lookup ();
  513. }
  514. return;
  515. }
  516. /*====================================================================*
  517. *
  518. * ~ointerface ()
  519. *
  520. *--------------------------------------------------------------------*/
  521. ointerface::~ointerface ()
  522. {
  523. return;
  524. }
  525. /*====================================================================*
  526. * end definition;
  527. *--------------------------------------------------------------------*/
  528. #endif