CPLChannel.cpp 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865
  1. /*====================================================================*
  2. Copyright (c) 2020 Qualcomm Technologies, Inc.
  3. All Rights Reserved.
  4. Confidential and Proprietary - Qualcomm Technologies, Inc.
  5. ******************************************************************
  6. 2013 Qualcomm Atheros, Inc.
  7. *====================================================================*/
  8. /*====================================================================*
  9. *
  10. * CPLChannel.cpp - CPLChannel class definition;
  11. *
  12. * Ethernet I/O channel managment for powerline applications;
  13. *
  14. * Contributor(s):
  15. * Charles Maier <charles.maier@intellon.com>
  16. *
  17. *--------------------------------------------------------------------*/
  18. #ifndef CPLCHANNEL_SOURCE
  19. #define CPLCHANNEL_SOURCE
  20. #define SLOTS 6
  21. #define CARRIERS 1155
  22. /*====================================================================*
  23. * system header files;
  24. *--------------------------------------------------------------------*/
  25. #include <unistd.h>
  26. #include <iostream>
  27. #include <cstring>
  28. #include <cstdlib>
  29. /*====================================================================*
  30. * system header files;
  31. *--------------------------------------------------------------------*/
  32. #if defined (__linux__)
  33. # include <sys/socket.h>
  34. # include <sys/ioctl.h>
  35. # include <sys/poll.h>
  36. # include <linux/if_packet.h>
  37. # include <net/ethernet.h>
  38. # include <net/if_arp.h>
  39. # include <net/if.h>
  40. #elif defined (__APPLE__)
  41. # include <sys/types.h>
  42. # include <sys/socket.h>
  43. # include <sys/ioctl.h>
  44. # include <netinet/if_ether.h>
  45. # include <net/if_dl.h>
  46. # include <net/bpf.h>
  47. # include <fcntl.h>
  48. #elif defined (__OpenBSD__)
  49. # include <sys/types.h>
  50. # include <sys/socket.h>
  51. # include <sys/ioctl.h>
  52. # include <net/bpf.h>
  53. # include <unistd.h>
  54. # include <fcntl.h>
  55. #elif defined (WINPCAP)
  56. #elif defined (LIBPCAP)
  57. #else
  58. #error "Unknown Environment"
  59. #endif
  60. /*====================================================================*
  61. * custom header files;
  62. *--------------------------------------------------------------------*/
  63. #include "../classes/CPLChannel.hpp"
  64. #include "../classes/ohomeplug.hpp"
  65. #include "../classes/ointellon.hpp"
  66. #include "../classes/omemory.hpp"
  67. #include "../classes/oerror.hpp"
  68. /*====================================================================*
  69. *
  70. * signed Descriptor () const;
  71. *
  72. * return the channel socket file descriptor;
  73. *
  74. *--------------------------------------------------------------------*/
  75. signed CPLChannel::Descriptor () const
  76. {
  77. return (this->mfd);
  78. }
  79. /*====================================================================*
  80. *
  81. * signed Bridges (void * memory, size_t extent);
  82. *
  83. * encode memory with a consecutive list of bridge device hardware
  84. * addresses; return the number of addresses encoded; return -1 on
  85. * memory overflow;
  86. *
  87. * this is the start point for device discovery; each bridge could
  88. * be the gateway to a separate powerline network;
  89. *
  90. *--------------------------------------------------------------------*/
  91. signed CPLChannel::Bridges (void * memory, size_t extent)
  92. {
  93. ointellon intellon;
  94. byte * origin = (byte *)(memory);
  95. byte * offset = (byte *)(memory);
  96. byte message [ETHER_MAX_LEN];
  97. std::memset (memory, 0, extent);
  98. std::memset (message, 0, sizeof (message));
  99. intellon.ImportHostAddress (this->HostAddress ());
  100. intellon.ExportHeader (message);
  101. if (this->SendMessage (message, ETHER_MIN_LEN) > 0)
  102. {
  103. while (this->ReadMessage (message, sizeof (message)) > 0)
  104. {
  105. if (extent < ETHER_ADDR_LEN)
  106. {
  107. oerror::error (0, EOVERFLOW, "Bridge address lost");
  108. continue;
  109. }
  110. intellon.ImportHeader (message);
  111. if (intellon.IsMessageType (0, VS_SW_VER| MMTYPE_CNF))
  112. {
  113. intellon.ExportHostAddress (offset);
  114. offset += ETHER_ADDR_LEN;
  115. extent -= ETHER_ADDR_LEN;
  116. }
  117. }
  118. }
  119. return ((signed)(offset - origin) / ETHER_ADDR_LEN);
  120. }
  121. /*====================================================================*
  122. *
  123. * signed Neighbors (void * memory, size_t extent);
  124. *
  125. * return a list powerline network device addresses on a powerline
  126. * network; the list consists of a known device plus all others on
  127. * the same powerline network; the device is defined by the channel
  128. * peer address and appears first in the list;
  129. *
  130. * the device address must be explicit; it cannot be the emptycast,
  131. * broadcast or localcast address;
  132. *
  133. * the first (known) device is omitted here as an expriment despite
  134. * what was said above; - charlie maier
  135. *
  136. *--------------------------------------------------------------------*/
  137. signed CPLChannel::Neighbors (void * memory, size_t extent)
  138. {
  139. ointellon intellon;
  140. byte * origin = (byte *)(memory);
  141. byte * offset = (byte *)(memory);
  142. byte message [ETHER_MAX_LEN];
  143. #ifndef __GNUC__
  144. #pragma pack (push,1)
  145. #endif
  146. struct __packed station
  147. {
  148. uint8_t DA [ETHER_ADDR_LEN];
  149. uint8_t TEI;
  150. uint8_t BDA [ETHER_ADDR_LEN];
  151. uint8_t AVGTX;
  152. uint8_t AVGRX;
  153. }
  154. * station;
  155. struct __packed network
  156. {
  157. uint8_t NID [7];
  158. uint8_t SNID;
  159. uint8_t TEI;
  160. uint8_t ROLE;
  161. uint8_t CCO_MACADDR [ETHER_ADDR_LEN];
  162. uint8_t CCO_TEI;
  163. uint8_t NUMSTAS;
  164. struct station station [1];
  165. }
  166. * network;
  167. struct __packed networks
  168. {
  169. uint8_t NUMAVLNS;
  170. struct network network [1];
  171. }
  172. * networks;
  173. #ifndef __GNUC__
  174. #pragma pack (pop)
  175. #endif
  176. std::memset (memory, 0, extent);
  177. if (!std::memcmp (this->PeerAddress (), oethernet::EmptycastAddress, ETHER_ADDR_LEN))
  178. {
  179. oerror::error (0, ECANCELED, "Emptycast address used to explore network");
  180. return (0);
  181. }
  182. if (!std::memcmp (this->PeerAddress (), oethernet::BroadcastAddress, ETHER_ADDR_LEN))
  183. {
  184. oerror::error (0, ECANCELED, "Broadcast address used to explore network");
  185. return (0);
  186. }
  187. if (!std::memcmp (this->PeerAddress (), ointellon::LocalcastAddress, ETHER_ADDR_LEN))
  188. {
  189. oerror::error (0, ECANCELED, "Localcast address used to explore network");
  190. return (0);
  191. }
  192. std::memset (message, 0, sizeof (message));
  193. intellon.ImportPeerAddress (this->PeerAddress ());
  194. intellon.ImportHostAddress (this->HostAddress ());
  195. intellon.SetMessageType (VS_NW_INFO | MMTYPE_REQ);
  196. networks = (struct networks *)(intellon.ExportHeader (message));
  197. if (this->SendMessage (message, ETHER_MIN_LEN) <= 0)
  198. {
  199. oerror::error (0, errno, CPLCHANNEL_CANTSEND);
  200. return (0);
  201. }
  202. if (this->ReadMessage (message, sizeof (message)) <= 0)
  203. {
  204. oerror::error (0, errno, CPLCHANNEL_CANTREAD);
  205. return (0);
  206. }
  207. network = (struct network *)(&networks->network);
  208. while (networks->NUMAVLNS-- > 0)
  209. {
  210. if (extent < ETHER_ADDR_LEN)
  211. {
  212. oerror::error (0, EOVERFLOW, "Bridge address lost");
  213. return (-1);
  214. }
  215. #if 0
  216. intellon.ImportHeader (message);
  217. intellon.ExportHostAddress (offset);
  218. offset += ETHER_ADDR_LEN;
  219. extent -= ETHER_ADDR_LEN;
  220. #endif
  221. station = (struct station *)(&network->station);
  222. while (network->NUMSTAS-- > 0)
  223. {
  224. if (extent < ETHER_ADDR_LEN)
  225. {
  226. oerror::error (0, EOVERFLOW, "Device address lost");
  227. return (-1);
  228. }
  229. if (std::memcmp (station->DA, oethernet::BroadcastAddress, ETHER_ADDR_LEN))
  230. {
  231. std::memcpy (offset, station->DA, sizeof (station->DA));
  232. offset += ETHER_ADDR_LEN;
  233. extent -= ETHER_ADDR_LEN;
  234. }
  235. station++;
  236. }
  237. network = (struct network *)(station);
  238. }
  239. return ((signed)(offset - origin) / ETHER_ADDR_LEN);
  240. }
  241. /*====================================================================*
  242. *
  243. * signed SendMessage (void const * message, signed length);
  244. *
  245. *--------------------------------------------------------------------*/
  246. signed CPLChannel::SendMessage (void const * memory, signed extent)
  247. {
  248. this->dump (memory, extent);
  249. #if defined (__linux__)
  250. extent = sendto (this->mfd, memory, extent, 0, (struct sockaddr *) (0), (socklen_t) (0));
  251. #elif defined (__APPLE__) || defined (__OpenBSD__)
  252. extent = write (this->mfd, memory, extent);
  253. #elif defined (WINPCAP) || defined (LIBPCAP)
  254. if (pcap_sendpacket (this->msocket, (const u_char *)(memory), extent))
  255. {
  256. extent = -1;
  257. }
  258. #else
  259. #error "Unknown Environment"
  260. #endif
  261. return (extent);
  262. }
  263. /*====================================================================*
  264. *
  265. * signed ReadMessage (void * memory, signed extent);
  266. *
  267. * encode external memory with an incoming Ethernet frame; return
  268. * frame length on success, 0 on timeout or -1 on error;
  269. *
  270. * on linux/osx, this method returns as soon as a frame arrives or
  271. * once the timeout has expired; consequently, long timeout values
  272. * do not affect performance;
  273. *
  274. * on winpcap this method does not return until timeout expires;
  275. * consequenty, long timeouts affect performance;
  276. *
  277. *--------------------------------------------------------------------*/
  278. signed CPLChannel::ReadMessage (void * memory, signed extent)
  279. {
  280. #if defined (__linux__)
  281. struct pollfd pollfd =
  282. {
  283. this->mfd,
  284. POLLIN,
  285. 0
  286. };
  287. int status = poll (&pollfd, 1, this->mtimeout);
  288. std::memset (memory, 0, extent);
  289. if (status < 0)
  290. {
  291. oerror::error (0, errno, "poll");
  292. return (-1);
  293. }
  294. if (status > 0)
  295. {
  296. extent = recvfrom (this->mfd, memory, extent, 0, (struct sockaddr *) (0), (socklen_t *)(0));
  297. if (extent == -1)
  298. {
  299. oerror::error (0, errno, "recvfrom");
  300. return (-1);
  301. }
  302. this->dump (memory, extent);
  303. return (extent);
  304. }
  305. #elif defined (__APPLE__) || defined (__OpenBSD__)
  306. byte buffer [this->bpf_length];
  307. struct bpf_hdr * bpf_hdr = (struct bpf_hdr *)(buffer);
  308. std::memset (memory, 0, extent);
  309. std::memset (buffer, 0, sizeof (buffer));
  310. extent = read (this->mfd, buffer, sizeof (buffer));
  311. if (extent < 0)
  312. {
  313. oerror::error (0, errno, "bpf");
  314. return (-1);
  315. }
  316. if (extent > 0)
  317. {
  318. extent = bpf_hdr->bh_caplen;
  319. std::memcpy (memory, buffer + bpf_hdr->bh_hdrlen, bpf_hdr->bh_caplen);
  320. this->dump (memory, extent);
  321. return (extent);
  322. }
  323. #elif defined (WINPCAP) || defined (LIBPCAP)
  324. struct pcap_pkthdr * header;
  325. const byte * data;
  326. signed status = pcap_next_ex (this->msocket, &header, &data);
  327. std::memset (memory, 0, extent);
  328. if (status < 0)
  329. {
  330. oerror::error (0, errno, "pcap_next_ex");
  331. return (-1);
  332. }
  333. if (status > 0)
  334. {
  335. extent = header->caplen;
  336. std::memcpy (memory, data, header->caplen);
  337. this->dump (memory, extent);
  338. return (extent);
  339. }
  340. #else
  341. #error "Unknown Environment"
  342. #endif
  343. return (0);
  344. }
  345. /*====================================================================*
  346. *
  347. * CPLChannel & dump (void const * memory, signed extent);
  348. *
  349. * print Ethernet frames in hex dump format on stderr when verbose
  350. * flag is set; use this for testing and debugging purposes;
  351. *
  352. *--------------------------------------------------------------------*/
  353. CPLChannel & CPLChannel::dump (void const * memory, size_t extent)
  354. {
  355. if (oflagword::anyset (CPLCHANNEL_FLAG_VERBOSE))
  356. {
  357. omemory::hexdump (memory, 0, extent, &std::cerr);
  358. std::cerr << std::endl;
  359. }
  360. return (*this);
  361. }
  362. /*====================================================================*
  363. *
  364. * CPLChannel & open ()
  365. *
  366. * open a raw ethernet socket on the designated interface and apply
  367. * a packet filter; the filter accepts HomePlug AV frames addressed
  368. * to either this host or the ethernet broadcast address; set the
  369. * channel host address to the interface hardware address;
  370. *
  371. * if you don't understand this code then you probably have a life;
  372. *
  373. *--------------------------------------------------------------------*/
  374. CPLChannel & CPLChannel::open ()
  375. {
  376. #if defined (__linux__)
  377. struct ifreq ifreq;
  378. struct sockaddr_ll sockaddr_ll =
  379. {
  380. PF_PACKET,
  381. 0x0000,
  382. 0x0000,
  383. ARPHRD_ETHER,
  384. PACKET_OTHERHOST,
  385. ETHER_ADDR_LEN,
  386. {
  387. 0x00,
  388. 0x00,
  389. 0x00,
  390. 0x00,
  391. 0x00,
  392. 0x00,
  393. 0x00,
  394. 0x00
  395. }
  396. };
  397. std::memset (&ifreq, 0, sizeof (ifreq));
  398. oethernet::ExportProtocol (&sockaddr_ll.sll_protocol);
  399. if ((this->mfd = socket (sockaddr_ll.sll_family, SOCK_RAW, sockaddr_ll.sll_protocol)) == -1)
  400. {
  401. oerror::error (1, errno, "%s", ifreq.ifr_name);
  402. }
  403. std::memcpy (ifreq.ifr_name, this->Name (), sizeof (ifreq.ifr_name));
  404. if (ioctl (this->mfd, SIOCGIFINDEX, &ifreq) == -1)
  405. {
  406. oerror::error (0, errno, "%s", ifreq.ifr_name);
  407. }
  408. sockaddr_ll.sll_ifindex = ifreq.ifr_ifindex;
  409. if (ioctl (this->mfd, SIOCGIFHWADDR, &ifreq) == -1)
  410. {
  411. oerror::error (0, errno, "%s", ifreq.ifr_name);
  412. }
  413. std::memcpy (sockaddr_ll.sll_addr, ifreq.ifr_ifru.ifru_hwaddr.sa_data, sizeof (sockaddr_ll.sll_addr));
  414. if (bind (this->mfd, (struct sockaddr *) (&sockaddr_ll), sizeof (sockaddr_ll)) == -1)
  415. {
  416. oerror::error (0, errno, "%s", ifreq.ifr_name);
  417. }
  418. if (ioctl (this->mfd, SIOCGIFFLAGS, &ifreq) == -1)
  419. {
  420. oerror::error (0, errno, "%s", ifreq.ifr_name);
  421. }
  422. ifreq.ifr_flags |= (IFF_UP | IFF_BROADCAST | IFF_MULTICAST);
  423. ifreq.ifr_flags &= ~(IFF_ALLMULTI | IFF_PROMISC);
  424. if (ioctl (this->mfd, SIOCSIFFLAGS, &ifreq) == -1)
  425. {
  426. oerror::error (0, errno, "%s", ifreq.ifr_name);
  427. }
  428. #else
  429. struct bpf_program bpf_program;
  430. static struct bpf_insn bpf_insn [] =
  431. {
  432. {
  433. BPF_LD + BPF_H + BPF_ABS,
  434. 0,
  435. 0,
  436. 12
  437. },
  438. {
  439. BPF_JMP + BPF_JEQ + BPF_K,
  440. 0,
  441. 18,
  442. 0
  443. },
  444. {
  445. BPF_LD + BPF_B + BPF_ABS,
  446. 0,
  447. 0,
  448. 0
  449. },
  450. {
  451. BPF_JMP + BPF_JEQ + BPF_K,
  452. 0,
  453. 10,
  454. 0
  455. },
  456. {
  457. BPF_LD + BPF_B + BPF_ABS,
  458. 0,
  459. 0,
  460. 1
  461. },
  462. {
  463. BPF_JMP + BPF_JEQ + BPF_K,
  464. 0,
  465. 8,
  466. 0
  467. },
  468. {
  469. BPF_LD + BPF_B + BPF_ABS,
  470. 0,
  471. 0,
  472. 2
  473. },
  474. {
  475. BPF_JMP + BPF_JEQ + BPF_K,
  476. 0,
  477. 6,
  478. 0
  479. },
  480. {
  481. BPF_LD + BPF_B + BPF_ABS,
  482. 0,
  483. 0,
  484. 3
  485. },
  486. {
  487. BPF_JMP + BPF_JEQ + BPF_K,
  488. 0,
  489. 4,
  490. 0
  491. },
  492. {
  493. BPF_LD + BPF_B + BPF_ABS,
  494. 0,
  495. 0,
  496. 4
  497. },
  498. {
  499. BPF_JMP + BPF_JEQ + BPF_K,
  500. 0,
  501. 2,
  502. 0
  503. },
  504. {
  505. BPF_LD + BPF_B + BPF_ABS,
  506. 0,
  507. 0,
  508. 5
  509. },
  510. {
  511. BPF_JMP + BPF_JEQ + BPF_K,
  512. 4,
  513. 0,
  514. 0
  515. },
  516. {
  517. BPF_LD + BPF_W + BPF_ABS,
  518. 0,
  519. 0,
  520. 0
  521. },
  522. {
  523. BPF_JMP + BPF_JEQ + BPF_K,
  524. 0,
  525. 4,
  526. 0xFFFFFFFF
  527. },
  528. {
  529. BPF_LD + BPF_H + BPF_ABS,
  530. 0,
  531. 0,
  532. 4
  533. },
  534. {
  535. BPF_JMP + BPF_JEQ + BPF_K,
  536. 0,
  537. 2,
  538. 0xFFFF
  539. },
  540. {
  541. BPF_LD + BPF_W + BPF_LEN,
  542. 0,
  543. 0,
  544. 0
  545. },
  546. {
  547. BPF_RET + BPF_A,
  548. 0,
  549. 0,
  550. 0
  551. },
  552. {
  553. BPF_RET + BPF_K,
  554. 0,
  555. 0,
  556. 0
  557. }
  558. };
  559. #if defined (__APPLE__) || defined (__OpenBSD__)
  560. struct ifreq ifreq;
  561. struct timeval timer =
  562. {
  563. 0,
  564. 0
  565. };
  566. const byte * hwaddr = this->HardwareAddress ();
  567. char filename [FILENAME_MAX];
  568. unsigned count;
  569. unsigned state;
  570. for (count = 0; count < 100; count++)
  571. {
  572. std::snprintf (filename, sizeof (filename), CPLCHANNEL_BPFDEVICE, count);
  573. if ((this->mfd =::open (filename, O_RDWR)) != -1)
  574. {
  575. break;
  576. }
  577. }
  578. if (this->mfd == -1)
  579. {
  580. oerror::error (1, ECANCELED, "No bpf devices available");
  581. }
  582. std::memcpy (ifreq.ifr_name, ointerface::Name (), sizeof (ifreq.ifr_name));
  583. if (ioctl (this->mfd, BIOCSETIF, &ifreq) == -1)
  584. {
  585. oerror::error (0, errno, "1 %s", ifreq.ifr_name);
  586. return (*this);
  587. }
  588. if (ioctl (this->mfd, BIOCGBLEN, &this->bpf_length) == -1)
  589. {
  590. oerror::error (0, errno, "Can't determine buffer length");
  591. }
  592. state = true;
  593. if (ioctl (this->mfd, BIOCIMMEDIATE, &state) == -1)
  594. {
  595. oerror::error (0, errno, "Can't activate immediate mode");
  596. }
  597. #if defined (__APPLE__)
  598. state = false;
  599. if (ioctl (this->mfd, BIOCSSEESENT, &state) == -1)
  600. {
  601. oerror::error (0, errno, "Can't hide outgoing frames");
  602. }
  603. #elif defined (__OpenBSD__)
  604. state = BPF_DIRECTION_OUT;
  605. if (ioctl (this->mfd, BIOCSDIRFILT, &state) == -1)
  606. {
  607. oerror::error (0, errno, "Can't hide outgoing frames");
  608. }
  609. #else
  610. #error "Abandon all hope"
  611. #endif
  612. #if defined (__MAC_10_6)
  613. /*
  614. * accommodate know bug in BPF on MAC OS X 10.6; shorter times may cause socket
  615. * read operations to block indefinitely when no frames are available;
  616. */
  617. timer.tv_sec = 1;
  618. #else
  619. timer.tv_usec = this->mtimeout * 1000;
  620. #endif
  621. if (ioctl (this->mfd, BIOCSRTIMEOUT, &timer) == -1)
  622. {
  623. oerror::error (0, errno, "Can't set timeout");
  624. }
  625. bpf_program.bf_len = sizeof (bpf_insn)/sizeof (struct bpf_insn);
  626. bpf_program.bf_insns = bpf_insn;
  627. bpf_insn [1].k = oethernet::Protocol ();
  628. bpf_insn [3].k = hwaddr [0];
  629. bpf_insn [5].k = hwaddr [1];
  630. bpf_insn [7].k = hwaddr [2];
  631. bpf_insn [9].k = hwaddr [3];
  632. bpf_insn [11].k = hwaddr [4];
  633. bpf_insn [13].k = hwaddr [5];
  634. if (ioctl (this->mfd, BIOCSETF, &bpf_program) == -1)
  635. {
  636. oerror::error (0, errno, "Can't use filter");
  637. }
  638. #elif defined (WINPCAP) || defined (LIBPCAP)
  639. const byte * hostaddr = this->HardwareAddress ();
  640. this->msocket = pcap_open_live (this->Name (), 65536, 0, this->mtimeout, this->merrbuf);
  641. if (!this->msocket)
  642. {
  643. oerror::error (1, errno, "No such adapter: %s", ointerface::Name ());
  644. }
  645. bpf_program.bf_len = sizeof (bpf_insn)/sizeof (struct bpf_insn);
  646. bpf_program.bf_insns = bpf_insn;
  647. bpf_insn [1].k = oethernet::Protocol ();
  648. bpf_insn [3].k = hostaddr [0];
  649. bpf_insn [5].k = hostaddr [1];
  650. bpf_insn [7].k = hostaddr [2];
  651. bpf_insn [9].k = hostaddr [3];
  652. bpf_insn [11].k = hostaddr [4];
  653. bpf_insn [13].k = hostaddr [5];
  654. if (pcap_setfilter (this->msocket, &bpf_program) < 0)
  655. {
  656. oerror::error (0, errno, "Can't use filter: %s", ointerface::Name ());
  657. }
  658. if (pcap_setmintocopy (this->msocket, ETHER_MIN_LEN))
  659. {
  660. oerror::error (0, errno, "Can't open socket: %s", ointerface::Name ());
  661. }
  662. #else
  663. #error "Unknown Environment"
  664. #endif
  665. #endif
  666. oethernet::ImportHostAddress (this->HardwareAddress ());
  667. return (*this);
  668. }
  669. /*====================================================================*
  670. *
  671. * CPLChannel & link ()
  672. *
  673. * find any available powerline bridge and set the channel peer
  674. * address; read all responses because this is a local broadcast;
  675. * the last response read will be the lucky device;
  676. *
  677. *--------------------------------------------------------------------*/
  678. CPLChannel & CPLChannel::link ()
  679. {
  680. ointellon intellon;
  681. byte message [ETHER_MAX_LEN];
  682. std::memset (message, 0, sizeof (message));
  683. intellon.ImportHostAddress (this->HostAddress ());
  684. intellon.ExportHeader (message);
  685. if (this->SendMessage (message, ETHER_MIN_LEN) > 0)
  686. {
  687. while (this->ReadMessage (message, sizeof (message)) > 0)
  688. {
  689. intellon.ImportHeader (message);
  690. if (intellon.IsMessageType (0, VS_SW_VER|MMTYPE_CNF))
  691. {
  692. this->ImportPeerAddress (intellon.HostAddress ());
  693. }
  694. }
  695. }
  696. return (*this);
  697. }
  698. /*====================================================================*
  699. *
  700. * CPLChannel & init (unsigned timeout)
  701. *
  702. * initialize class members; set the channel Ethernet header to
  703. * the default Intellon header;
  704. *
  705. *--------------------------------------------------------------------*/
  706. CPLChannel & CPLChannel::init (unsigned timeout)
  707. {
  708. ointellon intellon;
  709. this->mfd = -1;
  710. this->mtimeout = timeout;
  711. this->ImportPeerAddress (intellon.PeerAddress ());
  712. this->ImportHostAddress (intellon.HostAddress ());
  713. this->SetProtocol (intellon.Protocol ());
  714. return (*this);
  715. }
  716. /*====================================================================*
  717. *
  718. * CPLChannel (unsigned ifindex, unsigned timeout)
  719. *
  720. *--------------------------------------------------------------------*/
  721. CPLChannel::CPLChannel (unsigned ifindex, unsigned timeout): ointerface (ifindex)
  722. {
  723. this->init (timeout);
  724. this->open ();
  725. this->link ();
  726. return;
  727. }
  728. /*====================================================================*
  729. *
  730. * CPLChannel (char const * ifname, unsigned vTimeout)
  731. *
  732. *--------------------------------------------------------------------*/
  733. CPLChannel::CPLChannel (char const * ifname, unsigned vTimeout): ointerface (ifname)
  734. {
  735. this->init (vTimeout);
  736. this->open ();
  737. this->link ();
  738. return;
  739. }
  740. /*====================================================================*
  741. *
  742. * ~CPLChannel ()
  743. *
  744. * free sockets and descriptors;
  745. *
  746. *--------------------------------------------------------------------*/
  747. CPLChannel::~CPLChannel ()
  748. {
  749. #if defined (__linux__)
  750. ::close (this->mfd);
  751. #elif defined (__APPLE__) || defined (__OpenBSD__)
  752. ::close (this->mfd);
  753. #elif defined (WINPCAP) || defined (LIBPCAP)
  754. pcap_close (this->msocket);
  755. #else
  756. #error "Unknown Environment"
  757. #endif
  758. return;
  759. }
  760. /*====================================================================*
  761. * end definition;
  762. *--------------------------------------------------------------------*/
  763. #endif