2 * Routines for Sinec H1 packet disassembly
3 * Gerrit Gehnen <G.Gehnen@atrie.de>
5 * Wireshark - Network traffic analyzer
6 * By Gerald Combs <gerald@wireshark.org>
7 * Copyright 1998 Gerald Combs
9 * SPDX-License-Identifier: GPL-2.0-or-later
14 #include <epan/packet.h>
15 #include <epan/exceptions.h>
17 void proto_register_h1(void);
18 void proto_reg_handoff_h1(void);
21 static int hf_h1_header
;
23 static int hf_h1_block_type
;
24 static int hf_h1_block_len
;
25 static int hf_h1_opcode
;
26 static int hf_h1_dbnr
;
27 static int hf_h1_dwnr
;
28 static int hf_h1_dlen
;
30 static int hf_h1_response_value
;
33 #define EMPTY_BLOCK 0xFF
34 #define OPCODE_BLOCK 0x01
35 #define REQUEST_BLOCK 0x03
36 #define RESPONSE_BLOCK 0x0F
38 static const value_string block_type_vals
[] = {
39 { EMPTY_BLOCK
, "Empty Block" },
40 { OPCODE_BLOCK
, "Opcode Block" },
41 { REQUEST_BLOCK
, "Request Block" },
42 { RESPONSE_BLOCK
, "Response Block" },
47 static const value_string opcode_vals
[] = {
49 {4, "Write Response"},
55 static const value_string org_vals
[] = {
71 static const value_string returncode_vals
[] = {
73 {0x02, "Requested block does not exist"},
74 {0x03, "Requested block too small"},
75 {0xFF, "Error, reason unknown"},
82 static bool dissect_h1(tvbuff_t
*tvb
, packet_info
*pinfo
, proto_tree
*tree
, void *data _U_
)
84 proto_tree
*h1_tree
, *block_tree
;
85 proto_item
*h1_ti
, *block_ti
;
86 int offset
= 0, offset_block_start
;
88 uint8_t block_type
, block_len
;
91 if (tvb_captured_length(tvb
) < 2) {
92 /* Not enough data captured to hold the "S5" header; don't try
93 to interpret it as H1. */
97 if (!(tvb_get_uint8(tvb
, 0) == 'S' && tvb_get_uint8(tvb
, 1) == '5')) {
101 col_set_str (pinfo
->cinfo
, COL_PROTOCOL
, "H1");
102 col_set_str(pinfo
->cinfo
, COL_INFO
, "S5: ");
104 h1_ti
= proto_tree_add_item(tree
, proto_h1
, tvb
, offset
, -1, ENC_NA
);
105 h1_tree
= proto_item_add_subtree(h1_ti
, ett_h1
);
107 proto_tree_add_item(h1_tree
, hf_h1_header
, tvb
, offset
, 2, ENC_BIG_ENDIAN
);
110 h1_len
= tvb_get_uint8(tvb
, offset
);
111 proto_tree_add_item(h1_tree
, hf_h1_len
, tvb
, offset
, 1, ENC_BIG_ENDIAN
);
112 proto_item_set_len(h1_ti
, h1_len
);
115 while (offset
< h1_len
) {
116 offset_block_start
= offset
;
118 block_type
= tvb_get_uint8(tvb
, offset
);
119 block_len
= tvb_get_uint8(tvb
, offset
+1);
121 if (!try_val_to_str(block_type
, block_type_vals
)) {
122 /* XXX - should we skip unknown blocks? */
125 if (block_len
== 0) {
126 /* XXX - expert info */
130 block_tree
= proto_tree_add_subtree_format(h1_tree
,
131 tvb
, offset
, -1, ett_block
, &block_ti
, "%s",
132 val_to_str_const(block_type
, block_type_vals
, "Unknown block"));
134 proto_tree_add_item(block_tree
, hf_h1_block_type
,
135 tvb
, offset
, 1, ENC_BIG_ENDIAN
);
136 /* we keep increasing offset as we go though the block
137 however, to find the beginning of the next block,
138 we use the current block's start offset and its length field */
140 proto_tree_add_item(block_tree
, hf_h1_block_len
,
141 tvb
, offset
, 1, ENC_BIG_ENDIAN
);
142 proto_item_set_len(block_ti
, block_len
);
145 switch (block_type
) {
147 proto_tree_add_item(block_tree
, hf_h1_opcode
,
148 tvb
, offset
, 1, ENC_BIG_ENDIAN
);
149 col_append_str (pinfo
->cinfo
, COL_INFO
,
150 val_to_str (tvb_get_uint8(tvb
, offset
),
151 opcode_vals
, "Unknown Opcode (0x%2.2x)"));
155 proto_tree_add_item(block_tree
, hf_h1_org
, tvb
,
156 offset
, 1, ENC_BIG_ENDIAN
);
157 col_append_fstr(pinfo
->cinfo
, COL_INFO
, " %s",
158 val_to_str (tvb_get_uint8(tvb
, offset
),
159 org_vals
,"Unknown Type (0x%2.2x)"));
162 proto_tree_add_item(block_tree
, hf_h1_dbnr
, tvb
,
163 offset
, 1, ENC_BIG_ENDIAN
);
164 col_append_fstr(pinfo
->cinfo
, COL_INFO
, " %d",
165 tvb_get_uint8(tvb
, offset
));
168 proto_tree_add_item(block_tree
, hf_h1_dwnr
, tvb
,
169 offset
, 2, ENC_BIG_ENDIAN
);
170 col_append_fstr (pinfo
->cinfo
, COL_INFO
, " DW %d",
171 tvb_get_ntohs(tvb
, offset
));
174 proto_tree_add_item(block_tree
, hf_h1_dlen
, tvb
,
175 offset
, 2, ENC_BIG_ENDIAN
);
176 col_append_fstr (pinfo
->cinfo
, COL_INFO
, " Count %d",
177 tvb_get_ntohs(tvb
, offset
));
181 proto_tree_add_item(block_tree
, hf_h1_response_value
,
182 tvb
, offset
, 1, ENC_BIG_ENDIAN
);
183 col_append_fstr (pinfo
->cinfo
, COL_INFO
, " %s",
184 val_to_str (tvb_get_uint8(tvb
, offset
),
185 returncode_vals
,"Unknown Returncode (0x%2.2x"));
189 offset
= offset_block_start
+ block_len
; /* see the comment above */
192 if (tvb_reported_length_remaining(tvb
, offset
) > 0) {
193 next_tvb
= tvb_new_subset_remaining(tvb
, offset
);
194 call_data_dissector(next_tvb
, pinfo
, tree
);
202 proto_register_h1 (void)
204 static hf_register_info hf
[] = {
206 {"H1-Header", "h1.header", FT_UINT16
, BASE_HEX
, NULL
, 0x0,
209 {"Length indicator", "h1.len", FT_UINT16
, BASE_DEC
, NULL
, 0x0,
212 {"Block type", "h1.block_type", FT_UINT8
, BASE_HEX
, VALS(block_type_vals
), 0x0,
215 {"Block length", "h1.block_len", FT_UINT8
, BASE_DEC
, NULL
, 0x0, NULL
, HFILL
}},
217 {"Opcode", "h1.opcode", FT_UINT8
, BASE_HEX
, VALS (opcode_vals
), 0x0,
220 {"Memory type", "h1.org", FT_UINT8
, BASE_HEX
, VALS (org_vals
), 0x0,
223 {"Memory block number", "h1.dbnr", FT_UINT8
, BASE_DEC
, NULL
, 0x0, NULL
, HFILL
}},
225 {"Address within memory block", "h1.dwnr", FT_UINT16
, BASE_DEC
, NULL
, 0x0,
228 {"Length in words", "h1.dlen", FT_INT16
, BASE_DEC
, NULL
, 0x0, NULL
, HFILL
}},
229 {&hf_h1_response_value
,
230 {"Response value", "h1.resvalue", FT_UINT8
, BASE_DEC
,
231 VALS (returncode_vals
), 0x0, NULL
, HFILL
}}
234 static int *ett
[] = {
239 proto_h1
= proto_register_protocol ("Sinec H1 Protocol", "H1", "h1");
240 proto_register_field_array (proto_h1
, hf
, array_length (hf
));
241 proto_register_subtree_array (ett
, array_length (ett
));
245 proto_reg_handoff_h1(void)
247 heur_dissector_add("cotp", dissect_h1
,
248 "Sinec H1 over COTP", "hi_cotp", proto_h1
, HEURISTIC_ENABLE
);
249 heur_dissector_add("cotp_is", dissect_h1
,
250 "Sinec H1 over COTP (inactive subset)", "hi_cotp_is", proto_h1
, HEURISTIC_ENABLE
);
251 heur_dissector_add("tcp", dissect_h1
,
252 "Sinec H1 over TCP", "hi_tcp", proto_h1
, HEURISTIC_ENABLE
);
256 * Editor modelines - https://www.wireshark.org/tools/modelines.html
261 * indent-tabs-mode: nil
264 * ex: set shiftwidth=4 tabstop=8 expandtab:
265 * :indentSize=4:tabSize=8:noTabs=true: