backoff.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520
  1. /*====================================================================*
  2. * Copyright (c) 2018 Qualcomm Technologies, Inc.
  3. * All Rights Reserved.
  4. * Confidential and Proprietary - Qualcomm Technologies, Inc.
  5. *--------------------------------------------------------------------*/
  6. /*====================================================================*
  7. *
  8. * backoff.c
  9. *
  10. * Contributor(s):
  11. * Nisha K <nishk@qti.qualcomm.com>
  12. *
  13. *--------------------------------------------------------------------*/
  14. #include <stdio.h>
  15. #include <unistd.h>
  16. #include <stdlib.h>
  17. #include <stdint.h>
  18. #include <limits.h>
  19. #include <sys/time.h>
  20. /*====================================================================*
  21. * custom header files;
  22. *--------------------------------------------------------------------*/
  23. #include "../tools/getoptv.h"
  24. #include "../tools/putoptv.h"
  25. #include "../tools/memory.h"
  26. #include "../tools/symbol.h"
  27. #include "../tools/number.h"
  28. #include "../tools/timer.h"
  29. #include "../tools/endian.h"
  30. #include "../tools/types.h"
  31. #include "../tools/flags.h"
  32. #include "../tools/files.h"
  33. #include "../tools/error.h"
  34. #include "../tools/tlv.h"
  35. #include "../plc/plc.h"
  36. /*====================================================================*
  37. * custom source files;
  38. *--------------------------------------------------------------------*/
  39. #ifndef MAKEFILE
  40. #include "../plc/Devices.c"
  41. #include "../plc/Failure.c"
  42. #include "../plc/ReadMME.c"
  43. #include "../plc/SendMME.c"
  44. #endif
  45. #ifndef MAKEFILE
  46. #include "../tools/error.c"
  47. #include "../tools/getoptv.c"
  48. #include "../tools/putoptv.c"
  49. #include "../tools/version.c"
  50. #include "../tools/uintspec.c"
  51. #include "../tools/hexload.c"
  52. #include "../tools/timer.h"
  53. #include "../tools/hexdump.c"
  54. #include "../tools/hexencode.c"
  55. #include "../tools/hexdecode.c"
  56. #include "../tools/hexstring.c"
  57. #include "../tools/todigit.c"
  58. #include "../tools/synonym.c"
  59. #endif
  60. #ifndef MAKEFILE
  61. #include "../ether/openchannel.c"
  62. #include "../ether/closechannel.c"
  63. #include "../ether/readpacket.c"
  64. #include "../ether/sendpacket.c"
  65. #include "../ether/channel.c"
  66. #endif
  67. #ifndef MAKEFILE
  68. #include "../mme/EthernetHeader.c"
  69. #include "../mme/QualcommHeader.c"
  70. #include "../mme/UnwantedMessage.c"
  71. #include "../mme/MMECode.c"
  72. #endif
  73. /*====================================================================*
  74. *
  75. * signed getprescalervalues (struct plc * plc, uint8_t numofbands);
  76. *
  77. *===================================================================*/
  78. static signed getprescalervalues (struct plc * plc)
  79. {
  80. struct channel * channel = (struct channel *) (plc->channel);
  81. struct message * message = (struct message *) (plc->message);
  82. #ifndef __GNUC__
  83. #pragma pack (push,1)
  84. #endif
  85. typedef struct __packed {
  86. uint16_t startidx;
  87. uint8_t startbackoffdB;
  88. uint16_t endidx;
  89. uint8_t endbackoffdB;
  90. } banddesc;
  91. struct __packed vs_get_prescaler_backoff_request
  92. {
  93. struct ethernet_hdr ethernet;
  94. struct qualcomm_hdr qualcomm;
  95. uint8_t RESERVED[4];
  96. }
  97. * request = (struct vs_get_prescaler_backoff_request *) (message);
  98. struct __packed vs_get_prescaler_backoff_confirm
  99. {
  100. struct ethernet_hdr ethernet;
  101. struct qualcomm_hdr qualcomm;
  102. uint8_t MSTATUS;
  103. uint8_t NUMOFBANDSPRIMARY;
  104. banddesc BANDDESC_PRI[16];
  105. uint8_t NUMOFBANDSALTERNATE;
  106. banddesc BANDDESC_ALT[16];
  107. }
  108. * confirm = (struct vs_get_prescaler_backoff_confirm *) (message);
  109. #ifndef __GNUC__
  110. #pragma pack (pop)
  111. #endif
  112. uint8_t i;
  113. memset (message, 0, sizeof (* message));
  114. EthernetHeader (& request->ethernet, channel->peer, channel->host, channel->type);
  115. QualcommHeader (& request->qualcomm, 0, (VS_GET_PRESCALER_BACKOFF | MMTYPE_REQ));
  116. plc->packetsize = (ETHER_MIN_LEN - ETHER_CRC_LEN);
  117. if (SendMME (plc) <= 0)
  118. {
  119. error (PLC_EXIT (plc), errno, CHANNEL_CANTSEND);
  120. return (-1);
  121. }
  122. if (ReadMME (plc, 0, (VS_GET_PRESCALER_BACKOFF | MMTYPE_CNF)) <= 0)
  123. {
  124. error (PLC_EXIT (plc), errno, CHANNEL_CANTREAD);
  125. return (-1);
  126. }
  127. if (confirm->MSTATUS)
  128. {
  129. Failure (plc, PLC_WONTDOIT);
  130. return (-1);
  131. }
  132. if (confirm->NUMOFBANDSPRIMARY) {
  133. printf("Frequency bands in Primary chain\n");
  134. printf("Band | Start index | Start back off(dB) | End index | End back off(dB)\n");
  135. for (i=0; i<confirm->NUMOFBANDSPRIMARY; i++)
  136. {
  137. printf("%d", i);
  138. printf("%10d", confirm->BANDDESC_PRI[i].startidx);
  139. printf("%#15x", confirm->BANDDESC_PRI[i].startbackoffdB);
  140. printf("%20d", confirm->BANDDESC_PRI[i].endidx);
  141. printf("%#15x\n", confirm->BANDDESC_PRI[i].endbackoffdB);
  142. }
  143. } else
  144. printf("No frequency bands in Primary chain\n");
  145. if (confirm->NUMOFBANDSALTERNATE) {
  146. printf("Frequency bands in Alternate chain\n");
  147. printf("Band | Start index | Start back off(dB) | End index | End back off(dB)\n");
  148. for (i=0; i<confirm->NUMOFBANDSALTERNATE; i++)
  149. {
  150. printf("%d", i);
  151. printf("%10d", confirm->BANDDESC_ALT[i].startidx);
  152. printf("%#15x", confirm->BANDDESC_ALT[i].startbackoffdB);
  153. printf("%20d", confirm->BANDDESC_ALT[i].endidx);
  154. printf("%#15x\n", confirm->BANDDESC_ALT[i].endbackoffdB);
  155. }
  156. } else
  157. printf("No frequency bands in Alternate chain\n");
  158. return 0;
  159. }
  160. /*====================================================================*
  161. *
  162. * signed setprescalervalues (struct plc * plc, uint8_t numofbands);
  163. *
  164. *===================================================================*/
  165. static signed setprescalervalues (struct plc * plc, void* memory, uint8_t numofbands)
  166. {
  167. struct channel * channel = (struct channel *) (plc->channel);
  168. struct message * message = (struct message *) (plc->message);
  169. struct timeval ts;
  170. struct timeval tc;
  171. unsigned timer = 0;
  172. #ifndef __GNUC__
  173. #pragma pack (push,1)
  174. #endif
  175. typedef struct {
  176. uint16_t startidx;
  177. uint8_t startbackoffdB;
  178. uint16_t endidx;
  179. uint8_t endbackoffdB;
  180. } banddesc;
  181. struct __packed vs_set_prescaler_backoff_request
  182. {
  183. struct ethernet_hdr ethernet;
  184. struct qualcomm_hdr qualcomm;
  185. uint8_t COUPLING;
  186. uint8_t NUMOFFREQBANDS;
  187. banddesc BANDDESCRIPTOR[16];
  188. uint8_t RESERVED[4];
  189. }
  190. * request = (struct vs_set_prescaler_backoff_request *) (message);
  191. struct __packed vs_set_prescaler_backoff_confirm
  192. {
  193. struct ethernet_hdr ethernet;
  194. struct qualcomm_hdr qualcomm;
  195. uint8_t MSTATUS;
  196. }
  197. * confirm = (struct vs_set_prescaler_backoff_confirm *) (message);
  198. struct __packed vs_set_prescaler_backoff_ind
  199. {
  200. struct ethernet_hdr ethernet;
  201. struct qualcomm_hdr qualcomm;
  202. uint8_t MSTATUS;
  203. uint8_t NUMOFBANDSPRIMARY;
  204. banddesc BANDDESC_PRI[16];
  205. uint8_t NUMOFBANDSALTERNATE;
  206. banddesc BANDDESC_ALT[16];
  207. }
  208. * indicate = (struct vs_set_prescaler_backoff_ind *) (message);
  209. #ifndef __GNUC__
  210. #pragma pack (pop)
  211. #endif
  212. uint8_t i;
  213. int length = numofbands * 6;
  214. bool rcvd = 0;
  215. memset (message, 0, sizeof (* message));
  216. EthernetHeader (& request->ethernet, channel->peer, channel->host, channel->type);
  217. QualcommHeader (& request->qualcomm, 0, (VS_SET_PRESCALER_BACKOFF | MMTYPE_REQ));
  218. plc->packetsize = sizeof (struct vs_set_prescaler_backoff_request);
  219. request->COUPLING = plc->coupling;
  220. request->NUMOFFREQBANDS = numofbands;
  221. memcpy (request->BANDDESCRIPTOR, (byte *)memory, length);
  222. if (SendMME (plc) <= 0)
  223. {
  224. error (PLC_EXIT (plc), errno, CHANNEL_CANTSEND);
  225. return (-1);
  226. }
  227. if (ReadMME (plc, 0, (VS_SET_PRESCALER_BACKOFF | MMTYPE_CNF)) <= 0)
  228. {
  229. error (PLC_EXIT (plc), errno, CHANNEL_CANTREAD);
  230. return (-1);
  231. }
  232. if (confirm->MSTATUS)
  233. {
  234. Failure (plc, PLC_WONTDOIT);
  235. return (-1);
  236. }
  237. if (gettimeofday (& ts, NULL) == -1)
  238. {
  239. error (1, errno, CANT_START_TIMER);
  240. }
  241. for (timer = 0; timer < 30; timer = SECONDS (ts, tc)) //wait 30 secs for indication
  242. {
  243. if (ReadMME (plc, 0, (VS_SET_PRESCALER_BACKOFF | MMTYPE_IND)) > 0)
  244. {
  245. rcvd = 1;
  246. break;
  247. }
  248. if (gettimeofday (& tc, NULL) == -1)
  249. {
  250. error (1, errno, CANT_RESET_TIMER);
  251. }
  252. }
  253. if(!rcvd)
  254. {
  255. error (PLC_EXIT (plc), errno, "IND MME not received");
  256. return (-1);
  257. }
  258. if (indicate->MSTATUS == 0x00)
  259. {
  260. printf("Back-off Applied Successfully\n");
  261. } else if(indicate->MSTATUS == 0x05) {
  262. printf("Previous Backoff Restored\n");
  263. } else if (indicate->MSTATUS == 0x07) {
  264. error (PLC_EXIT (plc), errno, "Backoff Request Failed");
  265. return (-1);
  266. }
  267. if (indicate->NUMOFBANDSPRIMARY) {
  268. printf("Frequency bands in Primary chain\n");
  269. printf("Band | Start index | Start back off(dB) | End index | End back off(dB)\n");
  270. for (i=0; i<indicate->NUMOFBANDSPRIMARY; i++)
  271. {
  272. printf("%d", i);
  273. printf("%10d", indicate->BANDDESC_PRI[i].startidx);
  274. printf("%#15x", indicate->BANDDESC_PRI[i].startbackoffdB);
  275. printf("%20d", indicate->BANDDESC_PRI[i].endidx);
  276. printf("%#15x\n", indicate->BANDDESC_PRI[i].endbackoffdB);
  277. }
  278. } else
  279. printf("No frequency bands in Primary chain\n");
  280. if (indicate->NUMOFBANDSALTERNATE) {
  281. printf("Frequency bands in Alternate chain\n");
  282. printf("Band | Start index | Start back off(dB) | End index | End back off(dB)\n");
  283. for (i=0; i<indicate->NUMOFBANDSALTERNATE; i++)
  284. {
  285. printf("%d", i);
  286. printf("%10d", indicate->BANDDESC_ALT[i].startidx);
  287. printf("%#15x", indicate->BANDDESC_ALT[i].startbackoffdB);
  288. printf("%20d", indicate->BANDDESC_ALT[i].endidx);
  289. printf("%#15x\n", indicate->BANDDESC_ALT[i].endbackoffdB);
  290. }
  291. } else
  292. printf("No frequency bands in Alternate chain\n");
  293. return (0);
  294. }
  295. /*====================================================================*
  296. *
  297. * void manager (struct plc * plc, uint8_t numofbands);
  298. *
  299. *===================================================================*/
  300. void manager (struct plc * plc, void* memory, uint8_t numofbands)
  301. {
  302. if (_anyset (plc->flags, PLC_SET_PRESCALER))
  303. {
  304. setprescalervalues (plc, memory, numofbands);
  305. } else {
  306. getprescalervalues (plc);
  307. }
  308. return;
  309. }
  310. /*====================================================================*
  311. *
  312. * int main (int argc, char const * argv[]);
  313. *
  314. *--------------------------------------------------------------------*/
  315. int main (int argc, char const * argv [])
  316. {
  317. extern struct channel channel;
  318. static char const * optv [] =
  319. {
  320. "c:egi:n:qs:t:vx",
  321. "device [device] [...]",
  322. "Qualcomm QCA75xx prescaler control MME utility",
  323. "c n\tcoupling [" LITERAL (PLCOUPLING) "]",
  324. "e\tredirect stderr to stdout",
  325. "g\tget prescaler values",
  326. #if defined (WINPCAP) || defined (LIBPCAP)
  327. "i n\thost interface is (n) [" LITERAL (CHANNEL_ETHNUMBER) "]",
  328. #else
  329. "i s\thost interface is (s) [" LITERAL (CHANNEL_ETHDEVICE) "]",
  330. #endif
  331. "n\tnumber of frequency bands",
  332. "q\tquiet mode",
  333. "s f\tset prescaler values from input file (f)\n\tinput file should contain only desriptor values",
  334. "t n\tread timeout is (n) milliseconds [" LITERAL (CHANNEL_TIMEOUT) "]",
  335. "v\tverbose mode",
  336. "x\texit on error",
  337. (char const *) (0)
  338. };
  339. static const struct _term_ coupling [] =
  340. {
  341. {
  342. "alt",
  343. "1"
  344. },
  345. {
  346. "mimo",
  347. "2"
  348. },
  349. {
  350. "pri",
  351. "0"
  352. }
  353. };
  354. #include "../plc/plc.c"
  355. FILE * fp;
  356. byte memory [96];
  357. uint8_t numofbands = 0;
  358. signed c;
  359. if (getenv (PLCDEVICE))
  360. {
  361. #if defined (WINPCAP) || defined (LIBPCAP)
  362. channel.ifindex = atoi (getenv (PLCDEVICE));
  363. #else
  364. channel.ifname = strdup (getenv (PLCDEVICE));
  365. #endif
  366. }
  367. optind = 1;
  368. memset (memory, 0, sizeof (memory));
  369. while (~ (c = getoptv (argc, argv, optv)))
  370. {
  371. switch (c)
  372. {
  373. case 'c':
  374. plc.coupling = (byte) (uintspec (synonym (optarg, coupling, SIZEOF (coupling)), 0, UCHAR_MAX));
  375. break;
  376. case 'e':
  377. dup2 (STDOUT_FILENO, STDERR_FILENO);
  378. break;
  379. case 'g':
  380. _setbits (plc.flags, PLC_GET_PRESCALER);
  381. break;
  382. case 'i':
  383. #if defined (WINPCAP) || defined (LIBPCAP)
  384. channel.ifindex = atoi (optarg);
  385. #else
  386. channel.ifname = optarg;
  387. #endif
  388. break;
  389. case 'n':
  390. numofbands = (signed) (uintspec (optarg, 0, UINT_MAX));
  391. break;
  392. case 'q':
  393. _setbits (channel.flags, CHANNEL_SILENCE);
  394. _setbits (plc.flags, PLC_SILENCE);
  395. break;
  396. case 's':
  397. if (! (fp = fopen (optarg, "rb")))
  398. {
  399. error (1, errno, "%s", optarg);
  400. }
  401. hexload (memory, sizeof (memory), fp);
  402. fclose (fp);
  403. _setbits (plc.flags, PLC_SET_PRESCALER);
  404. break;
  405. case 't':
  406. channel.timeout = (signed) (uintspec (optarg, 0, UINT_MAX));
  407. break;
  408. case 'v':
  409. _setbits (channel.flags, CHANNEL_VERBOSE);
  410. _setbits (plc.flags, PLC_VERBOSE);
  411. break;
  412. case 'x':
  413. _setbits (plc.flags, PLC_BAILOUT);
  414. break;
  415. default:
  416. break;
  417. }
  418. }
  419. if (optind == 1)
  420. exit (0);
  421. if (_anyset (plc.flags, PLC_SET_PRESCALER))
  422. {
  423. if (!numofbands)
  424. error (1, errno, "missing mandatory arguments");
  425. }
  426. argc -= optind;
  427. argv += optind;
  428. openchannel (& channel);
  429. if (!(plc.message = malloc (sizeof (* plc.message))))
  430. {
  431. error (1, errno, PLC_NOMEMORY);
  432. }
  433. if (! argc)
  434. {
  435. manager (& plc, memory, numofbands);
  436. }
  437. while ((argc) && (* argv))
  438. {
  439. if (! hexencode (channel.peer, sizeof (channel.peer), synonym (* argv, devices, SIZEOF (devices))))
  440. {
  441. error (1, errno, PLC_BAD_MAC, * argv);
  442. }
  443. manager (& plc, memory, numofbands);
  444. argc--;
  445. argv++;
  446. }
  447. free (plc.message);
  448. closechannel (& channel);
  449. exit (0);
  450. return (0);
  451. }