gfamininetmst.c 26 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <string.h>
  4. #include <malloc.h>
  5. #include <time.h>
  6. #include <fcntl.h>
  7. #include <errno.h>
  8. #include <termios.h>
  9. #include <unistd.h>
  10. #include <ctype.h>
  11. #include <sys/file.h>
  12. #include "gfamininetmst.h"
  13. #include "dbghlp.h"
  14. /////////////////////////////////////////////////////////////////////////////
  15. // https://www.racom.eu/eng/support/prot/mininet/index.html
  16. /////////////////////////////////////////////////////////////////////////////
  17. #define STX ((uint8_t)0x02)
  18. #define ACK ((uint8_t)0x06)
  19. #define SLAVE_IND ((uint8_t)0x80)
  20. #define START_INDEX 0x40
  21. #define MIN_INDEX 0x3F
  22. #define MAX_INDEX 0x7F
  23. #define RESET_INDEX MIN_INDEX
  24. #define MAX_DATA_PAYLOAD_LENGTH 250
  25. #define MAX_SAVE_TIMOUT_DEPTH 8
  26. #define _TRACE_UNEXPECTED 1
  27. /////////////////////////////////////////////////////////////////////////////
  28. typedef enum _GfaMininetRxStates
  29. {
  30. GfaRxMNS_Stx,
  31. GfaRxMNS_Len,
  32. GfaRxMNS_Node,
  33. GfaRxMNS_Index,
  34. GfaRxMNS_Data,
  35. GfaRxMNS_Zero,
  36. GfaRxMNS_Check
  37. }GfaMininetRxStates, *LPGfaMininetRxStates;
  38. #if _TRACE_UNEXPECTED
  39. static const char *g_pszExpected[] =
  40. {
  41. "STX or ACK",
  42. "Length (>= 5)",
  43. "Node Addr.",
  44. "Index (< 0x3F)",
  45. "Data",
  46. "STX-Zero",
  47. "CRC32"
  48. };
  49. #define TRACE_UNEXPECTED(p, s, b) if(p->nVerbosity >= 4) \
  50. fprintf(p->pDumpCtx, "%s: Expected: %s - Received: 0x%02hhX\n", __FUNCTION__, g_pszExpected[s], b)
  51. #else // _TRACE_UNEXPECTED
  52. #define TRACE_UNEXPECTED(p, s, b) (void)0
  53. #endif // _TRACE_UNEXPECTED
  54. /////////////////////////////////////////////////////////////////////////////
  55. typedef struct _GFA_MININET_MASTER
  56. {
  57. HMINETDEV hDev;
  58. struct timeval tvRXSave[MAX_SAVE_TIMOUT_DEPTH];
  59. struct timeval tvTXSave[MAX_SAVE_TIMOUT_DEPTH];
  60. struct timespec tsStart;
  61. int nVerbosity;
  62. int nSaveToIndex;
  63. FILE *pDumpCtx;
  64. uint8_t nIndexTable[256];
  65. }GFA_MININET_MASTER, *LPGFA_MININET_MASTER;
  66. typedef const GFA_MININET_MASTER *LPCGFA_MININET_MASTER;
  67. /////////////////////////////////////////////////////////////////////////////
  68. static bool _IsValidIndex(uint8_t nIndex)
  69. {
  70. return ((nIndex >= MIN_INDEX) && (nIndex <= MAX_INDEX));
  71. }
  72. /////////////////////////////////////////////////////////////////////////////
  73. static uint8_t _GetNextIndex(HGFAMINEMST hMst, uint8_t nNode)
  74. {
  75. if(hMst)
  76. {
  77. LPGFA_MININET_MASTER pMst = (LPGFA_MININET_MASTER)hMst;
  78. uint8_t idx = ++pMst->nIndexTable[nNode];
  79. if(idx > MAX_INDEX)
  80. {
  81. pMst->nIndexTable[nNode] = START_INDEX;
  82. idx = pMst->nIndexTable[nNode];
  83. }
  84. return idx;
  85. }
  86. return 0xFF;
  87. }
  88. /////////////////////////////////////////////////////////////////////////////
  89. static uint8_t _GetCurIndex(HGFAMINEMST hMst, uint8_t nNode)
  90. {
  91. if(hMst)
  92. {
  93. LPGFA_MININET_MASTER pMst = (LPGFA_MININET_MASTER)hMst;
  94. return pMst->nIndexTable[nNode];
  95. }
  96. return 0xFF;
  97. }
  98. /////////////////////////////////////////////////////////////////////////////
  99. static int _CountStxBytes(const void *pData, size_t nCbData)
  100. {
  101. int nStx = 0;
  102. const uint8_t *pbData = (const uint8_t*)pData;
  103. uint8_t b, n = 0;
  104. while(nCbData > 0)
  105. {
  106. b = *pbData++;
  107. if((n == STX) && (b == 0))
  108. {
  109. ++nStx;
  110. ++nCbData;
  111. }
  112. n = b;
  113. --nCbData;
  114. }
  115. return nStx;
  116. }
  117. /////////////////////////////////////////////////////////////////////////////
  118. HGFAMINEMST GfaMininetMasterOpen(LPCGFA_MININET_MST_CFG_PARAMS pmmcp)
  119. {
  120. if(pmmcp)
  121. {
  122. HMINETDEV hDev = GfaMininetDeviceOpen(&pmmcp->devcfg);
  123. if(hDev)
  124. {
  125. LPGFA_MININET_MASTER pMst = malloc(sizeof(GFA_MININET_MASTER));
  126. memset(pMst, 0, sizeof(GFA_MININET_MASTER));
  127. memset(pMst->nIndexTable, START_INDEX, sizeof(pMst->nIndexTable));
  128. pMst->hDev = hDev;
  129. pMst->pDumpCtx = stdout;
  130. clock_gettime(CLOCK_MONOTONIC, &pMst->tsStart);
  131. return (HGFAMINEMST)pMst;
  132. }
  133. return NULL;
  134. }
  135. errno = EINVAL;
  136. return NULL;
  137. }
  138. /////////////////////////////////////////////////////////////////////////////
  139. void GfaMininetMasterClose(HGFAMINEMST hMst)
  140. {
  141. if(hMst)
  142. {
  143. LPGFA_MININET_MASTER pMst = (LPGFA_MININET_MASTER)hMst;
  144. GfaMininetDeviceClose(pMst->hDev);
  145. free(pMst);
  146. }
  147. }
  148. /////////////////////////////////////////////////////////////////////////////
  149. bool GfaMininetMasterGetTimeouts(HGFAMINEMST hMst, struct timeval *ptvRX, struct timeval *ptvTX)
  150. {
  151. if(hMst)
  152. {
  153. LPGFA_MININET_MASTER pMst = (LPGFA_MININET_MASTER)hMst;
  154. return GfaMininetDeviceGetTimeouts(pMst->hDev, ptvRX, ptvTX);
  155. }
  156. errno = EINVAL;
  157. return false;
  158. }
  159. /////////////////////////////////////////////////////////////////////////////
  160. bool GfaMininetMasterSetTimeouts(HGFAMINEMST hMst, const struct timeval *ptvRX, const struct timeval *ptvTX)
  161. {
  162. if(hMst)
  163. {
  164. LPGFA_MININET_MASTER pMst = (LPGFA_MININET_MASTER)hMst;
  165. return GfaMininetDeviceSetTimeouts(pMst->hDev, ptvRX, ptvTX);
  166. }
  167. errno = EINVAL;
  168. return false;
  169. }
  170. /////////////////////////////////////////////////////////////////////////////
  171. bool GfaMininetMasterSaveTimeouts(HGFAMINEMST hMst)
  172. {
  173. if(hMst)
  174. {
  175. LPGFA_MININET_MASTER pMst = (LPGFA_MININET_MASTER)hMst;
  176. if(pMst->nSaveToIndex < MAX_SAVE_TIMOUT_DEPTH)
  177. {
  178. int nIndex = pMst->nSaveToIndex++;
  179. return GfaMininetMasterGetTimeouts(hMst, &pMst->tvRXSave[nIndex], &pMst->tvTXSave[nIndex]);
  180. }
  181. else
  182. {
  183. errno = EPERM;
  184. return false;
  185. }
  186. }
  187. errno = EINVAL;
  188. return false;
  189. }
  190. /////////////////////////////////////////////////////////////////////////////
  191. bool GfaMininetMasterRestoreTimeouts(HGFAMINEMST hMst)
  192. {
  193. if(hMst)
  194. {
  195. LPGFA_MININET_MASTER pMst = (LPGFA_MININET_MASTER)hMst;
  196. if(pMst->nSaveToIndex > 0)
  197. {
  198. int nIndex = --pMst->nSaveToIndex;
  199. return GfaMininetMasterSetTimeouts(hMst, &pMst->tvRXSave[nIndex], &pMst->tvTXSave[nIndex]);
  200. }
  201. else
  202. {
  203. errno = EPERM;
  204. return false;
  205. }
  206. }
  207. errno = EINVAL;
  208. return false;
  209. }
  210. /////////////////////////////////////////////////////////////////////////////
  211. ssize_t GfaMininetMasterGetConfigParams(HGFAMINEMST hMst, void *pDevParams, size_t nSizeDevParams)
  212. {
  213. if(hMst)
  214. {
  215. LPGFA_MININET_MASTER pMst = (LPGFA_MININET_MASTER)hMst;
  216. return GfaMininetDeviceGetConfigParams(pMst->hDev, pDevParams, nSizeDevParams);
  217. }
  218. errno = EINVAL;
  219. return false;
  220. }
  221. /////////////////////////////////////////////////////////////////////////////
  222. int GfaMininetMasterSetConfigParams(HGFAMINEMST hMst, const void *pDevParams, size_t nSizeDevParams)
  223. {
  224. if(hMst)
  225. {
  226. LPGFA_MININET_MASTER pMst = (LPGFA_MININET_MASTER)hMst;
  227. return GfaMininetDeviceSetConfigParams(pMst->hDev, pDevParams, nSizeDevParams);
  228. }
  229. errno = EINVAL;
  230. return false;
  231. }
  232. /////////////////////////////////////////////////////////////////////////////
  233. bool GfaMininetMasterIsValidBaudrate(HGFAMINEMST hMst, uint32_t nBaudrate)
  234. {
  235. if(hMst)
  236. {
  237. LPGFA_MININET_MASTER pMst = (LPGFA_MININET_MASTER)hMst;
  238. return GfaMininetDeviceIsValidBaudrate(pMst->hDev, nBaudrate);
  239. }
  240. errno = EINVAL;
  241. return false;
  242. }
  243. /////////////////////////////////////////////////////////////////////////////
  244. int GfaMininetMasterGetBaudrate(HGFAMINEMST hMst, uint32_t *pnBaudrate)
  245. {
  246. if(hMst)
  247. {
  248. LPGFA_MININET_MASTER pMst = (LPGFA_MININET_MASTER)hMst;
  249. return GfaMininetDeviceGetBaudrate(pMst->hDev, pnBaudrate);
  250. }
  251. errno = EINVAL;
  252. return -1;
  253. }
  254. /////////////////////////////////////////////////////////////////////////////
  255. int GfaMininetMasterSetBaudrate(HGFAMINEMST hMst, uint32_t nBaudrate)
  256. {
  257. if(hMst)
  258. {
  259. LPGFA_MININET_MASTER pMst = (LPGFA_MININET_MASTER)hMst;
  260. return GfaMininetDeviceSetBaudrate(pMst->hDev, nBaudrate);
  261. }
  262. errno = EINVAL;
  263. return -1;
  264. }
  265. /////////////////////////////////////////////////////////////////////////////
  266. HMINETDEV GfaMininetMasterGetDeviceHandle(HGFAMINEMST hMst)
  267. {
  268. if(hMst)
  269. {
  270. LPGFA_MININET_MASTER pMst = (LPGFA_MININET_MASTER)hMst;
  271. return pMst->hDev;
  272. }
  273. errno = EINVAL;
  274. return NULL;
  275. }
  276. /////////////////////////////////////////////////////////////////////////////
  277. uint8_t GfaMininetMasterCalcChk(const void *pData, size_t nCbData, bool bProcessSTX)
  278. {
  279. size_t i = 0;
  280. uint16_t chk = 0;
  281. uint8_t cc, lc = 0;
  282. const uint8_t *pszData = (const uint8_t*)pData;
  283. for(i = 0; i < nCbData; ++i)
  284. {
  285. cc = *pszData++;
  286. if(cc || (lc != STX))
  287. {
  288. chk <<= 1;
  289. if(chk & 0x0100)
  290. chk = (chk + 1) & 0x00FF;
  291. chk += cc;
  292. if(chk > 0x00FF)
  293. chk = (chk + 1) & 0x00FF;
  294. }
  295. if(bProcessSTX && (i > 3))
  296. lc = cc;
  297. }
  298. if((uint8_t)chk == STX)
  299. chk = ~chk;
  300. return (uint8_t)chk;
  301. }
  302. /////////////////////////////////////////////////////////////////////////////
  303. size_t GfaMininetMasterBuildFrame(HGFAMINEMST hMst, uint8_t nNode, uint8_t nIndex, const void *pDataPayload, size_t nCbDataPayload, void *pFrameBuffer, size_t nCbFrameBuffer)
  304. {
  305. if(hMst && pFrameBuffer && (nCbFrameBuffer >= 5) && (nCbDataPayload <= MAX_DATA_PAYLOAD_LENGTH))
  306. {
  307. int i = 0;
  308. uint8_t b, c;
  309. size_t nLen = 4;
  310. const uint8_t *pszData = (const uint8_t*)pDataPayload;
  311. LPGFA_MININET_FRAME pf = (LPGFA_MININET_FRAME)pFrameBuffer;
  312. pf->stx = STX;
  313. pf->len = 5 + nCbDataPayload;
  314. pf->node = nNode;
  315. pf->index = _IsValidIndex(nIndex) ? nIndex : _GetNextIndex(hMst, nNode);
  316. if(pszData && nCbDataPayload)
  317. {
  318. while(nCbDataPayload--)
  319. {
  320. b = *pszData++;
  321. do
  322. {
  323. if(++nLen == nCbFrameBuffer)
  324. {
  325. errno = ENOMEM;
  326. return 0;
  327. }
  328. pf->data.by[i++] = b;
  329. c = b;
  330. b = 0;
  331. }
  332. while(c == STX);
  333. }
  334. }
  335. pf->data.by[i++] = GfaMininetMasterCalcChk(pFrameBuffer, nLen, true);
  336. ++nLen;
  337. return nLen;
  338. }
  339. errno = EINVAL;
  340. return 0;
  341. }
  342. /////////////////////////////////////////////////////////////////////////////
  343. int GfaMininetMasterResetSlaveIndex(HGFAMINEMST hMst, uint8_t nNode)
  344. {
  345. if(hMst)
  346. {
  347. LPGFA_MININET_MASTER pMst = (LPGFA_MININET_MASTER)hMst;
  348. ssize_t nLen, nRet;
  349. uint8_t nIndex;
  350. uint8_t txb[16];
  351. uint8_t rxb[16];
  352. nLen = GfaMininetMasterBuildFrame(hMst, nNode, RESET_INDEX, "\x1b\x52", 2, txb, sizeof(txb));
  353. if((nRet = GfaMininetMasterTransmitFrame(hMst, txb, nLen)) != nLen)
  354. {
  355. GfaMininetDevicePurgeRXBuffer(pMst->hDev);
  356. return -1;
  357. }
  358. if(!NODE_IS_BROADCAST(nNode) && !NODE_IS_GROUPCAST(nNode))
  359. {
  360. // we expect a response
  361. if((nRet = GfaMininetMasterReceiveFrame(hMst, rxb, sizeof(rxb), false)) <= 0)
  362. return -1;
  363. if((nRet = GfaMininetMasterEvaluateSlaveResponse(hMst, nNode, rxb, nRet, false, &nIndex)) == MINET_SLAVE_RESPONSE_INDEX_IS_STATUS_CODE)
  364. {
  365. if(nIndex == MINET_SLAVE_STATUS_INDEX_RESET_DONE)
  366. return GfaMininetMasterResetLocalIndex(hMst, nNode);
  367. else
  368. {
  369. errno = nIndex;
  370. nRet = -1;
  371. }
  372. }
  373. else
  374. {
  375. errno = EPROTO;
  376. nRet = -1;
  377. }
  378. }
  379. else
  380. {
  381. // we don't expect a response. so let's just hope that all slaves have received the request and have performed an index reset!
  382. return GfaMininetMasterResetLocalIndex(hMst, nNode);
  383. }
  384. return nRet;
  385. }
  386. errno = EINVAL;
  387. return -1;
  388. }
  389. /////////////////////////////////////////////////////////////////////////////
  390. int GfaMininetMasterResetLocalIndex(HGFAMINEMST hMst, uint8_t nNode)
  391. {
  392. if(hMst)
  393. {
  394. LPGFA_MININET_MASTER pMst = (LPGFA_MININET_MASTER)hMst;
  395. if(NODE_IS_BROADCAST(nNode))
  396. memset(pMst->nIndexTable, START_INDEX, sizeof(pMst->nIndexTable));
  397. else if(NODE_IS_GROUPCAST(nNode))
  398. memset(&pMst->nIndexTable[nNode], START_INDEX, 10);
  399. else
  400. pMst->nIndexTable[nNode] = START_INDEX;
  401. return 0;
  402. }
  403. errno = EINVAL;
  404. return -1;
  405. }
  406. /////////////////////////////////////////////////////////////////////////////
  407. int GfaMininetMasterPurgeDeviceRXBuffer(HGFAMINEMST hMst)
  408. {
  409. if(hMst)
  410. {
  411. LPGFA_MININET_MASTER pMst = (LPGFA_MININET_MASTER)hMst;
  412. return GfaMininetDevicePurgeRXBuffer(pMst->hDev);
  413. }
  414. errno = EINVAL;
  415. return -1;
  416. }
  417. /////////////////////////////////////////////////////////////////////////////
  418. int GfaMininetMasterPingSlave(HGFAMINEMST hMst, uint8_t nNode)
  419. {
  420. if(hMst)
  421. {
  422. uint8_t nIndex;
  423. ssize_t nRet, nLen;
  424. char txb[32], rxb[32];
  425. nLen = GfaMininetMasterBuildFrame(hMst, nNode, 0, NULL, 0, txb, sizeof(txb));
  426. if((nRet = GfaMininetMasterTransmitFrame(hMst, txb, nLen)) < nLen)
  427. return -1;
  428. if((nLen = GfaMininetMasterReceiveFrame(hMst, rxb, sizeof(rxb), true)) <= 0)
  429. return -1;
  430. nRet = GfaMininetMasterEvaluateSlaveResponse(hMst, nNode, rxb, nLen, true, &nIndex);
  431. if(nRet == MINET_SLAVE_RESPONSE_INDEX_IS_STATUS_CODE)
  432. {
  433. if(nIndex == MINET_SLAVE_STATUS_INDEX_INVALID_PARAM)
  434. return 0;
  435. else
  436. {
  437. errno = -(int)nIndex;
  438. return -1;
  439. }
  440. }
  441. return -1;
  442. }
  443. errno = EINVAL;
  444. return -1;
  445. }
  446. /////////////////////////////////////////////////////////////////////////////
  447. int GfaMininetMasterEvaluateSlaveResponse(HGFAMINEMST hMst, uint8_t nNode, const void *pFrame, size_t nCbFrame, bool bAckPossible, uint8_t *pbIndex)
  448. {
  449. if(hMst && pFrame)
  450. {
  451. bool bIsStatusIndex = false;
  452. LPCGFA_MININET_FRAME pf = (LPCGFA_MININET_FRAME)pFrame;
  453. /////////////////////////////////////////////////////////////////////
  454. if(nCbFrame == 0)
  455. {
  456. errno = MINET_SLAVE_RESPONSE_ERROR_INVALID_LENGTH;
  457. return -1;
  458. }
  459. else if(nCbFrame < 6)
  460. {
  461. if(bAckPossible && (pf->stx == ACK))
  462. {
  463. return MINET_SLAVE_RESPONSE_ACK;
  464. }
  465. else
  466. {
  467. errno = MINET_SLAVE_RESPONSE_ERROR_INVALID_LENGTH;
  468. return -1;
  469. }
  470. }
  471. else
  472. {
  473. /////////////////////////////////////////////////////////////////
  474. // STX / ACK
  475. if(bAckPossible && (pf->stx == ACK))
  476. return MINET_SLAVE_RESPONSE_ACK;
  477. else if(pf->stx != STX)
  478. {
  479. errno = MINET_SLAVE_RESPONSE_ERROR_STX_ERROR;
  480. return -1;
  481. }
  482. /////////////////////////////////////////////////////////////////
  483. // Length
  484. if(pf->len > nCbFrame)
  485. {
  486. errno = MINET_SLAVE_RESPONSE_ERROR_INCOMPLETE_DATA;
  487. return -1;
  488. }
  489. /////////////////////////////////////////////////////////////////
  490. // Node address
  491. if(pf->node != (nNode & 0xF0))
  492. {
  493. errno = MINET_SLAVE_RESPONSE_ERROR_INVALID_NODE_ADDRESS;
  494. return -1;
  495. }
  496. /////////////////////////////////////////////////////////////////
  497. // Index range / match / status code
  498. if(pbIndex)
  499. *pbIndex = pf->index;
  500. if(_IsValidIndex(pf->index))
  501. {
  502. uint8_t nIndex = _GetCurIndex(hMst, nNode);
  503. if(pf->index != nIndex)
  504. {
  505. errno = MINET_SLAVE_RESPONSE_ERROR_INDEX_NO_MATCH;
  506. return -1;
  507. }
  508. }
  509. else if(pf->index < MIN_INDEX)
  510. {
  511. errno = MINET_SLAVE_RESPONSE_ERROR_INDEX_OUT_OF_RANGE;
  512. return -1;
  513. }
  514. else // pf->index > MAX_INDEX
  515. {
  516. switch(pf->index)
  517. {
  518. case MINET_SLAVE_STATUS_INDEX_RESET_DONE:
  519. case MINET_SLAVE_STATUS_INDEX_ERROR:
  520. case MINET_SLAVE_STATUS_INDEX_CMD_ERROR:
  521. case MINET_SLAVE_STATUS_INDEX_INVALID_PARAM:
  522. case MINET_SLAVE_STATUS_INDEX_UNKNOWN_CMD:
  523. case MINET_SLAVE_STATUS_INDEX_CMD_ALREADY_EX:
  524. bIsStatusIndex = true;
  525. break;
  526. default:
  527. errno = MINET_SLAVE_RESPONSE_ERROR_INDEX_OUT_OF_RANGE;
  528. return -1;
  529. }
  530. }
  531. /////////////////////////////////////////////////////////////////
  532. // Data / Slave indicator
  533. if(pf->data.by[0] != SLAVE_IND)
  534. {
  535. errno = MINET_SLAVE_RESPONSE_ERROR_INVALID_SLAVE_INDICATOR;
  536. return -1;
  537. }
  538. /////////////////////////////////////////////////////////////////
  539. // Checksum
  540. if(pf->data.by[pf->len - 5] != GfaMininetMasterCalcChk(pf, pf->len - 1, false))
  541. {
  542. errno = MINET_SLAVE_RESPONSE_ERROR_INVALID_CHECKSUM;
  543. return -1;
  544. }
  545. else
  546. {
  547. return bIsStatusIndex ? MINET_SLAVE_RESPONSE_INDEX_IS_STATUS_CODE : MINET_SLAVE_RESPONSE_SUCCESS;
  548. }
  549. }
  550. }
  551. errno = EINVAL;
  552. return -1;
  553. }
  554. /////////////////////////////////////////////////////////////////////////////
  555. ssize_t GfaMininetMasterTransmitFrame(HGFAMINEMST hMst, const void *pData, size_t nCbData)
  556. {
  557. if(hMst && pData && nCbData)
  558. {
  559. LPGFA_MININET_MASTER pMst = (LPGFA_MININET_MASTER)hMst;
  560. GfaMininetMasterDumpFrame(hMst, pMst->pDumpCtx, (LPCGFA_MININET_FRAME)pData, true, NULL);
  561. return GfaMininetDeviceTransmit(pMst->hDev, pData, nCbData);
  562. }
  563. errno = EINVAL;
  564. return -1;
  565. }
  566. /////////////////////////////////////////////////////////////////////////////
  567. ssize_t GfaMininetMasterReceiveFrame(HGFAMINEMST hMst, void *pBuffer, size_t nCbBuffer, bool bAckPossible)
  568. {
  569. if(hMst && pBuffer && nCbBuffer)
  570. {
  571. uint8_t b, c, buf[256] = {0};
  572. bool bLoop = true;
  573. GfaMininetRxStates nState = GfaRxMNS_Stx;
  574. ssize_t nRet, nCbDataPayloadExp = 0, nCbDataPayloadRcv = 0;
  575. LPGFA_MININET_MASTER pMst = (LPGFA_MININET_MASTER)hMst;
  576. LPGFA_MININET_FRAME pFrameRx = (LPGFA_MININET_FRAME)buf;
  577. if(!pBuffer || !nCbBuffer)
  578. {
  579. errno = EINVAL;
  580. return -1;
  581. }
  582. do
  583. {
  584. switch(nState)
  585. {
  586. ////////////////////////////////////////////////////////////////////////
  587. // handle STX
  588. case GfaRxMNS_Stx:
  589. if((nRet = GfaMininetDevicePop(pMst->hDev, &b)) != 1)
  590. {
  591. bLoop = false;
  592. break;
  593. }
  594. if(b != STX)
  595. {
  596. if(bAckPossible && (b == ACK))
  597. {
  598. *(uint8_t*)pBuffer = b;
  599. nRet = 1;
  600. bLoop = false;
  601. GfaMininetMasterDumpFrame(hMst, pMst->pDumpCtx, (LPCGFA_MININET_FRAME)pBuffer, false, NULL);
  602. break;
  603. }
  604. else
  605. {
  606. TRACE_UNEXPECTED(pMst, nState, b);
  607. }
  608. continue;
  609. }
  610. pFrameRx->stx = b;
  611. ++nState;
  612. // fall through
  613. ////////////////////////////////////////////////////////////////////////
  614. // handle length
  615. case GfaRxMNS_Len:
  616. if((nRet = GfaMininetDevicePop(pMst->hDev, &b)) != 1)
  617. {
  618. bLoop = false;
  619. break;
  620. }
  621. if(b < 5)
  622. {
  623. TRACE_UNEXPECTED(pMst, nState, b);
  624. if(b == STX)
  625. continue;
  626. else
  627. {
  628. nState = GfaRxMNS_Stx;
  629. continue;
  630. }
  631. }
  632. pFrameRx->len = b;
  633. nCbDataPayloadExp = b - 5;
  634. ++nState;
  635. // fall through
  636. ////////////////////////////////////////////////////////////////////////
  637. // handle node
  638. case GfaRxMNS_Node:
  639. if((nRet = GfaMininetDevicePop(pMst->hDev, &b)) != 1)
  640. {
  641. bLoop = false;
  642. break;
  643. }
  644. if(b == STX)
  645. {
  646. TRACE_UNEXPECTED(pMst, nState, b);
  647. nState = GfaRxMNS_Len;
  648. continue;
  649. }
  650. pFrameRx->node = b;
  651. ++nState;
  652. // fall through
  653. ////////////////////////////////////////////////////////////////////////
  654. // handle index
  655. case GfaRxMNS_Index:
  656. if((nRet = GfaMininetDevicePop(pMst->hDev, &b)) != 1)
  657. {
  658. bLoop = false;
  659. break;
  660. }
  661. if(b == STX)
  662. {
  663. TRACE_UNEXPECTED(pMst, nState, b);
  664. nState = GfaRxMNS_Len;
  665. continue;
  666. }
  667. else if(b < MIN_INDEX)
  668. {
  669. TRACE_UNEXPECTED(pMst, nState, b);
  670. nState = GfaRxMNS_Stx;
  671. continue;
  672. }
  673. pFrameRx->index = b;
  674. ++nState;
  675. // fall through
  676. ////////////////////////////////////////////////////////////////////////
  677. // handle data, if any
  678. case GfaRxMNS_Data:
  679. if(nCbDataPayloadExp > nCbDataPayloadRcv)
  680. {
  681. if((nRet = GfaMininetDevicePop(pMst->hDev, &b)) != 1)
  682. {
  683. bLoop = false;
  684. break;
  685. }
  686. if(b == STX)
  687. ++nState;
  688. pFrameRx->data.by[nCbDataPayloadRcv++] = b;
  689. continue;
  690. }
  691. else
  692. {
  693. nState = GfaRxMNS_Check;
  694. continue;
  695. }
  696. ////////////////////////////////////////////////////////////////////////
  697. // handle 0
  698. case GfaRxMNS_Zero:
  699. if((nRet = GfaMininetDevicePop(pMst->hDev, &b)) != 1)
  700. {
  701. bLoop = false;
  702. break;
  703. }
  704. if(b == 0)
  705. {
  706. nState = GfaRxMNS_Data;
  707. continue;
  708. }
  709. else if(b == STX)
  710. {
  711. TRACE_UNEXPECTED(pMst, nState, b);
  712. nCbDataPayloadRcv = nCbDataPayloadExp = 0;
  713. nState = GfaRxMNS_Len;
  714. continue;
  715. }
  716. else if(b >= 5)
  717. {
  718. TRACE_UNEXPECTED(pMst, nState, b);
  719. pFrameRx->len = b;
  720. nCbDataPayloadRcv = 0;
  721. nCbDataPayloadExp = b - 5;
  722. nState = GfaRxMNS_Node;
  723. continue;
  724. }
  725. else
  726. {
  727. TRACE_UNEXPECTED(pMst, nState, b);
  728. nCbDataPayloadRcv = nCbDataPayloadExp = 0;
  729. nState = GfaRxMNS_Stx;
  730. continue;
  731. }
  732. ////////////////////////////////////////////////////////////////////////
  733. // handle Checksum
  734. case GfaRxMNS_Check:
  735. if((nRet = GfaMininetDevicePop(pMst->hDev, &b)) != 1)
  736. {
  737. bLoop = false;
  738. break;
  739. }
  740. if(b == STX)
  741. {
  742. TRACE_UNEXPECTED(pMst, nState, b);
  743. nCbDataPayloadRcv = nCbDataPayloadExp = 0;
  744. nState = GfaRxMNS_Len;
  745. continue;
  746. }
  747. c = GfaMininetMasterCalcChk(pFrameRx, pFrameRx->len - 1, false);
  748. if(b == c)
  749. {
  750. if((size_t)pFrameRx->len <= nCbBuffer)
  751. {
  752. pFrameRx->data.by[nCbDataPayloadRcv] = b;
  753. nRet = pFrameRx->len;
  754. memcpy(pBuffer, pFrameRx, nRet);
  755. GfaMininetMasterDumpFrame(hMst, pMst->pDumpCtx, (LPCGFA_MININET_FRAME)pFrameRx, false, NULL);
  756. }
  757. else
  758. {
  759. errno = ENOMEM;
  760. nRet = -1;
  761. }
  762. }
  763. else
  764. {
  765. pFrameRx->data.by[nCbDataPayloadRcv] = b;
  766. GfaMininetMasterDumpFrame(hMst, pMst->pDumpCtx, (LPCGFA_MININET_FRAME)pFrameRx, false, "CRC32!!!");
  767. errno = EPROTO;
  768. nRet = -1;
  769. }
  770. bLoop = false;
  771. break;
  772. ////////////////////////////////////////////////////////////////////////
  773. default:
  774. nRet = -1;
  775. bLoop = false;
  776. break;
  777. }
  778. }
  779. while(bLoop);
  780. return nRet;
  781. }
  782. errno = EINVAL;
  783. return -1;
  784. }
  785. /////////////////////////////////////////////////////////////////////////////
  786. ssize_t GfaMininetMasterGetDataFromFrame(const void *pFrame, size_t nCbFrame, void *pData, size_t nCbData)
  787. {
  788. if(pFrame && (nCbFrame >= 5) && pData && (nCbData > 0))
  789. {
  790. LPCGFA_MININET_FRAME pf = (LPCGFA_MININET_FRAME)pFrame;
  791. if(pf->len == 5)
  792. return 0; // no data
  793. else if(pf->len > 5)
  794. {
  795. if(nCbFrame >= pf->len)
  796. {
  797. size_t nDatalen = (size_t)pf->len - 5;
  798. if(nDatalen <= nCbData)
  799. {
  800. memcpy(pData, pf->data.by, nDatalen);
  801. return (ssize_t)nDatalen;
  802. }
  803. else
  804. {
  805. errno = ENOMEM;
  806. return -1;
  807. }
  808. }
  809. }
  810. else
  811. {
  812. errno = EPROTO;
  813. return -1;
  814. }
  815. }
  816. errno = EINVAL;
  817. return -1;
  818. }
  819. /////////////////////////////////////////////////////////////////////////////
  820. ssize_t GfaMininetMasterGetDataFromSlaveFrame(const void *pFrame, size_t nCbFrame, void *pData, size_t nCbData)
  821. {
  822. uint8_t data[256];
  823. ssize_t nRet = GfaMininetMasterGetDataFromFrame(pFrame, nCbFrame, data, sizeof(data));
  824. if(nRet > 0)
  825. {
  826. if(data[0] == SLAVE_IND)
  827. {
  828. if(--nRet <= (ssize_t)nCbData)
  829. {
  830. if(nRet > 0)
  831. memcpy(pData, &data[1], nRet);
  832. }
  833. else
  834. {
  835. errno = ENOMEM;
  836. nRet = -1;
  837. }
  838. }
  839. else
  840. {
  841. errno = EPROTO;
  842. nRet = -1;
  843. }
  844. }
  845. return nRet;
  846. }
  847. /////////////////////////////////////////////////////////////////////////////
  848. void GfaMininetMasterDumpFrame(HGFAMINEMST hMst, FILE *pf, LPCGFA_MININET_FRAME pFrame, bool bTX, const char *pszAnnotation)
  849. {
  850. if(hMst && pf)
  851. {
  852. LPGFA_MININET_MASTER pMst = (LPGFA_MININET_MASTER)hMst;
  853. if(pMst->nVerbosity >= 4)
  854. {
  855. struct timespec tsCur, tsIntv;
  856. clock_gettime(CLOCK_MONOTONIC, &tsCur);
  857. uint64_t nInterval = TimespecDiff(&tsCur, &pMst->tsStart);
  858. Ns2Timespec(nInterval, &tsIntv);
  859. if(pFrame)
  860. {
  861. if(pFrame->stx == ACK)
  862. {
  863. fprintf(pf, "\n///////////////////////////////////////\n");
  864. fprintf(pf, "// %ld.%03ld\n", tsIntv.tv_sec, tsIntv.tv_nsec / 1000000);
  865. fprintf(pf, "// %s:\n", bTX ? "TX" : "RX");
  866. fprintf(pf, "ACK: 06 (6)\n");
  867. }
  868. else if(pFrame->len >= 5)
  869. {
  870. int i, nCbData = pFrame->len - 5;
  871. fprintf(pf, "\n///////////////////////////////////////\n");
  872. fprintf(pf, "// %ld.%03ld\n", tsIntv.tv_sec, tsIntv.tv_nsec / 1000000);
  873. fprintf(pf, "// %s:\n", bTX ? "TX" : "RX");
  874. fprintf(pf, "STX: %02hhX (%hhu)\n", pFrame->stx, pFrame->stx);
  875. fprintf(pf, "Length: %02hhX (%hhu)\n", pFrame->len, pFrame->len);
  876. fprintf(pf, "Node: %02hhX (%hhu)\n", pFrame->node, pFrame->node);
  877. fprintf(pf, "Index: %02hhX (%hhu)\n", pFrame->index, pFrame->index);
  878. fprintf(pf, "Data: ");
  879. nCbData += _CountStxBytes(pFrame->data.by, nCbData);
  880. for(i = 0; i < nCbData; i++)
  881. {
  882. if((i < 2) && (pFrame->data.by[i] >= 0x41) && (pFrame->data.by[i] <= 0x5A))
  883. fprintf(pf, "<%c>", (char)pFrame->data.by[i]);
  884. else
  885. fprintf(pf, "[%02hhX]", (char)pFrame->data.by[i]);
  886. }
  887. fprintf(pf, "\nCRC: %02hhX (%hhu)\n", pFrame->data.by[i], pFrame->data.by[i]);
  888. if(pszAnnotation)
  889. fprintf(pf, "Annot.: %s\n", pszAnnotation);
  890. }
  891. }
  892. else
  893. {
  894. fprintf(pf, "\n///////////////////////////////////////\n");
  895. fprintf(pf, "// %ld:%03ld\n", tsIntv.tv_sec, tsIntv.tv_nsec / 1000000);
  896. fprintf(pf, "// %s:\n", bTX ? "TX" : "RX");
  897. fprintf(pf, "Invalid Mininet-Frame!\n");
  898. }
  899. fprintf(pf, "\n");
  900. fflush(pf);
  901. }
  902. }
  903. }
  904. /////////////////////////////////////////////////////////////////////////////
  905. int GfaMininetMasterSetVerbosity(HGFAMINEMST hMst, int nVerbosity)
  906. {
  907. if(hMst)
  908. {
  909. LPGFA_MININET_MASTER pMst = (LPGFA_MININET_MASTER)hMst;
  910. if(nVerbosity < 0)
  911. nVerbosity = 0;
  912. else if(nVerbosity > 4)
  913. nVerbosity = 4;
  914. pMst->nVerbosity = nVerbosity;
  915. return 0;
  916. }
  917. errno = EINVAL;
  918. return -1;
  919. }
  920. /////////////////////////////////////////////////////////////////////////////
  921. const char* GfaMininetMasterStrError(int nErrorCode)
  922. {
  923. switch(nErrorCode)
  924. {
  925. case MINET_SLAVE_RESPONSE_ERROR_STX_ERROR:
  926. return "First byte in Mininet Frame is neither STX nor ACK";
  927. case MINET_SLAVE_RESPONSE_ERROR_INVALID_ARGUMENT:
  928. return "An invalid argument was passed to GfaMininetMasterEvaluateSlaveResponse";
  929. case MINET_SLAVE_RESPONSE_ERROR_INVALID_LENGTH:
  930. return "Invalid Mininet Frame length";
  931. case MINET_SLAVE_RESPONSE_ERROR_INVALID_SLAVE_INDICATOR:
  932. return "Invalid Mininet slave indicator";
  933. case MINET_SLAVE_RESPONSE_ERROR_INCOMPLETE_DATA:
  934. return "Not enough data for Mininet Frame";
  935. case MINET_SLAVE_RESPONSE_ERROR_INVALID_NODE_ADDRESS:
  936. return "Invalid Mininet Node address";
  937. case MINET_SLAVE_RESPONSE_ERROR_INDEX_OUT_OF_RANGE:
  938. return "Mininet Index out of range";
  939. case MINET_SLAVE_RESPONSE_ERROR_INDEX_NO_MATCH:
  940. return "Mininet Index no match";
  941. case MINET_SLAVE_RESPONSE_ERROR_INVALID_CHECKSUM:
  942. return "Mininet Checksum error";
  943. case -MINET_SLAVE_STATUS_INDEX_RESET_DONE:
  944. return "Mininet index reset done";
  945. case -MINET_SLAVE_STATUS_INDEX_ERROR:
  946. return "Mininet index error";
  947. case -MINET_SLAVE_STATUS_INDEX_CMD_ERROR:
  948. return "Mininet command error";
  949. case -MINET_SLAVE_STATUS_INDEX_INVALID_PARAM:
  950. return "Mininet invalid parameter";
  951. case -MINET_SLAVE_STATUS_INDEX_UNKNOWN_CMD:
  952. return "Mininet unknown command";
  953. case -MINET_SLAVE_STATUS_INDEX_CMD_ALREADY_EX:
  954. return "Mininet command alearx executing";
  955. default:
  956. return strerror(nErrorCode);
  957. }
  958. }