Antiphon.c 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104
  1. /*====================================================================*
  2. *
  3. * Copyright (c) 2013 Qualcomm Atheros, Inc.
  4. *
  5. * All rights reserved.
  6. *
  7. *====================================================================*/
  8. /*====================================================================*
  9. *
  10. * signed Antiphon (struct plc * plc);
  11. *
  12. * plc.h
  13. *
  14. * instruct one powerline device to send trash frames to another
  15. * for a given period (in seconds) to establish source device RX
  16. * PHY rate and target device TX PHY rate; this does not work if
  17. * the source device is also the local device, for some reason;
  18. *
  19. * Contributor(s):
  20. * Charles Maier <cmaier@qca.qualcomm.com>
  21. *
  22. *--------------------------------------------------------------------*/
  23. #ifndef ANTIPHON_SOURCE
  24. #define ANTIPHON_SOURCE
  25. #include <memory.h>
  26. #include <errno.h>
  27. #include "../tools/error.h"
  28. #include "../tools/flags.h"
  29. #include "../plc/plc.h"
  30. signed Antiphon (struct plc * plc, byte source [], byte target [])
  31. {
  32. struct channel * channel = (struct channel *) (plc->channel);
  33. struct message * message = (struct message *) (plc->message);
  34. #ifndef __GNUC__
  35. #pragma pack (push,1)
  36. #endif
  37. struct __packed vs_fr_lbk_request
  38. {
  39. struct ethernet_hdr ethernet;
  40. struct qualcomm_hdr qualcomm;
  41. uint8_t DURATION;
  42. uint8_t RESERVED;
  43. uint16_t LENGTH;
  44. uint8_t DATA [1038];
  45. }
  46. * request = (struct vs_fr_lbk_request *) (message);
  47. #ifndef __GNUC__
  48. #pragma pack (pop)
  49. #endif
  50. if (_allclr (plc->flags, PLC_SILENCE))
  51. {
  52. char sourcename [ETHER_ADDR_LEN * 3];
  53. char targetname [ETHER_ADDR_LEN * 3];
  54. hexdecode (source, ETHER_ADDR_LEN, sourcename, sizeof (sourcename));
  55. hexdecode (target, ETHER_ADDR_LEN, targetname, sizeof (targetname));
  56. fprintf (stderr, "%s %s %s\n", channel->ifname, sourcename, targetname);
  57. }
  58. memset (message, 0, sizeof (* message));
  59. EthernetHeader (& request->ethernet, source, target, channel->type);
  60. QualcommHeader (& request->qualcomm, 0, (VS_FR_LBK | MMTYPE_REQ));
  61. request->DURATION = plc->timer;
  62. request->LENGTH = HTOLE16 (sizeof (request->DATA));
  63. memset (request->DATA, 0xA5, sizeof (request->DATA));
  64. EthernetHeader (request->DATA, target, source, ETHERTYPE_IP);
  65. plc->packetsize = sizeof (* request);
  66. if (SendMME (plc) <= 0)
  67. {
  68. error (1, errno, CHANNEL_CANTSEND);
  69. }
  70. #if 0
  71. /*
  72. * This causes a multi-device session to terminate if the device has recently
  73. * been removed or powered-off; The device appears to be present but will not
  74. * respond; Also, this terminates a session if the network is overloaded with
  75. * traffic;
  76. */
  77. if (ReadMME (plc, 0, (VS_FR_LBK | MMTYPE_CNF)) <= 0)
  78. {
  79. error (1, errno, CHANNEL_CANTREAD);
  80. }
  81. #endif
  82. sleep (plc->timer);
  83. return (0);
  84. }
  85. #endif