CMEncrypt.c 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330
  1. /*====================================================================*
  2. *
  3. * Copyright (c) 2013 Qualcomm Atheros, Inc.
  4. *
  5. * All rights reserved.
  6. *
  7. *====================================================================*/
  8. /*====================================================================*
  9. * system header files;
  10. *--------------------------------------------------------------------*/
  11. #include <unistd.h>
  12. #include <stdlib.h>
  13. #include <stdint.h>
  14. #include <time.h>
  15. /*====================================================================*
  16. * custom header files;
  17. *--------------------------------------------------------------------*/
  18. #include "../tools/getoptv.h"
  19. #include "../tools/putoptv.h"
  20. #include "../tools/memory.h"
  21. #include "../tools/number.h"
  22. #include "../tools/symbol.h"
  23. #include "../tools/types.h"
  24. #include "../tools/flags.h"
  25. #include "../tools/files.h"
  26. #include "../tools/error.h"
  27. #include "../key/SHA256.h"
  28. #include "../plc/plc.h"
  29. /*====================================================================*
  30. * custom source files;
  31. *--------------------------------------------------------------------*/
  32. #ifndef MAKEFILE
  33. #include "../tools/getoptv.c"
  34. #include "../tools/putoptv.c"
  35. #include "../tools/version.c"
  36. #include "../tools/uintspec.c"
  37. #include "../tools/hexencode.c"
  38. #include "../tools/hexdecode.c"
  39. #include "../tools/hexdump.c"
  40. #include "../tools/todigit.c"
  41. #include "../tools/error.c"
  42. #include "../tools/synonym.c"
  43. #endif
  44. #ifndef MAKEFILE
  45. #include "../plc/Confirm.c"
  46. #include "../plc/Failure.c"
  47. #include "../plc/Request.c"
  48. #include "../plc/Devices.c"
  49. #endif
  50. #ifndef MAKEFILE
  51. #include "../ether/channel.c"
  52. #include "../ether/openchannel.c"
  53. #include "../ether/closechannel.c"
  54. #include "../ether/readpacket.c"
  55. #include "../ether/sendpacket.c"
  56. #endif
  57. #ifndef MAKEFILE
  58. #include "../key/SHA256Reset.c"
  59. #include "../key/SHA256Write.c"
  60. #include "../key/SHA256Block.c"
  61. #include "../key/SHA256Fetch.c"
  62. #endif
  63. #ifndef MAKEFILE
  64. #include "../mme/EthernetHeader.c"
  65. #include "../mme/HomePlugHeader.c"
  66. #include "../mme/MMECode.c"
  67. #endif
  68. /*====================================================================*
  69. * program constants;
  70. *--------------------------------------------------------------------*/
  71. #define CMENCRYPT_PEKS 0x0F
  72. #define CMENCRYPT_AVLN 0x00
  73. #define CMENCRYPT_PID 0x04
  74. /*====================================================================*
  75. *
  76. * int main (int argc, char const * argv[]);
  77. *
  78. *--------------------------------------------------------------------*/
  79. int main (int argc, char const * argv [])
  80. {
  81. extern struct channel channel;
  82. static char const * optv [] =
  83. {
  84. "A:f:i:K:P:qv",
  85. "device [device] [...]",
  86. "Send an encrypted payload using CM_ENCRYPTED_PAYLOAD",
  87. "A n\tAVLN Status [" LITERAL (CMENCRYPT_AVLN) "]",
  88. "K n\tPayload Encryption Key Select (PEKS) [" LITERAL (CMENCRYPT_PEKS) "]",
  89. "P n\tProtocol Identifier (PID) [" LITERAL (CMENCRYPT_PID) "]",
  90. #if defined (WINPCAP) || defined (LIBPCAP)
  91. "i n\thost interface is (n) [" LITERAL (CHANNEL_ETHNUMBER) "]",
  92. #else
  93. "i s\thost interface is (s) [" LITERAL (CHANNEL_ETHDEVICE) "]",
  94. #endif
  95. "f f\tpayload file",
  96. "q\tquiet mode",
  97. "v\tverbose mode",
  98. (char const *) (0)
  99. };
  100. #ifndef __GNUC__
  101. #pragma pack(push,1)
  102. #endif
  103. struct __packed cm_encrypted_payload
  104. {
  105. uint8_t PEKS;
  106. uint8_t AVLN;
  107. uint8_t PID;
  108. uint16_t PRN;
  109. uint8_t PMN;
  110. uint8_t UUID [16];
  111. uint16_t LEN;
  112. }
  113. template =
  114. {
  115. CMENCRYPT_PEKS,
  116. CMENCRYPT_AVLN,
  117. CMENCRYPT_PID,
  118. 0x0000,
  119. 0x00,
  120. {
  121. 0x00,
  122. 0x00,
  123. 0x00,
  124. 0x00,
  125. 0x00,
  126. 0x00,
  127. 0x00,
  128. 0x00,
  129. 0x00,
  130. 0x00,
  131. 0x00,
  132. 0x00,
  133. 0x00,
  134. 0x00,
  135. 0x00,
  136. 0x00
  137. },
  138. 0x0000
  139. };
  140. #ifndef __GNUC__
  141. #pragma pack (pop)
  142. #endif
  143. struct _file_ file =
  144. {
  145. -1,
  146. (char const *) (0)
  147. };
  148. struct sha256 sha256;
  149. uint8_t digest [SHA256_DIGEST_LENGTH];
  150. time_t timer = time ((time_t *) (0));
  151. uint8_t packet [ETHER_MAX_LEN];
  152. uint8_t * buffer;
  153. signed extent;
  154. signed c;
  155. if (getenv (PLCDEVICE))
  156. {
  157. #if defined (WINPCAP) || defined (LIBPCAP)
  158. channel.ifindex = atoi (getenv (PLCDEVICE));
  159. #else
  160. channel.ifname = strdup (getenv (PLCDEVICE));
  161. #endif
  162. }
  163. optind = 1;
  164. while (~ (c = getoptv (argc, argv, optv)))
  165. {
  166. switch (c)
  167. {
  168. case 'f':
  169. if ((file.file = open (file.name = optarg, O_BINARY | O_RDONLY)) == -1)
  170. {
  171. error (1, errno, "%s", file.name);
  172. }
  173. break;
  174. case 'P':
  175. template.PID = (byte) (uintspec (optarg, 0x00, 0x0F));
  176. break;
  177. case 'A':
  178. template.AVLN = (byte) (uintspec (optarg, 0x00, 0x08));
  179. break;
  180. case 'K':
  181. template.PEKS = (byte) (uintspec (optarg, 0x00, 0xFF));
  182. break;
  183. case 'i':
  184. #if defined (WINPCAP) || defined (LIBPCAP)
  185. channel.ifindex = atoi (optarg);
  186. #else
  187. channel.ifname = optarg;
  188. #endif
  189. break;
  190. case 'q':
  191. _setbits (channel.flags, CHANNEL_SILENCE);
  192. break;
  193. case 'v':
  194. _setbits (channel.flags, CHANNEL_VERBOSE);
  195. break;
  196. default:
  197. break;
  198. }
  199. }
  200. argc -= optind;
  201. argv += optind;
  202. /*
  203. * load entire file into memory;
  204. */
  205. if (file.file == -1)
  206. {
  207. error (1, ECANCELED, "No payload file given: Use -f <file>");
  208. }
  209. if ((extent = lseek (file.file, 0, SEEK_END)) == -1)
  210. {
  211. error (1, errno, FILE_CANTSIZE, file.name);
  212. }
  213. if (! (buffer = malloc (extent)))
  214. {
  215. error (1, errno, FILE_CANTLOAD, file.name);
  216. }
  217. if (lseek (file.file, 0, SEEK_SET))
  218. {
  219. error (1, errno, FILE_CANTHOME, file.name);
  220. }
  221. if (read (file.file, buffer, extent) != extent)
  222. {
  223. error (1, errno, FILE_CANTREAD, file.name);
  224. }
  225. close (file.file);
  226. SHA256Reset (& sha256);
  227. SHA256Write (& sha256, buffer, extent);
  228. SHA256Fetch (& sha256, digest);
  229. if (! argc)
  230. {
  231. error (1, ECANCELED, "No destination given");
  232. }
  233. openchannel (& channel);
  234. while ((argc) && (* argv))
  235. {
  236. signed offset = 0;
  237. signed remain = extent;
  238. #if 0
  239. signed length = sizeof (struct packet_ms) - sizeof (template);
  240. #else
  241. signed length = 502 - sizeof (template);
  242. #endif
  243. if (! hexencode (channel.peer, sizeof (channel.peer), synonym (* argv, devices, SIZEOF (devices))))
  244. {
  245. error (1, errno, PLC_BAD_MAC, * argv);
  246. }
  247. template.PRN = (uint16_t) (timer);
  248. template.PMN = 0;
  249. memcpy (template.UUID, digest, sizeof (template.UUID));
  250. while (remain)
  251. {
  252. uint8_t * memory = packet;
  253. if (length > remain)
  254. {
  255. length = remain;
  256. }
  257. template.PMN++;
  258. template.LEN = HTOLE16 (length);
  259. memset (memory, 0, sizeof (struct message));
  260. memory += EthernetHeader ((struct ethernet_hdr *) (memory), channel.peer, channel.host, channel.type);
  261. memory += HomePlugHeader ((struct homeplug_hdr *) (memory), 0, (CM_ENCRYPTED_PAYLOAD | MMTYPE_IND));
  262. memcpy (memory, & template, sizeof (template));
  263. memory += sizeof (template);
  264. memcpy (memory, buffer + offset, length);
  265. memory += length;
  266. extent = (signed) (memory - packet);
  267. if (extent < (ETHER_MIN_LEN - ETHER_CRC_LEN))
  268. {
  269. extent = (ETHER_MIN_LEN - ETHER_CRC_LEN);
  270. }
  271. if (sendpacket (& channel, packet, extent) < extent)
  272. {
  273. error (1, errno, CHANNEL_CANTSEND);
  274. }
  275. offset += length;
  276. remain -= length;
  277. }
  278. argc--;
  279. argv++;
  280. }
  281. closechannel (& channel);
  282. free (buffer);
  283. exit (0);
  284. }