PageRenderTime 55ms CodeModel.GetById 24ms RepoModel.GetById 0ms app.codeStats 0ms

/external/bsd/tcpdump/dist/print-forces.c

https://github.com/rumpkernel/netbsd-userspace-src
C | 1755 lines | 1410 code | 204 blank | 141 comment | 218 complexity | f8a3d10fa48a8ec93bdc050f4684152d MD5 | raw file
  1. /*
  2. * Redistribution and use in source and binary forms, with or without
  3. * modification, are permitted provided that: (1) source code
  4. * distributions retain the above copyright notice and this paragraph
  5. * in its entirety, and (2) distributions including binary code include
  6. * the above copyright notice and this paragraph in its entirety in
  7. * the documentation or other materials provided with the distribution.
  8. * THIS SOFTWARE IS PROVIDED ``AS IS'' AND
  9. * WITHOUT ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, WITHOUT
  10. * LIMITATION, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
  11. * FOR A PARTICULAR PURPOSE.
  12. *
  13. * Copyright (c) 2009 Mojatatu Networks, Inc
  14. *
  15. */
  16. #ifdef HAVE_CONFIG_H
  17. #include "config.h"
  18. #endif
  19. #include <tcpdump-stdinc.h>
  20. #include <stdio.h>
  21. #include <stdlib.h>
  22. #include "interface.h"
  23. #include "extract.h"
  24. /*
  25. * Per draft-ietf-forces-protocol-22
  26. */
  27. #define ForCES_VERS 1
  28. #define ForCES_HDRL 24
  29. #define ForCES_ALNL 4U
  30. #define TLV_HDRL 4
  31. #define ILV_HDRL 8
  32. #define TOM_RSVD 0x0
  33. #define TOM_ASSNSETUP 0x1
  34. #define TOM_ASSNTEARD 0x2
  35. #define TOM_CONFIG 0x3
  36. #define TOM_QUERY 0x4
  37. #define TOM_EVENTNOT 0x5
  38. #define TOM_PKTREDIR 0x6
  39. #define TOM_HEARTBT 0x0F
  40. #define TOM_ASSNSETREP 0x11
  41. #define TOM_CONFIGREP 0x13
  42. #define TOM_QUERYREP 0x14
  43. /*
  44. * tom_h Flags: resv1(8b):maxtlvs(4b):resv2(2b):mintlv(2b)
  45. */
  46. #define ZERO_TTLV 0x01
  47. #define ZERO_MORE_TTLV 0x02
  48. #define ONE_MORE_TTLV 0x04
  49. #define ZERO_TLV 0x00
  50. #define ONE_TLV 0x10
  51. #define TWO_TLV 0x20
  52. #define MAX_TLV 0xF0
  53. #define TTLV_T1 (ONE_MORE_TTLV|ONE_TLV)
  54. #define TTLV_T2 (ONE_MORE_TTLV|MAX_TLV)
  55. struct tom_h {
  56. u_int32_t v;
  57. u_int16_t flags;
  58. u_int16_t op_msk;
  59. const char *s;
  60. int (*print) (register const u_char * pptr, register u_int len,
  61. u_int16_t op_msk, int indent);
  62. };
  63. enum {
  64. TOM_RSV_I,
  65. TOM_ASS_I,
  66. TOM_AST_I,
  67. TOM_CFG_I,
  68. TOM_QRY_I,
  69. TOM_EVN_I,
  70. TOM_RED_I,
  71. TOM_HBT_I,
  72. TOM_ASR_I,
  73. TOM_CNR_I,
  74. TOM_QRR_I,
  75. _TOM_RSV_MAX
  76. };
  77. #define TOM_MAX_IND (_TOM_RSV_MAX - 1)
  78. static inline int tom_valid(u_int8_t tom)
  79. {
  80. if (tom > 0) {
  81. if (tom >= 0x7 && tom <= 0xe)
  82. return 0;
  83. if (tom == 0x10)
  84. return 0;
  85. if (tom > 0x14)
  86. return 0;
  87. return 1;
  88. } else
  89. return 0;
  90. }
  91. static inline const char *ForCES_node(u_int32_t node)
  92. {
  93. if (node <= 0x3FFFFFFF)
  94. return "FE";
  95. if (node >= 0x40000000 && node <= 0x7FFFFFFF)
  96. return "CE";
  97. if (node >= 0xC0000000 && node <= 0xFFFFFFEF)
  98. return "AllMulticast";
  99. if (node == 0xFFFFFFFD)
  100. return "AllCEsBroadcast";
  101. if (node == 0xFFFFFFFE)
  102. return "AllFEsBroadcast";
  103. if (node == 0xFFFFFFFF)
  104. return "AllBroadcast";
  105. return "ForCESreserved";
  106. }
  107. static inline const char *ForCES_ACKp(u_int32_t flg)
  108. {
  109. if (flg == 0x0)
  110. return "NoACK";
  111. if (flg == 0x1)
  112. return "SuccessACK";
  113. if (flg == 0x2)
  114. return "FailureACK";
  115. if (flg == 0x3)
  116. return "AlwaysACK";
  117. return "ACKUnknown";
  118. }
  119. static inline const char *ForCES_EMp(u_int32_t flg)
  120. {
  121. if (flg == 0x0)
  122. return "EMReserved";
  123. if (flg == 0x1)
  124. return "execute-all-or-none";
  125. if (flg == 0x2)
  126. return "execute-until-failure";
  127. if (flg == 0x3)
  128. return "continue-execute-on-failure";
  129. return "EMUnknown";
  130. }
  131. static inline const char *ForCES_ATp(u_int32_t flg)
  132. {
  133. if (flg == 0x0)
  134. return "Standalone";
  135. if (flg == 0x1)
  136. return "2PCtransaction";
  137. return "ATUnknown";
  138. }
  139. static inline const char *ForCES_TPp(u_int32_t flg)
  140. {
  141. if (flg == 0x0)
  142. return "StartofTransaction";
  143. if (flg == 0x1)
  144. return "MiddleofTransaction";
  145. if (flg == 0x2)
  146. return "EndofTransaction";
  147. if (flg == 0x3)
  148. return "abort";
  149. return "TPUnknown";
  150. }
  151. /*
  152. * Structure of forces header, naked of TLVs.
  153. */
  154. struct forcesh {
  155. u_int8_t fm_vrsvd; /* version and reserved */
  156. #define ForCES_V(forcesh) ((forcesh)->fm_vrsvd >> 4)
  157. u_int8_t fm_tom; /* type of message */
  158. u_int16_t fm_len; /* total length * 4 bytes */
  159. #define ForCES_BLN(forcesh) ((u_int32_t)(EXTRACT_16BITS(&(forcesh)->fm_len) << 2))
  160. u_int32_t fm_sid; /* Source ID */
  161. #define ForCES_SID(forcesh) EXTRACT_32BITS(&(forcesh)->fm_sid)
  162. u_int32_t fm_did; /* Destination ID */
  163. #define ForCES_DID(forcesh) EXTRACT_32BITS(&(forcesh)->fm_did)
  164. u_int8_t fm_cor[8]; /* correlator */
  165. u_int32_t fm_flags; /* flags */
  166. #define ForCES_ACK(forcesh) ((EXTRACT_32BITS(&(forcesh)->fm_flags)&0xC0000000) >> 30)
  167. #define ForCES_PRI(forcesh) ((EXTRACT_32BITS(&(forcesh)->fm_flags)&0x38000000) >> 27)
  168. #define ForCES_RS1(forcesh) ((EXTRACT_32BITS(&(forcesh)->fm_flags)&0x07000000) >> 24)
  169. #define ForCES_EM(forcesh) ((EXTRACT_32BITS(&(forcesh)->fm_flags)&0x00C00000) >> 22)
  170. #define ForCES_AT(forcesh) ((EXTRACT_32BITS(&(forcesh)->fm_flags)&0x00200000) >> 21)
  171. #define ForCES_TP(forcesh) ((EXTRACT_32BITS(&(forcesh)->fm_flags)&0x00180000) >> 19)
  172. #define ForCES_RS2(forcesh) ((EXTRACT_32BITS(&(forcesh)->fm_flags)&0x0007FFFF) >> 0)
  173. };
  174. #define ForCES_HLN_VALID(fhl,tlen) ((tlen) >= ForCES_HDRL && \
  175. (fhl) >= ForCES_HDRL && \
  176. (fhl) == (tlen))
  177. #define F_LFB_RSVD 0x0
  178. #define F_LFB_FEO 0x1
  179. #define F_LFB_FEPO 0x2
  180. static const struct tok ForCES_LFBs[] = {
  181. {F_LFB_RSVD, "Invalid TLV"},
  182. {F_LFB_FEO, "FEObj LFB"},
  183. {F_LFB_FEPO, "FEProtoObj LFB"},
  184. {0, NULL}
  185. };
  186. enum {
  187. F_OP_RSV,
  188. F_OP_SET,
  189. F_OP_SETPROP,
  190. F_OP_SETRESP,
  191. F_OP_SETPRESP,
  192. F_OP_DEL,
  193. F_OP_DELRESP,
  194. F_OP_GET,
  195. F_OP_GETPROP,
  196. F_OP_GETRESP,
  197. F_OP_GETPRESP,
  198. F_OP_REPORT,
  199. F_OP_COMMIT,
  200. F_OP_RCOMMIT,
  201. F_OP_RTRCOMP,
  202. _F_OP_MAX
  203. };
  204. #define F_OP_MAX (_F_OP_MAX - 1)
  205. enum {
  206. B_OP_SET = 1 << (F_OP_SET - 1),
  207. B_OP_SETPROP = 1 << (F_OP_SETPROP - 1),
  208. B_OP_SETRESP = 1 << (F_OP_SETRESP - 1),
  209. B_OP_SETPRESP = 1 << (F_OP_SETPRESP - 1),
  210. B_OP_DEL = 1 << (F_OP_DEL - 1),
  211. B_OP_DELRESP = 1 << (F_OP_DELRESP - 1),
  212. B_OP_GET = 1 << (F_OP_GET - 1),
  213. B_OP_GETPROP = 1 << (F_OP_GETPROP - 1),
  214. B_OP_GETRESP = 1 << (F_OP_GETRESP - 1),
  215. B_OP_GETPRESP = 1 << (F_OP_GETPRESP - 1),
  216. B_OP_REPORT = 1 << (F_OP_REPORT - 1),
  217. B_OP_COMMIT = 1 << (F_OP_COMMIT - 1),
  218. B_OP_RCOMMIT = 1 << (F_OP_RCOMMIT - 1),
  219. B_OP_RTRCOMP = 1 << (F_OP_RTRCOMP - 1),
  220. };
  221. struct optlv_h {
  222. u_int16_t flags;
  223. u_int16_t op_msk;
  224. const char *s;
  225. int (*print) (register const u_char * pptr, register u_int len,
  226. u_int16_t op_msk, int indent);
  227. };
  228. static int genoptlv_print(register const u_char * pptr, register u_int len,
  229. u_int16_t op_msk, int indent);
  230. static int recpdoptlv_print(register const u_char * pptr, register u_int len,
  231. u_int16_t op_msk, int indent);
  232. static int invoptlv_print(register const u_char * pptr, register u_int len,
  233. u_int16_t op_msk, int indent);
  234. #define OP_MIN_SIZ 8
  235. struct pathdata_h {
  236. u_int16_t pflags;
  237. u_int16_t pIDcnt;
  238. };
  239. #define B_FULLD 0x1
  240. #define B_SPARD 0x2
  241. #define B_RESTV 0x4
  242. #define B_KEYIN 0x8
  243. #define B_APPND 0x10
  244. #define B_TRNG 0x20
  245. static const struct optlv_h OPTLV_msg[F_OP_MAX + 1] = {
  246. /* F_OP_RSV */ {ZERO_TTLV, 0, "Invalid OPTLV", invoptlv_print},
  247. /* F_OP_SET */ {TTLV_T2, B_FULLD | B_SPARD, " Set", recpdoptlv_print},
  248. /* F_OP_SETPROP */
  249. {TTLV_T2, B_FULLD | B_SPARD, " SetProp", recpdoptlv_print},
  250. /* F_OP_SETRESP */ {TTLV_T2, B_RESTV, " SetResp", recpdoptlv_print},
  251. /* F_OP_SETPRESP */ {TTLV_T2, B_RESTV, " SetPropResp", recpdoptlv_print},
  252. /* F_OP_DEL */ {ZERO_TTLV, 0, " Del", recpdoptlv_print},
  253. /* F_OP_DELRESP */ {TTLV_T2, B_RESTV, " DelResp", recpdoptlv_print},
  254. /* F_OP_GET */ {ZERO_TTLV, 0, " Get", recpdoptlv_print},
  255. /* F_OP_GETPROP */ {ZERO_TTLV, 0, " GetProp", recpdoptlv_print},
  256. /* F_OP_GETRESP */
  257. {TTLV_T2, B_FULLD | B_SPARD | B_RESTV, " GetResp", recpdoptlv_print},
  258. /* F_OP_GETPRESP */
  259. {TTLV_T2, B_FULLD | B_RESTV, " GetPropResp", recpdoptlv_print},
  260. /* F_OP_REPORT */
  261. {TTLV_T2, B_FULLD | B_SPARD, " Report", recpdoptlv_print},
  262. /* F_OP_COMMIT */ {ZERO_TTLV, 0, " Commit", NULL},
  263. /* F_OP_RCOMMIT */ {TTLV_T1, B_RESTV, " RCommit", genoptlv_print},
  264. /* F_OP_RTRCOMP */ {ZERO_TTLV, 0, " RTRCOMP", NULL},
  265. };
  266. static inline const struct optlv_h *get_forces_optlv_h(u_int16_t opt)
  267. {
  268. if (opt > F_OP_MAX || opt <= F_OP_RSV)
  269. return &OPTLV_msg[F_OP_RSV];
  270. return &OPTLV_msg[opt];
  271. }
  272. #define IND_SIZE 256
  273. #define IND_CHR ' '
  274. #define IND_PREF '\n'
  275. #define IND_SUF 0x0
  276. char ind_buf[IND_SIZE];
  277. static inline char *indent_pr(int indent, int nlpref)
  278. {
  279. int i = 0;
  280. char *r = ind_buf;
  281. if (indent > (IND_SIZE - 1))
  282. indent = IND_SIZE - 1;
  283. if (nlpref) {
  284. r[i] = IND_PREF;
  285. i++;
  286. indent--;
  287. }
  288. while (--indent >= 0)
  289. r[i++] = IND_CHR;
  290. r[i] = IND_SUF;
  291. return r;
  292. }
  293. static inline int op_valid(u_int16_t op, u_int16_t mask)
  294. {
  295. int opb = 1 << (op - 1);
  296. if (op == 0)
  297. return 0;
  298. if (opb & mask)
  299. return 1;
  300. /* I guess we should allow vendor operations? */
  301. if (op >= 0x8000)
  302. return 1;
  303. return 0;
  304. }
  305. #define F_TLV_RSVD 0x0000
  306. #define F_TLV_REDR 0x0001
  307. #define F_TLV_ASRS 0x0010
  308. #define F_TLV_ASRT 0x0011
  309. #define F_TLV_LFBS 0x1000
  310. #define F_TLV_PDAT 0x0110
  311. #define F_TLV_KEYI 0x0111
  312. #define F_TLV_FULD 0x0112
  313. #define F_TLV_SPAD 0x0113
  314. #define F_TLV_REST 0x0114
  315. #define F_TLV_METD 0x0115
  316. #define F_TLV_REDD 0x0116
  317. #define F_TLV_TRNG 0x0117
  318. #define F_TLV_VNST 0x8000
  319. static const struct tok ForCES_TLV[] = {
  320. {F_TLV_RSVD, "Invalid TLV"},
  321. {F_TLV_REDR, "REDIRECT TLV"},
  322. {F_TLV_ASRS, "ASResult TLV"},
  323. {F_TLV_ASRT, "ASTreason TLV"},
  324. {F_TLV_LFBS, "LFBselect TLV"},
  325. {F_TLV_PDAT, "PATH-DATA TLV"},
  326. {F_TLV_KEYI, "KEYINFO TLV"},
  327. {F_TLV_FULD, "FULLDATA TLV"},
  328. {F_TLV_SPAD, "SPARSEDATA TLV"},
  329. {F_TLV_REST, "RESULT TLV"},
  330. {F_TLV_METD, "METADATA TLV"},
  331. {F_TLV_REDD, "REDIRECTDATA TLV"},
  332. {0, NULL}
  333. };
  334. #define TLV_HLN 4
  335. static inline int ttlv_valid(u_int16_t ttlv)
  336. {
  337. if (ttlv > 0) {
  338. if (ttlv == 1 || ttlv == 0x1000)
  339. return 1;
  340. if (ttlv >= 0x10 && ttlv <= 0x11)
  341. return 1;
  342. if (ttlv >= 0x110 && ttlv <= 0x116)
  343. return 1;
  344. if (ttlv >= 0x8000)
  345. return 0; /* XXX: */
  346. }
  347. return 0;
  348. }
  349. struct forces_ilv {
  350. u_int32_t type;
  351. u_int32_t length;
  352. };
  353. struct forces_tlv {
  354. u_int16_t type;
  355. u_int16_t length;
  356. };
  357. #define F_ALN_LEN(len) ( ((len)+ForCES_ALNL-1) & ~(ForCES_ALNL-1) )
  358. #define GET_TOP_TLV(fhdr) ((struct forces_tlv *)((fhdr) + sizeof (struct forcesh)))
  359. #define TLV_SET_LEN(len) (F_ALN_LEN(TLV_HDRL) + (len))
  360. #define TLV_ALN_LEN(len) F_ALN_LEN(TLV_SET_LEN(len))
  361. #define TLV_RDAT_LEN(tlv) ((int)(EXTRACT_16BITS(&(tlv)->length) - TLV_SET_LEN(0))
  362. #define TLV_DATA(tlvp) ((void*)(((char*)(tlvp)) + TLV_SET_LEN(0)))
  363. #define GO_NXT_TLV(tlv,rlen) ((rlen) -= F_ALN_LEN(EXTRACT_16BITS(&(tlv)->length)), \
  364. (struct forces_tlv*)(((char*)(tlv)) \
  365. + F_ALN_LEN(EXTRACT_16BITS(&(tlv)->length))))
  366. #define ILV_SET_LEN(len) (F_ALN_LEN(ILV_HDRL) + (len))
  367. #define ILV_ALN_LEN(len) F_ALN_LEN(ILV_SET_LEN(len))
  368. #define ILV_RDAT_LEN(ilv) ((int)(EXTRACT_32BITS(&(ilv)->length)) - ILV_SET_LEN(0))
  369. #define ILV_DATA(ilvp) ((void*)(((char*)(ilvp)) + ILV_SET_LEN(0)))
  370. #define GO_NXT_ILV(ilv,rlen) ((rlen) -= F_ALN_LEN(EXTRACT_32BITS(&(ilv)->length)), \
  371. (struct forces_ilv *)(((char*)(ilv)) \
  372. + F_ALN_LEN(EXTRACT_32BITS(&(ilv)->length))))
  373. #define INVALID_RLEN -1
  374. #define INVALID_STLN -2
  375. #define INVALID_LTLN -3
  376. #define INVALID_ALEN -4
  377. static const struct tok ForCES_TLV_err[] = {
  378. {INVALID_RLEN, "Invalid total length"},
  379. {INVALID_STLN, "xLV too short"},
  380. {INVALID_LTLN, "xLV too long"},
  381. {INVALID_ALEN, "data padding missing"},
  382. {0, NULL}
  383. };
  384. static inline int tlv_valid(const struct forces_tlv *tlv, u_int rlen)
  385. {
  386. if (rlen < TLV_HDRL)
  387. return INVALID_RLEN;
  388. if (EXTRACT_16BITS(&tlv->length) < TLV_HDRL)
  389. return INVALID_STLN;
  390. if (EXTRACT_16BITS(&tlv->length) > rlen)
  391. return INVALID_LTLN;
  392. if (rlen < F_ALN_LEN(EXTRACT_16BITS(&tlv->length)))
  393. return INVALID_ALEN;
  394. return 0;
  395. }
  396. static inline int ilv_valid(const struct forces_ilv *ilv, u_int rlen)
  397. {
  398. if (rlen < ILV_HDRL)
  399. return INVALID_RLEN;
  400. if (EXTRACT_32BITS(&ilv->length) < ILV_HDRL)
  401. return INVALID_STLN;
  402. if (EXTRACT_32BITS(&ilv->length) > rlen)
  403. return INVALID_LTLN;
  404. if (rlen < F_ALN_LEN(EXTRACT_32BITS(&ilv->length)))
  405. return INVALID_ALEN;
  406. return 0;
  407. }
  408. static int lfbselect_print(register const u_char * pptr, register u_int len,
  409. u_int16_t op_msk, int indent);
  410. static int redirect_print(register const u_char * pptr, register u_int len,
  411. u_int16_t op_msk, int indent);
  412. static int asrtlv_print(register const u_char * pptr, register u_int len,
  413. u_int16_t op_msk, int indent);
  414. static int asttlv_print(register const u_char * pptr, register u_int len,
  415. u_int16_t op_msk, int indent);
  416. struct forces_lfbsh {
  417. u_int32_t class;
  418. u_int32_t instance;
  419. };
  420. #define ASSNS_OPS (B_OP_REPORT)
  421. #define CFG_OPS (B_OP_SET|B_OP_SETPROP|B_OP_DEL|B_OP_COMMIT|B_OP_RTRCOMP)
  422. #define CFG_ROPS (B_OP_SETRESP|B_OP_SETPRESP|B_OP_DELRESP|B_OP_RCOMMIT)
  423. #define CFG_QY (B_OP_GET|B_OP_GETPROP)
  424. #define CFG_QYR (B_OP_GETRESP|B_OP_GETPRESP)
  425. #define CFG_EVN (B_OP_REPORT)
  426. static const struct tom_h ForCES_msg[TOM_MAX_IND + 1] = {
  427. /* TOM_RSV_I */ {TOM_RSVD, ZERO_TTLV, 0, "Invalid message", NULL},
  428. /* TOM_ASS_I */ {TOM_ASSNSETUP, ZERO_MORE_TTLV | TWO_TLV, ASSNS_OPS,
  429. "Association Setup", lfbselect_print},
  430. /* TOM_AST_I */
  431. {TOM_ASSNTEARD, TTLV_T1, 0, "Association TearDown", asttlv_print},
  432. /* TOM_CFG_I */ {TOM_CONFIG, TTLV_T2, CFG_OPS, "Config", lfbselect_print},
  433. /* TOM_QRY_I */ {TOM_QUERY, TTLV_T2, CFG_QY, "Query", lfbselect_print},
  434. /* TOM_EVN_I */ {TOM_EVENTNOT, TTLV_T1, CFG_EVN, "Event Notification",
  435. lfbselect_print},
  436. /* TOM_RED_I */
  437. {TOM_PKTREDIR, TTLV_T2, 0, "Packet Redirect", redirect_print},
  438. /* TOM_HBT_I */ {TOM_HEARTBT, ZERO_TTLV, 0, "HeartBeat", NULL},
  439. /* TOM_ASR_I */
  440. {TOM_ASSNSETREP, TTLV_T1, 0, "Association Response", asrtlv_print},
  441. /* TOM_CNR_I */ {TOM_CONFIGREP, TTLV_T2, CFG_ROPS, "Config Response",
  442. lfbselect_print},
  443. /* TOM_QRR_I */
  444. {TOM_QUERYREP, TTLV_T2, CFG_QYR, "Query Response", lfbselect_print},
  445. };
  446. static inline const struct tom_h *get_forces_tom(u_int8_t tom)
  447. {
  448. int i;
  449. for (i = TOM_RSV_I; i <= TOM_MAX_IND; i++) {
  450. const struct tom_h *th = &ForCES_msg[i];
  451. if (th->v == tom)
  452. return th;
  453. }
  454. return &ForCES_msg[TOM_RSV_I];
  455. }
  456. struct pdata_ops {
  457. u_int32_t v;
  458. u_int16_t flags;
  459. u_int16_t op_msk;
  460. const char *s;
  461. int (*print) (register const u_char * pptr, register u_int len,
  462. u_int16_t op_msk, int indent);
  463. };
  464. enum {
  465. PD_RSV_I,
  466. PD_SEL_I,
  467. PD_FDT_I,
  468. PD_SDT_I,
  469. PD_RES_I,
  470. PD_PDT_I,
  471. _PD_RSV_MAX
  472. };
  473. #define PD_MAX_IND (_TOM_RSV_MAX - 1)
  474. static inline int pd_valid(u_int16_t pd)
  475. {
  476. if (pd >= F_TLV_PDAT && pd <= F_TLV_REST)
  477. return 1;
  478. return 0;
  479. }
  480. static inline void chk_op_type(u_int16_t type, u_int16_t msk, u_int16_t omsk)
  481. {
  482. if (type != F_TLV_PDAT) {
  483. if (msk & B_KEYIN) {
  484. if (type != F_TLV_KEYI) {
  485. printf
  486. ("Based on flags expected KEYINFO TLV!\n");
  487. }
  488. } else {
  489. if (!(msk & omsk)) {
  490. printf
  491. ("Illegal DATA encoding for type 0x%x programmed %x got %x \n",
  492. type, omsk, msk);
  493. }
  494. }
  495. }
  496. }
  497. #define F_SELKEY 1
  498. #define F_SELTABRANGE 2
  499. #define F_TABAPPEND 4
  500. struct res_val {
  501. u_int8_t result;
  502. u_int8_t resv1;
  503. u_int16_t resv2;
  504. };
  505. static int prestlv_print(register const u_char * pptr, register u_int len,
  506. u_int16_t op_msk, int indent);
  507. static int pkeyitlv_print(register const u_char * pptr, register u_int len,
  508. u_int16_t op_msk, int indent);
  509. static int fdatatlv_print(register const u_char * pptr, register u_int len,
  510. u_int16_t op_msk, int indent);
  511. static int sdatatlv_print(register const u_char * pptr, register u_int len,
  512. u_int16_t op_msk, int indent);
  513. static const struct pdata_ops ForCES_pdata[PD_MAX_IND + 1] = {
  514. /* PD_RSV_I */ {0, 0, 0, "Invalid message", NULL},
  515. /* PD_SEL_I */ {F_TLV_KEYI, 0, 0, "KEYINFO TLV", pkeyitlv_print},
  516. /* PD_FDT_I */ {F_TLV_FULD, 0, B_FULLD, "FULLDATA TLV", fdatatlv_print},
  517. /* PD_SDT_I */ {F_TLV_SPAD, 0, B_SPARD, "SPARSEDATA TLV", sdatatlv_print},
  518. /* PD_RES_I */ {F_TLV_REST, 0, B_RESTV, "RESULT TLV", prestlv_print},
  519. /* PD_PDT_I */
  520. {F_TLV_PDAT, 0, 0, "Inner PATH-DATA TLV", recpdoptlv_print},
  521. };
  522. static inline const struct pdata_ops *get_forces_pd(u_int16_t pd)
  523. {
  524. int i;
  525. for (i = PD_RSV_I + 1; i <= PD_MAX_IND; i++) {
  526. const struct pdata_ops *pdo = &ForCES_pdata[i];
  527. if (pdo->v == pd)
  528. return pdo;
  529. }
  530. return &ForCES_pdata[TOM_RSV_I];
  531. }
  532. enum {
  533. E_SUCCESS,
  534. E_INVALID_HEADER,
  535. E_LENGTH_MISMATCH,
  536. E_VERSION_MISMATCH,
  537. E_INVALID_DESTINATION_PID,
  538. E_LFB_UNKNOWN,
  539. E_LFB_NOT_FOUND,
  540. E_LFB_INSTANCE_ID_NOT_FOUND,
  541. E_INVALID_PATH,
  542. E_COMPONENT_DOES_NOT_EXIST,
  543. E_EXISTS,
  544. E_NOT_FOUND,
  545. E_READ_ONLY,
  546. E_INVALID_ARRAY_CREATION,
  547. E_VALUE_OUT_OF_RANGE,
  548. E_CONTENTS_TOO_LONG,
  549. E_INVALID_PARAMETERS,
  550. E_INVALID_MESSAGE_TYPE,
  551. E_INVALID_FLAGS,
  552. E_INVALID_TLV,
  553. E_EVENT_ERROR,
  554. E_NOT_SUPPORTED,
  555. E_MEMORY_ERROR,
  556. E_INTERNAL_ERROR,
  557. /* 0x18-0xFE are reserved .. */
  558. E_UNSPECIFIED_ERROR = 0XFF
  559. };
  560. static const struct tok ForCES_errs[] = {
  561. {E_SUCCESS, "SUCCESS"},
  562. {E_INVALID_HEADER, "INVALID HEADER"},
  563. {E_LENGTH_MISMATCH, "LENGTH MISMATCH"},
  564. {E_VERSION_MISMATCH, "VERSION MISMATCH"},
  565. {E_INVALID_DESTINATION_PID, "INVALID DESTINATION PID"},
  566. {E_LFB_UNKNOWN, "LFB UNKNOWN"},
  567. {E_LFB_NOT_FOUND, "LFB NOT FOUND"},
  568. {E_LFB_INSTANCE_ID_NOT_FOUND, "LFB INSTANCE ID NOT FOUND"},
  569. {E_INVALID_PATH, "INVALID PATH"},
  570. {E_COMPONENT_DOES_NOT_EXIST, "COMPONENT DOES NOT EXIST"},
  571. {E_EXISTS, "EXISTS ALREADY"},
  572. {E_NOT_FOUND, "NOT FOUND"},
  573. {E_READ_ONLY, "READ ONLY"},
  574. {E_INVALID_ARRAY_CREATION, "INVALID ARRAY CREATION"},
  575. {E_VALUE_OUT_OF_RANGE, "VALUE OUT OF RANGE"},
  576. {E_CONTENTS_TOO_LONG, "CONTENTS TOO LONG"},
  577. {E_INVALID_PARAMETERS, "INVALID PARAMETERS"},
  578. {E_INVALID_MESSAGE_TYPE, "INVALID MESSAGE TYPE"},
  579. {E_INVALID_FLAGS, "INVALID FLAGS"},
  580. {E_INVALID_TLV, "INVALID TLV"},
  581. {E_EVENT_ERROR, "EVENT ERROR"},
  582. {E_NOT_SUPPORTED, "NOT SUPPORTED"},
  583. {E_MEMORY_ERROR, "MEMORY ERROR"},
  584. {E_INTERNAL_ERROR, "INTERNAL ERROR"},
  585. {E_UNSPECIFIED_ERROR, "UNSPECIFIED ERROR"},
  586. {0, NULL}
  587. };
  588. #define RESLEN 4
  589. static int
  590. prestlv_print(register const u_char * pptr, register u_int len,
  591. u_int16_t op_msk _U_, int indent)
  592. {
  593. const struct forces_tlv *tlv = (struct forces_tlv *)pptr;
  594. register const u_char *tdp = (u_char *) TLV_DATA(tlv);
  595. struct res_val *r = (struct res_val *)tdp;
  596. u_int dlen;
  597. /*
  598. * pdatacnt_print() has ensured that len (the TLV length)
  599. * >= TLV_HDRL.
  600. */
  601. dlen = len - TLV_HDRL;
  602. if (dlen != RESLEN) {
  603. printf("illegal RESULT-TLV: %d bytes!\n", dlen);
  604. return -1;
  605. }
  606. TCHECK(*r);
  607. if (r->result >= 0x18 && r->result <= 0xFE) {
  608. printf("illegal reserved result code: 0x%x!\n", r->result);
  609. return -1;
  610. }
  611. if (vflag >= 3) {
  612. char *ib = indent_pr(indent, 0);
  613. printf("%s Result: %s (code 0x%x)\n", ib,
  614. tok2str(ForCES_errs, NULL, r->result), r->result);
  615. }
  616. return 0;
  617. trunc:
  618. fputs("[|forces]", stdout);
  619. return -1;
  620. }
  621. static int
  622. fdatatlv_print(register const u_char * pptr, register u_int len,
  623. u_int16_t op_msk _U_, int indent)
  624. {
  625. const struct forces_tlv *tlv = (struct forces_tlv *)pptr;
  626. u_int rlen;
  627. register const u_char *tdp = (u_char *) TLV_DATA(tlv);
  628. u_int16_t type;
  629. /*
  630. * pdatacnt_print() or pkeyitlv_print() has ensured that len
  631. * (the TLV length) >= TLV_HDRL.
  632. */
  633. rlen = len - TLV_HDRL;
  634. TCHECK(*tlv);
  635. type = EXTRACT_16BITS(&tlv->type);
  636. if (type != F_TLV_FULD) {
  637. printf("Error: expecting FULLDATA!\n");
  638. return -1;
  639. }
  640. if (vflag >= 3) {
  641. char *ib = indent_pr(indent + 2, 1);
  642. printf("%s[", &ib[1]);
  643. hex_print_with_offset(ib, tdp, rlen, 0);
  644. printf("\n%s]\n", &ib[1]);
  645. }
  646. return 0;
  647. trunc:
  648. fputs("[|forces]", stdout);
  649. return -1;
  650. }
  651. static int
  652. sdatailv_print(register const u_char * pptr, register u_int len,
  653. u_int16_t op_msk _U_, int indent)
  654. {
  655. u_int rlen;
  656. const struct forces_ilv *ilv = (struct forces_ilv *)pptr;
  657. int invilv;
  658. if (len < ILV_HDRL) {
  659. printf("Error: BAD SPARSEDATA-TLV!\n");
  660. return -1;
  661. }
  662. rlen = len;
  663. indent += 1;
  664. while (rlen != 0) {
  665. #if 0
  666. printf("Jamal - outstanding length <%d>\n", rlen);
  667. #endif
  668. char *ib = indent_pr(indent, 1);
  669. register const u_char *tdp = (u_char *) ILV_DATA(ilv);
  670. TCHECK(*ilv);
  671. invilv = ilv_valid(ilv, rlen);
  672. if (invilv) {
  673. printf("%s[", &ib[1]);
  674. hex_print_with_offset(ib, tdp, rlen, 0);
  675. printf("\n%s]\n", &ib[1]);
  676. return -1;
  677. }
  678. if (vflag >= 3) {
  679. int ilvl = EXTRACT_32BITS(&ilv->length);
  680. printf("\n%s ILV: type %x length %d\n", &ib[1],
  681. EXTRACT_32BITS(&ilv->type), ilvl);
  682. hex_print_with_offset("\t\t[", tdp, ilvl-ILV_HDRL, 0);
  683. }
  684. ilv = GO_NXT_ILV(ilv, rlen);
  685. }
  686. return 0;
  687. trunc:
  688. fputs("[|forces]", stdout);
  689. return -1;
  690. }
  691. static int
  692. sdatatlv_print(register const u_char * pptr, register u_int len,
  693. u_int16_t op_msk, int indent)
  694. {
  695. const struct forces_tlv *tlv = (struct forces_tlv *)pptr;
  696. u_int rlen;
  697. register const u_char *tdp = (u_char *) TLV_DATA(tlv);
  698. u_int16_t type;
  699. /*
  700. * pdatacnt_print() has ensured that len (the TLV length)
  701. * >= TLV_HDRL.
  702. */
  703. rlen = len - TLV_HDRL;
  704. TCHECK(*tlv);
  705. type = EXTRACT_16BITS(&tlv->type);
  706. if (type != F_TLV_SPAD) {
  707. printf("Error: expecting SPARSEDATA!\n");
  708. return -1;
  709. }
  710. return sdatailv_print(tdp, rlen, op_msk, indent);
  711. trunc:
  712. fputs("[|forces]", stdout);
  713. return -1;
  714. }
  715. static int
  716. pkeyitlv_print(register const u_char * pptr, register u_int len,
  717. u_int16_t op_msk, int indent)
  718. {
  719. const struct forces_tlv *tlv = (struct forces_tlv *)pptr;
  720. register const u_char *tdp = (u_char *) TLV_DATA(tlv);
  721. register const u_char *dp = tdp + 4;
  722. const struct forces_tlv *kdtlv = (struct forces_tlv *)dp;
  723. u_int32_t id;
  724. char *ib = indent_pr(indent, 0);
  725. u_int16_t type, tll;
  726. int invtlv;
  727. TCHECK(*tdp);
  728. id = EXTRACT_32BITS(tdp);
  729. printf("%sKeyinfo: Key 0x%x\n", ib, id);
  730. TCHECK(*kdtlv);
  731. type = EXTRACT_16BITS(&kdtlv->type);
  732. invtlv = tlv_valid(kdtlv, len);
  733. if (invtlv) {
  734. printf("%s TLV type 0x%x len %d\n",
  735. tok2str(ForCES_TLV_err, NULL, invtlv), type,
  736. EXTRACT_16BITS(&kdtlv->length));
  737. return -1;
  738. }
  739. /*
  740. * At this point, tlv_valid() has ensured that the TLV
  741. * length is large enough but not too large (it doesn't
  742. * go past the end of the containing TLV).
  743. */
  744. tll = EXTRACT_16BITS(&kdtlv->length);
  745. dp = (u_char *) TLV_DATA(kdtlv);
  746. return fdatatlv_print(dp, tll, op_msk, indent);
  747. trunc:
  748. fputs("[|forces]", stdout);
  749. return -1;
  750. }
  751. #define PTH_DESC_SIZE 12
  752. static int
  753. pdatacnt_print(register const u_char * pptr, register u_int len,
  754. u_int16_t IDcnt, u_int16_t op_msk, int indent)
  755. {
  756. u_int i;
  757. u_int32_t id;
  758. char *ib = indent_pr(indent, 0);
  759. if ((op_msk & B_APPND) && vflag >= 3) {
  760. printf("%sTABLE APPEND\n", ib);
  761. }
  762. for (i = 0; i < IDcnt; i++) {
  763. TCHECK2(*pptr, 4);
  764. if (len < 4)
  765. goto trunc;
  766. id = EXTRACT_32BITS(pptr);
  767. if (vflag >= 3)
  768. printf("%sID#%02u: %d\n", ib, i + 1, id);
  769. len -= 4;
  770. pptr += 4;
  771. }
  772. if ((op_msk & B_TRNG) || (op_msk & B_KEYIN)) {
  773. if (op_msk & B_TRNG) {
  774. u_int32_t starti, endi;
  775. if (len < PTH_DESC_SIZE) {
  776. printf("pathlength %d with key/range too short %d\n",
  777. len, PTH_DESC_SIZE);
  778. return -1;
  779. }
  780. pptr += sizeof(struct forces_tlv);
  781. len -= sizeof(struct forces_tlv);
  782. starti = EXTRACT_32BITS(pptr);
  783. pptr += 4;
  784. len -= 4;
  785. endi = EXTRACT_32BITS(pptr);
  786. pptr += 4;
  787. len -= 4;
  788. if (vflag >= 3)
  789. printf("%sTable range: [%d,%d]\n", ib, starti, endi);
  790. }
  791. if (op_msk & B_KEYIN) {
  792. struct forces_tlv *keytlv;
  793. u_int16_t tll;
  794. if (len < PTH_DESC_SIZE) {
  795. printf("pathlength %d with key/range too short %d\n",
  796. len, PTH_DESC_SIZE);
  797. return -1;
  798. }
  799. /* skip keyid */
  800. pptr += 4;
  801. len -= 4;
  802. keytlv = (struct forces_tlv *)pptr;
  803. /* skip header */
  804. pptr += sizeof(struct forces_tlv);
  805. len -= sizeof(struct forces_tlv);
  806. /* skip key content */
  807. tll = EXTRACT_16BITS(&keytlv->length);
  808. if (tll < TLV_HDRL) {
  809. printf("key content length %u < %u\n",
  810. tll, TLV_HDRL);
  811. return -1;
  812. }
  813. tll -= TLV_HDRL;
  814. if (len < tll) {
  815. printf("key content too short\n");
  816. return -1;
  817. }
  818. pptr += tll;
  819. len -= tll;
  820. }
  821. }
  822. if (len) {
  823. const struct forces_tlv *pdtlv = (struct forces_tlv *)pptr;
  824. u_int16_t type;
  825. u_int16_t tll;
  826. int pad = 0;
  827. u_int aln;
  828. int invtlv;
  829. TCHECK(*pdtlv);
  830. type = EXTRACT_16BITS(&pdtlv->type);
  831. invtlv = tlv_valid(pdtlv, len);
  832. if (invtlv) {
  833. printf
  834. ("%s Outstanding bytes %d for TLV type 0x%x TLV len %d\n",
  835. tok2str(ForCES_TLV_err, NULL, invtlv), len, type,
  836. EXTRACT_16BITS(&pdtlv->length));
  837. goto pd_err;
  838. }
  839. /*
  840. * At this point, tlv_valid() has ensured that the TLV
  841. * length is large enough but not too large (it doesn't
  842. * go past the end of the containing TLV).
  843. */
  844. tll = EXTRACT_16BITS(&pdtlv->length) - TLV_HDRL;
  845. aln = F_ALN_LEN(EXTRACT_16BITS(&pdtlv->length));
  846. if (aln > EXTRACT_16BITS(&pdtlv->length)) {
  847. if (aln > len) {
  848. printf
  849. ("Invalid padded pathdata TLV type 0x%x len %d missing %d pad bytes\n",
  850. type, EXTRACT_16BITS(&pdtlv->length), aln - len);
  851. } else {
  852. pad = aln - EXTRACT_16BITS(&pdtlv->length);
  853. }
  854. }
  855. if (pd_valid(type)) {
  856. const struct pdata_ops *ops = get_forces_pd(type);
  857. if (vflag >= 3 && ops->v != F_TLV_PDAT) {
  858. if (pad)
  859. printf
  860. ("%s %s (Length %d DataLen %d pad %d Bytes)\n",
  861. ib, ops->s, EXTRACT_16BITS(&pdtlv->length),
  862. tll, pad);
  863. else
  864. printf
  865. ("%s %s (Length %d DataLen %d Bytes)\n",
  866. ib, ops->s, EXTRACT_16BITS(&pdtlv->length),
  867. tll);
  868. }
  869. chk_op_type(type, op_msk, ops->op_msk);
  870. if (ops->print((const u_char *)pdtlv,
  871. tll + pad + TLV_HDRL, op_msk,
  872. indent + 2) == -1)
  873. return -1;
  874. len -= (TLV_HDRL + pad + tll);
  875. } else {
  876. printf("Invalid path data content type 0x%x len %d\n",
  877. type, EXTRACT_16BITS(&pdtlv->length));
  878. pd_err:
  879. if (EXTRACT_16BITS(&pdtlv->length)) {
  880. hex_print_with_offset("Bad Data val\n\t [",
  881. pptr, len, 0);
  882. printf("]\n");
  883. return -1;
  884. }
  885. }
  886. }
  887. return len;
  888. trunc:
  889. fputs("[|forces]", stdout);
  890. return -1;
  891. }
  892. static int
  893. pdata_print(register const u_char * pptr, register u_int len,
  894. u_int16_t op_msk, int indent)
  895. {
  896. const struct pathdata_h *pdh = (struct pathdata_h *)pptr;
  897. char *ib = indent_pr(indent, 0);
  898. u_int minsize = 0;
  899. int more_pd = 0;
  900. u_int16_t idcnt = 0;
  901. TCHECK(*pdh);
  902. if (len < sizeof(struct pathdata_h))
  903. goto trunc;
  904. if (vflag >= 3) {
  905. printf("\n%sPathdata: Flags 0x%x ID count %d\n",
  906. ib, EXTRACT_16BITS(&pdh->pflags), EXTRACT_16BITS(&pdh->pIDcnt));
  907. }
  908. if (EXTRACT_16BITS(&pdh->pflags) & F_SELKEY) {
  909. op_msk |= B_KEYIN;
  910. }
  911. /* Table GET Range operation */
  912. if (EXTRACT_16BITS(&pdh->pflags) & F_SELTABRANGE) {
  913. op_msk |= B_TRNG;
  914. }
  915. /* Table SET append operation */
  916. if (EXTRACT_16BITS(&pdh->pflags) & F_TABAPPEND) {
  917. op_msk |= B_APPND;
  918. }
  919. pptr += sizeof(struct pathdata_h);
  920. len -= sizeof(struct pathdata_h);
  921. idcnt = EXTRACT_16BITS(&pdh->pIDcnt);
  922. minsize = idcnt * 4;
  923. if (len < minsize) {
  924. printf("\t\t\ttruncated IDs expected %uB got %uB\n", minsize,
  925. len);
  926. hex_print_with_offset("\t\t\tID Data[", pptr, len, 0);
  927. printf("]\n");
  928. return -1;
  929. }
  930. if ((op_msk & B_TRNG) && (op_msk & B_KEYIN)) {
  931. printf("\t\t\tIllegal to have both Table ranges and keys\n");
  932. return -1;
  933. }
  934. more_pd = pdatacnt_print(pptr, len, idcnt, op_msk, indent);
  935. if (more_pd > 0) {
  936. int consumed = len - more_pd;
  937. pptr += consumed;
  938. len = more_pd;
  939. /* XXX: Argh, recurse some more */
  940. return recpdoptlv_print(pptr, len, op_msk, indent+1);
  941. } else
  942. return 0;
  943. trunc:
  944. fputs("[|forces]", stdout);
  945. return -1;
  946. }
  947. static int
  948. genoptlv_print(register const u_char * pptr, register u_int len,
  949. u_int16_t op_msk, int indent)
  950. {
  951. const struct forces_tlv *pdtlv = (struct forces_tlv *)pptr;
  952. u_int16_t type;
  953. int tll;
  954. int invtlv;
  955. char *ib = indent_pr(indent, 0);
  956. TCHECK(*pdtlv);
  957. type = EXTRACT_16BITS(&pdtlv->type);
  958. tll = EXTRACT_16BITS(&pdtlv->length) - TLV_HDRL;
  959. invtlv = tlv_valid(pdtlv, len);
  960. printf("genoptlvprint - %s TLV type 0x%x len %d\n",
  961. tok2str(ForCES_TLV, NULL, type), type, EXTRACT_16BITS(&pdtlv->length));
  962. if (!invtlv) {
  963. /*
  964. * At this point, tlv_valid() has ensured that the TLV
  965. * length is large enough but not too large (it doesn't
  966. * go past the end of the containing TLV).
  967. */
  968. register const u_char *dp = (u_char *) TLV_DATA(pdtlv);
  969. if (!ttlv_valid(type)) {
  970. printf("%s TLV type 0x%x len %d\n",
  971. tok2str(ForCES_TLV_err, NULL, invtlv), type,
  972. EXTRACT_16BITS(&pdtlv->length));
  973. return -1;
  974. }
  975. if (vflag >= 3)
  976. printf("%s%s, length %d (data length %d Bytes)",
  977. ib, tok2str(ForCES_TLV, NULL, type),
  978. EXTRACT_16BITS(&pdtlv->length), tll);
  979. return pdata_print(dp, tll, op_msk, indent + 1);
  980. } else {
  981. printf("\t\t\tInvalid ForCES TLV type=%x", type);
  982. return -1;
  983. }
  984. trunc:
  985. fputs("[|forces]", stdout);
  986. return -1;
  987. }
  988. static int
  989. recpdoptlv_print(register const u_char * pptr, register u_int len,
  990. u_int16_t op_msk, int indent)
  991. {
  992. const struct forces_tlv *pdtlv = (struct forces_tlv *)pptr;
  993. int tll;
  994. int invtlv;
  995. u_int16_t type;
  996. register const u_char *dp;
  997. char *ib;
  998. while (len != 0) {
  999. TCHECK(*pdtlv);
  1000. invtlv = tlv_valid(pdtlv, len);
  1001. if (invtlv) {
  1002. break;
  1003. }
  1004. /*
  1005. * At this point, tlv_valid() has ensured that the TLV
  1006. * length is large enough but not too large (it doesn't
  1007. * go past the end of the containing TLV).
  1008. */
  1009. ib = indent_pr(indent, 0);
  1010. type = EXTRACT_16BITS(&pdtlv->type);
  1011. dp = (u_char *) TLV_DATA(pdtlv);
  1012. tll = EXTRACT_16BITS(&pdtlv->length) - TLV_HDRL;
  1013. if (vflag >= 3)
  1014. printf
  1015. ("%s%s, length %d (data encapsulated %d Bytes)",
  1016. ib, tok2str(ForCES_TLV, NULL, type),
  1017. EXTRACT_16BITS(&pdtlv->length),
  1018. EXTRACT_16BITS(&pdtlv->length) - TLV_HDRL);
  1019. if (pdata_print(dp, tll, op_msk, indent + 1) == -1)
  1020. return -1;
  1021. pdtlv = GO_NXT_TLV(pdtlv, len);
  1022. }
  1023. if (len) {
  1024. printf
  1025. ("\n\t\tMessy PATHDATA TLV header, type (0x%x)\n\t\texcess of %d Bytes ",
  1026. EXTRACT_16BITS(&pdtlv->type), len - EXTRACT_16BITS(&pdtlv->length));
  1027. return -1;
  1028. }
  1029. return 0;
  1030. trunc:
  1031. fputs("[|forces]", stdout);
  1032. return -1;
  1033. }
  1034. static int
  1035. invoptlv_print(register const u_char * pptr, register u_int len,
  1036. u_int16_t op_msk _U_, int indent)
  1037. {
  1038. char *ib = indent_pr(indent, 1);
  1039. if (vflag >= 3) {
  1040. printf("%sData[", &ib[1]);
  1041. hex_print_with_offset(ib, pptr, len, 0);
  1042. printf("%s]\n", ib);
  1043. }
  1044. return -1;
  1045. }
  1046. static int
  1047. otlv_print(const struct forces_tlv *otlv, u_int16_t op_msk _U_, int indent)
  1048. {
  1049. int rc = 0;
  1050. register const u_char *dp = (u_char *) TLV_DATA(otlv);
  1051. u_int16_t type;
  1052. int tll;
  1053. char *ib = indent_pr(indent, 0);
  1054. const struct optlv_h *ops;
  1055. /*
  1056. * lfbselect_print() has ensured that EXTRACT_16BITS(&otlv->length)
  1057. * >= TLV_HDRL.
  1058. */
  1059. TCHECK(*otlv);
  1060. type = EXTRACT_16BITS(&otlv->type);
  1061. tll = EXTRACT_16BITS(&otlv->length) - TLV_HDRL;
  1062. ops = get_forces_optlv_h(type);
  1063. if (vflag >= 3) {
  1064. printf("%sOper TLV %s(0x%x) length %d\n", ib, ops->s, type,
  1065. EXTRACT_16BITS(&otlv->length));
  1066. }
  1067. /* empty TLVs like COMMIT and TRCOMMIT are empty, we stop here .. */
  1068. if (!ops->flags & ZERO_TTLV) {
  1069. if (tll != 0) /* instead of "if (tll)" - for readability .. */
  1070. printf("%s: Illegal - MUST be empty\n", ops->s);
  1071. return rc;
  1072. }
  1073. /* rest of ops must at least have 12B {pathinfo} */
  1074. if (tll < OP_MIN_SIZ) {
  1075. printf("\t\tOper TLV %s(0x%x) length %d\n", ops->s, type,
  1076. EXTRACT_16BITS(&otlv->length));
  1077. printf("\t\tTruncated data size %d minimum required %d\n", tll,
  1078. OP_MIN_SIZ);
  1079. return invoptlv_print(dp, tll, ops->op_msk, indent);
  1080. }
  1081. rc = ops->print(dp, tll, ops->op_msk, indent + 1);
  1082. return rc;
  1083. trunc:
  1084. fputs("[|forces]", stdout);
  1085. return -1;
  1086. }
  1087. #define ASTDLN 4
  1088. #define ASTMCD 255
  1089. static int
  1090. asttlv_print(register const u_char * pptr, register u_int len,
  1091. u_int16_t op_msk _U_, int indent)
  1092. {
  1093. u_int32_t rescode;
  1094. u_int dlen;
  1095. char *ib = indent_pr(indent, 0);
  1096. /*
  1097. * forces_type_print() has ensured that len (the TLV length)
  1098. * >= TLV_HDRL.
  1099. */
  1100. dlen = len - TLV_HDRL;
  1101. if (dlen != ASTDLN) {
  1102. printf("illegal ASTresult-TLV: %d bytes!\n", dlen);
  1103. return -1;
  1104. }
  1105. TCHECK2(*pptr, 4);
  1106. rescode = EXTRACT_32BITS(pptr);
  1107. if (rescode > ASTMCD) {
  1108. printf("illegal ASTresult result code: %d!\n", rescode);
  1109. return -1;
  1110. }
  1111. if (vflag >= 3) {
  1112. printf("Teardown reason:\n%s", ib);
  1113. switch (rescode) {
  1114. case 0:
  1115. printf("Normal Teardown");
  1116. break;
  1117. case 1:
  1118. printf("Loss of Heartbeats");
  1119. break;
  1120. case 2:
  1121. printf("Out of bandwidth");
  1122. break;
  1123. case 3:
  1124. printf("Out of Memory");
  1125. break;
  1126. case 4:
  1127. printf("Application Crash");
  1128. break;
  1129. default:
  1130. printf("Unknown Teardown reason");
  1131. break;
  1132. }
  1133. printf("(%x)\n%s", rescode, ib);
  1134. }
  1135. return 0;
  1136. trunc:
  1137. fputs("[|forces]", stdout);
  1138. return -1;
  1139. }
  1140. #define ASRDLN 4
  1141. #define ASRMCD 3
  1142. static int
  1143. asrtlv_print(register const u_char * pptr, register u_int len,
  1144. u_int16_t op_msk _U_, int indent)
  1145. {
  1146. u_int32_t rescode;
  1147. u_int dlen;
  1148. char *ib = indent_pr(indent, 0);
  1149. /*
  1150. * forces_type_print() has ensured that len (the TLV length)
  1151. * >= TLV_HDRL.
  1152. */
  1153. dlen = len - TLV_HDRL;
  1154. if (dlen != ASRDLN) { /* id, instance, oper tlv */
  1155. printf("illegal ASRresult-TLV: %d bytes!\n", dlen);
  1156. return -1;
  1157. }
  1158. TCHECK2(*pptr, 4);
  1159. rescode = EXTRACT_32BITS(pptr);
  1160. if (rescode > ASRMCD) {
  1161. printf("illegal ASRresult result code: %d!\n", rescode);
  1162. return -1;
  1163. }
  1164. if (vflag >= 3) {
  1165. printf("\n%s", ib);
  1166. switch (rescode) {
  1167. case 0:
  1168. printf("Success ");
  1169. break;
  1170. case 1:
  1171. printf("FE ID invalid ");
  1172. break;
  1173. case 2:
  1174. printf("permission denied ");
  1175. break;
  1176. default:
  1177. printf("Unknown ");
  1178. break;
  1179. }
  1180. printf("(%x)\n%s", rescode, ib);
  1181. }
  1182. return 0;
  1183. trunc:
  1184. fputs("[|forces]", stdout);
  1185. return -1;
  1186. }
  1187. #if 0
  1188. /*
  1189. * XXX - not used.
  1190. */
  1191. static int
  1192. gentltlv_print(register const u_char * pptr _U_, register u_int len,
  1193. u_int16_t op_msk _U_, int indent _U_)
  1194. {
  1195. u_int dlen = len - TLV_HDRL;
  1196. if (dlen < 4) { /* at least 32 bits must exist */
  1197. printf("truncated TLV: %d bytes missing! ", 4 - dlen);
  1198. return -1;
  1199. }
  1200. return 0;
  1201. }
  1202. #endif
  1203. #define RD_MIN 8
  1204. static int
  1205. print_metailv(register const u_char * pptr, register u_int len,
  1206. u_int16_t op_msk _U_, int indent)
  1207. {
  1208. u_int rlen;
  1209. char *ib = indent_pr(indent, 0);
  1210. /* XXX: check header length */
  1211. const struct forces_ilv *ilv = (struct forces_ilv *)pptr;
  1212. /*
  1213. * print_metatlv() has ensured that len (what remains in the
  1214. * ILV) >= ILV_HDRL.
  1215. */
  1216. rlen = EXTRACT_32BITS(&ilv->length) - ILV_HDRL;
  1217. TCHECK(*ilv);
  1218. printf("%sMetaID 0x%x length %d\n", ib, EXTRACT_32BITS(&ilv->type),
  1219. EXTRACT_32BITS(&ilv->length));
  1220. if (vflag >= 3) {
  1221. hex_print_with_offset("\t\t[", ILV_DATA(ilv), rlen, 0);
  1222. printf(" ]\n");
  1223. }
  1224. return 0;
  1225. trunc:
  1226. fputs("[|forces]", stdout);
  1227. return -1;
  1228. }
  1229. static int
  1230. print_metatlv(register const u_char * pptr, register u_int len,
  1231. u_int16_t op_msk _U_, int indent)
  1232. {
  1233. u_int dlen;
  1234. char *ib = indent_pr(indent, 0);
  1235. u_int rlen;
  1236. const struct forces_ilv *ilv = (struct forces_ilv *)pptr;
  1237. int invilv;
  1238. /*
  1239. * redirect_print() has ensured that len (what remains in the
  1240. * TLV) >= TLV_HDRL.
  1241. */
  1242. dlen = len - TLV_HDRL;
  1243. rlen = dlen;
  1244. printf("\n%s METADATA length %d \n", ib, rlen);
  1245. while (rlen != 0) {
  1246. TCHECK(*ilv);
  1247. invilv = ilv_valid(ilv, rlen);
  1248. if (invilv) {
  1249. break;
  1250. }
  1251. /*
  1252. * At this point, ilv_valid() has ensured that the ILV
  1253. * length is large enough but not too large (it doesn't
  1254. * go past the end of the containing TLV).
  1255. */
  1256. print_metailv((u_char *) ilv, rlen, 0, indent + 1);
  1257. ilv = GO_NXT_ILV(ilv, rlen);
  1258. }
  1259. return 0;
  1260. trunc:
  1261. fputs("[|forces]", stdout);
  1262. return -1;
  1263. }
  1264. static int
  1265. print_reddata(register const u_char * pptr, register u_int len,
  1266. u_int16_t op_msk _U_, int indent _U_)
  1267. {
  1268. u_int dlen;
  1269. char *ib = indent_pr(indent, 0);
  1270. u_int rlen;
  1271. dlen = len - TLV_HDRL;
  1272. rlen = dlen;
  1273. printf("\n%s Redirect Data length %d \n", ib, rlen);
  1274. if (vflag >= 3) {
  1275. printf("\t\t[");
  1276. hex_print_with_offset("\n\t\t", pptr, rlen, 0);
  1277. printf("\n\t\t]");
  1278. }
  1279. return 0;
  1280. }
  1281. static int
  1282. redirect_print(register const u_char * pptr, register u_int len,
  1283. u_int16_t op_msk _U_, int indent)
  1284. {
  1285. const struct forces_tlv *tlv = (struct forces_tlv *)pptr;
  1286. u_int dlen;
  1287. u_int rlen;
  1288. int invtlv;
  1289. /*
  1290. * forces_type_print() has ensured that len (the TLV length)
  1291. * >= TLV_HDRL.
  1292. */
  1293. dlen = len - TLV_HDRL;
  1294. if (dlen <= RD_MIN) {
  1295. printf("\n\t\ttruncated Redirect TLV: %d bytes missing! ",
  1296. RD_MIN - dlen);
  1297. return -1;
  1298. }
  1299. rlen = dlen;
  1300. indent += 1;
  1301. while (rlen != 0) {
  1302. TCHECK(*tlv);
  1303. invtlv = tlv_valid(tlv, rlen);
  1304. if (invtlv) {
  1305. printf("Bad Redirect data\n");
  1306. break;
  1307. }
  1308. /*
  1309. * At this point, tlv_valid() has ensured that the TLV
  1310. * length is large enough but not too large (it doesn't
  1311. * go past the end of the containing TLV).
  1312. */
  1313. if (EXTRACT_16BITS(&tlv->type) == F_TLV_METD) {
  1314. print_metatlv((u_char *) TLV_DATA(tlv),
  1315. EXTRACT_16BITS(&tlv->length), 0, indent);
  1316. } else if ((EXTRACT_16BITS(&tlv->type) == F_TLV_REDD)) {
  1317. print_reddata((u_char *) TLV_DATA(tlv),
  1318. EXTRACT_16BITS(&tlv->length), 0, indent);
  1319. } else {
  1320. printf("Unknown REDIRECT TLV 0x%x len %d\n",
  1321. EXTRACT_16BITS(&tlv->type),
  1322. EXTRACT_16BITS(&tlv->length));
  1323. }
  1324. tlv = GO_NXT_TLV(tlv, rlen);
  1325. }
  1326. if (rlen) {
  1327. printf
  1328. ("\n\t\tMessy Redirect TLV header, type (0x%x)\n\t\texcess of %d Bytes ",
  1329. EXTRACT_16BITS(&tlv->type),
  1330. rlen - EXTRACT_16BITS(&tlv->length));
  1331. return -1;
  1332. }
  1333. return 0;
  1334. trunc:
  1335. fputs("[|forces]", stdout);
  1336. return -1;
  1337. }
  1338. #define OP_OFF 8
  1339. #define OP_MIN 12
  1340. static int
  1341. lfbselect_print(register const u_char * pptr, register u_int len,
  1342. u_int16_t op_msk, int indent)
  1343. {
  1344. const struct forces_lfbsh *lfbs;
  1345. const struct forces_tlv *otlv;
  1346. char *ib = indent_pr(indent, 0);
  1347. u_int dlen;
  1348. u_int rlen;
  1349. int invtlv;
  1350. /*
  1351. * forces_type_print() has ensured that len (the TLV length)
  1352. * >= TLV_HDRL.
  1353. */
  1354. dlen = len - TLV_HDRL;
  1355. if (dlen <= OP_MIN) { /* id, instance, oper tlv header .. */
  1356. printf("\n\t\ttruncated lfb selector: %d bytes missing! ",
  1357. OP_MIN - dlen);
  1358. return -1;
  1359. }
  1360. /*
  1361. * At this point, we know that dlen > OP_MIN; OP_OFF < OP_MIN, so
  1362. * we also know that it's > OP_OFF.
  1363. */
  1364. rlen = dlen - OP_OFF;
  1365. lfbs = (const struct forces_lfbsh *)pptr;
  1366. TCHECK(*lfbs);
  1367. if (vflag >= 3) {
  1368. printf("\n%s%s(Classid %x) instance %x\n",
  1369. ib, tok2str(ForCES_LFBs, NULL, EXTRACT_32BITS(&lfbs->class)),
  1370. EXTRACT_32BITS(&lfbs->class),
  1371. EXTRACT_32BITS(&lfbs->instance));
  1372. }
  1373. otlv = (struct forces_tlv *)(lfbs + 1);
  1374. indent += 1;
  1375. while (rlen != 0) {
  1376. TCHECK(*otlv);
  1377. invtlv = tlv_valid(otlv, rlen);
  1378. if (invtlv)
  1379. break;
  1380. /*
  1381. * At this point, tlv_valid() has ensured that the TLV
  1382. * length is large enough but not too large (it doesn't
  1383. * go past the end of the containing TLV).
  1384. */
  1385. if (op_valid(EXTRACT_16BITS(&otlv->type), op_msk)) {
  1386. otlv_print(otlv, 0, indent);
  1387. } else {
  1388. if (vflag < 3)
  1389. printf("\n");
  1390. printf
  1391. ("\t\tINValid oper-TLV type 0x%x length %d for this ForCES message\n",
  1392. EXTRACT_16BITS(&otlv->type), EXTRACT_16BITS(&otlv->length));
  1393. invoptlv_print((u_char *)otlv, rlen, 0, indent);
  1394. }
  1395. otlv = GO_NXT_TLV(otlv, rlen);
  1396. }
  1397. if (rlen) {
  1398. printf
  1399. ("\n\t\tMessy oper TLV header, type (0x%x)\n\t\texcess of %d Bytes ",
  1400. EXTRACT_16BITS(&otlv->type), rlen - EXTRACT_16BITS(&otlv->length));
  1401. return -1;
  1402. }
  1403. return 0;
  1404. trunc:
  1405. fputs("[|forces]", stdout);
  1406. return -1;
  1407. }
  1408. static int
  1409. forces_type_print(register const u_char * pptr, const struct forcesh *fhdr _U_,
  1410. register u_int mlen, const struct tom_h *tops)
  1411. {
  1412. const struct forces_tlv *tltlv;
  1413. u_int rlen;
  1414. int invtlv;
  1415. int rc = 0;
  1416. int ttlv = 0;
  1417. /*
  1418. * forces_print() has already checked that mlen >= ForCES_HDRL
  1419. * by calling ForCES_HLN_VALID().
  1420. */
  1421. rlen = mlen - ForCES_HDRL;
  1422. if (rlen > TLV_HLN) {
  1423. if (tops->flags & ZERO_TTLV) {
  1424. printf("<0x%x>Illegal Top level TLV!\n", tops->flags);
  1425. return -1;
  1426. }
  1427. } else {
  1428. if (tops->flags & ZERO_MORE_TTLV)
  1429. return 0;
  1430. if (tops->flags & ONE_MORE_TTLV) {
  1431. printf("\tTop level TLV Data missing!\n");
  1432. return -1;
  1433. }
  1434. }
  1435. if (tops->flags & ZERO_TTLV) {
  1436. return 0;
  1437. }
  1438. ttlv = tops->flags >> 4;
  1439. tltlv = GET_TOP_TLV(pptr);
  1440. /*XXX: 15 top level tlvs will probably be fine
  1441. You are nuts if you send more ;-> */
  1442. while (rlen != 0) {
  1443. TCHECK(*tltlv);
  1444. invtlv = tlv_valid(tltlv, rlen);
  1445. if (invtlv)
  1446. break;
  1447. /*
  1448. * At this point, tlv_valid() has ensured that the TLV
  1449. * length is large enough but not too large (it doesn't
  1450. * go past the end of the packet).
  1451. */
  1452. if (!ttlv_valid(EXTRACT_16BITS(&tltlv->type))) {
  1453. printf("\n\tInvalid ForCES Top TLV type=0x%x",
  1454. EXTRACT_16BITS(&tltlv->type));
  1455. return -1;
  1456. }
  1457. if (vflag >= 3)
  1458. printf("\t%s, length %d (data length %d Bytes)",
  1459. tok2str(ForCES_TLV, NULL, EXTRACT_16BITS(&tltlv->type)),
  1460. EXTRACT_16BITS(&tltlv->length),
  1461. EXTRACT_16BITS(&tltlv->length) - TLV_HDRL);
  1462. rc = tops->print((u_char *) TLV_DATA(tltlv),
  1463. EXTRACT_16BITS(&tltlv->length), tops->op_msk, 9);
  1464. if (rc < 0) {
  1465. return -1;
  1466. }
  1467. tltlv = GO_NXT_TLV(tltlv, rlen);
  1468. ttlv--;
  1469. if (ttlv <= 0)
  1470. break;
  1471. }
  1472. /*
  1473. * XXX - if ttlv != 0, does that mean that the packet was too
  1474. * short, and didn't have *enough* TLVs in it?
  1475. */
  1476. if (rlen) {
  1477. printf("\tMess TopTLV header: min %u, total %d advertised %d ",
  1478. TLV_HDRL, rlen, EXTRACT_16BITS(&tltlv->length));
  1479. return -1;
  1480. }
  1481. return 0;
  1482. trunc:
  1483. fputs("[|forces]", stdout);
  1484. return -1;
  1485. }
  1486. void
  1487. forces_print(register const u_char * pptr, register u_int len)
  1488. {
  1489. const struct forcesh *fhdr;
  1490. u_int mlen;
  1491. u_int32_t flg_raw;
  1492. const struct tom_h *tops;
  1493. int rc = 0;
  1494. fhdr = (const struct forcesh *)pptr;
  1495. TCHECK(*fhdr);
  1496. if (!tom_valid(fhdr->fm_tom)) {
  1497. printf("Invalid ForCES message type %d\n", fhdr->fm_tom);
  1498. goto error;
  1499. }
  1500. mlen = ForCES_BLN(fhdr);
  1501. tops = get_forces_tom(fhdr->fm_tom);
  1502. if (tops->v == TOM_RSVD) {
  1503. printf("\n\tUnknown ForCES message type=0x%x", fhdr->fm_tom);
  1504. goto error;
  1505. }
  1506. printf("\n\tForCES %s ", tops->s);
  1507. if (!ForCES_HLN_VALID(mlen, len)) {
  1508. printf
  1509. ("Illegal ForCES pkt len - min %u, total recvd %d, advertised %d ",
  1510. ForCES_HDRL, len, ForCES_BLN(fhdr));
  1511. goto error;
  1512. }
  1513. TCHECK2(*(pptr + 20), 4);
  1514. flg_raw = EXTRACT_32BITS(pptr + 20);
  1515. if (vflag >= 1) {
  1516. printf("\n\tForCES Version %d len %uB flags 0x%08x ",
  1517. ForCES_V(fhdr), mlen, flg_raw);
  1518. printf("\n\tSrcID 0x%x(%s) DstID 0x%x(%s) Correlator 0x%" PRIx64,
  1519. ForCES_SID(fhdr), ForCES_node(ForCES_SID(fhdr)),
  1520. ForCES_DID(fhdr), ForCES_node(ForCES_DID(fhdr)),
  1521. EXTRACT_64BITS(fhdr->fm_cor));
  1522. }
  1523. if (vflag >= 2) {
  1524. printf
  1525. ("\n\tForCES flags:\n\t %s(0x%x), prio=%d, %s(0x%x),\n\t %s(0x%x), %s(0x%x)\n",
  1526. ForCES_ACKp(ForCES_ACK(fhdr)), ForCES_ACK(fhdr),
  1527. ForCES_PRI(fhdr),
  1528. ForCES_EMp(ForCES_EM(fhdr)), ForCES_EM(fhdr),
  1529. ForCES_ATp(ForCES_AT(fhdr)), ForCES_AT(fhdr),
  1530. ForCES_TPp(ForCES_TP(fhdr)), ForCES_TP(fhdr));
  1531. printf
  1532. ("\t Extra flags: rsv(b5-7) 0x%x rsv(b13-31) 0x%x\n",
  1533. ForCES_RS1(fhdr), ForCES_RS2(fhdr));
  1534. }
  1535. rc = forces_type_print(pptr, fhdr, mlen, tops);
  1536. if (rc < 0) {
  1537. error:
  1538. hex_print_with_offset("\n\t[", pptr, len, 0);
  1539. printf("\n\t]");
  1540. return;
  1541. }
  1542. if (vflag >= 4) {
  1543. printf("\n\t Raw ForCES message\n\t [");
  1544. hex_print_with_offset("\n\t ", pptr, len, 0);
  1545. printf("\n\t ]");
  1546. }
  1547. printf("\n");
  1548. return;
  1549. trunc:
  1550. fputs("[|forces]", stdout);
  1551. }