summaryrefslogtreecommitdiffstats
path: root/epan/dissectors/packet-vrt.c
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-10 20:34:10 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-10 20:34:10 +0000
commite4ba6dbc3f1e76890b22773807ea37fe8fa2b1bc (patch)
tree68cb5ef9081156392f1dd62a00c6ccc1451b93df /epan/dissectors/packet-vrt.c
parentInitial commit. (diff)
downloadwireshark-e4ba6dbc3f1e76890b22773807ea37fe8fa2b1bc.tar.xz
wireshark-e4ba6dbc3f1e76890b22773807ea37fe8fa2b1bc.zip
Adding upstream version 4.2.2.upstream/4.2.2
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'epan/dissectors/packet-vrt.c')
-rw-r--r--epan/dissectors/packet-vrt.c2479
1 files changed, 2479 insertions, 0 deletions
diff --git a/epan/dissectors/packet-vrt.c b/epan/dissectors/packet-vrt.c
new file mode 100644
index 00000000..13c4cae7
--- /dev/null
+++ b/epan/dissectors/packet-vrt.c
@@ -0,0 +1,2479 @@
+/* packet-vrt.c
+ * Routines for VRT (VITA 49) packet disassembly
+ * Copyright 2012 Ettus Research LLC - Nick Foster <nick@ettus.com>: original dissector
+ * Copyright 2013 Alexander Chemeris <alexander.chemeris@gmail.com>: dissector improvement
+ * Copyright 2013 Dario Lombardo (lomato@gmail.com): Official Wireshark port
+ * Copyright 2022 Amazon.com, Inc. or its affiliates - Cody Planteen <codplant@amazon.com>: context packet decoding
+ *
+ * Original dissector repository: https://github.com/bistromath/vrt-dissector
+ *
+ * Wireshark - Network traffic analyzer
+ * By Gerald Combs <gerald@wireshark.org>
+ * Copyright 1998 Gerald Combs
+ *
+ * SPDX-License-Identifier: GPL-2.0-or-later
+ */
+
+
+#include "config.h"
+#include <epan/packet.h>
+#include <epan/prefs.h>
+#include <math.h>
+
+void proto_register_vrt(void);
+void proto_reg_handoff_vrt(void);
+
+static dissector_handle_t vrt_handle;
+
+#define VITA_49_PORT 4991
+#define DEFAULT_EPHEMERIS_FIELDS { -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1 }
+#define DEFAULT_FORMATTED_GPS_INS_FIELDS { -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1 }
+
+typedef int (*complex_dissector_t)(proto_tree *tree, tvbuff_t *tvb, int offset);
+
+typedef struct {
+ int tsi; /* 2-bit timestamp type */
+ int tsf; /* 2-bit fractional timestamp type */
+ int oui; /* 24-bit GPS/INS manufacturer OUI */
+ int ts_int; /* 32-bit integer timestamp (opt.) */
+ int ts_picosecond; /* 64-bit fractional timestamp (mutually exclusive with below) */
+ int ts_frac_sample; /* 64-bit fractional timestamp (mutually exclusive with above) */
+ int pos_x; /* 32-bit position X */
+ int pos_y; /* 32-bit position Y */
+ int pos_z; /* 32-bit position Z */
+ int att_alpha; /* 32-bit attitude alpha */
+ int att_beta; /* 32-bit attitude beta */
+ int att_phi; /* 32-bit attitude phi */
+ int vel_dx; /* 32-bit velocity dX */
+ int vel_dy; /* 32-bit velocity dY */
+ int vel_dz; /* 32-bit velocity dZ */
+} ephemeris_fields;
+
+typedef struct {
+ int tsi; /* 2-bit timestamp type */
+ int tsf; /* 2-bit fractional timestamp type */
+ int oui; /* 24-bit GPS/INS manufacturer OUI */
+ int ts_int; /* 32-bit integer timestamp (opt.) */
+ int ts_picosecond; /* 64-bit fractional timestamp (mutually exclusive with below) */
+ int ts_frac_sample; /* 64-bit fractional timestamp (mutually exclusive with above) */
+ int lat; /* 32-bit latitude */
+ int lon; /* 32-bit longitude */
+ int alt; /* 32-bit altitude */
+ int speed; /* 32-bit speed over ground */
+ int heading; /* 32-bit heading angle */
+ int track; /* 32-bit track angle */
+ int mag_var; /* 32-bit magnetic variation */
+} formatted_gps_ins_fields;
+
+typedef int (*complex_dissector_t)(proto_tree *tree, tvbuff_t *tvb, int offset);
+
+static gboolean vrt_use_ettus_uhd_header_format = FALSE;
+
+static int proto_vrt = -1;
+
+/* fields */
+static int hf_vrt_header = -1; /* 32-bit header */
+static int hf_vrt_type = -1; /* 4-bit pkt type */
+static int hf_vrt_cidflag = -1; /* 1-bit class ID flag */
+static int hf_vrt_tflag = -1; /* 1-bit trailer flag */
+static int hf_vrt_tsmflag = -1; /* 1-bit timestamp mode */
+static int hf_vrt_tsi = -1; /* 2-bit timestamp type */
+static int hf_vrt_tsf = -1; /* 2-bit fractional timestamp type */
+static int hf_vrt_seq = -1; /* 4-bit sequence number */
+static int hf_vrt_len = -1; /* 16-bit length */
+static int hf_vrt_sid = -1; /* 32-bit stream ID (opt.) */
+static int hf_vrt_cid = -1; /* 64-bit class ID (opt.) */
+static int hf_vrt_cid_oui = -1; /* 24-bit class ID OUI */
+static int hf_vrt_cid_icc = -1; /* 16-bit class ID ICC */
+static int hf_vrt_cid_pcc = -1; /* 16-bit class ID PCC */
+static int hf_vrt_cif[8] = { -1, -1, -1, -1, -1, -1, -1, -1}; /* 32-bit CIF0-CIF7 (opt.) */
+static int hf_vrt_cif0_change_flag = -1; /* 1-bit context field change indicator */
+static int hf_vrt_cif0_ref_pt_id = -1; /* 1-bit reference point identifier */
+static int hf_vrt_cif0_bandwidth = -1; /* 1-bit bandwidth */
+static int hf_vrt_cif0_if_freq = -1; /* 1-bit IF reference frequency */
+static int hf_vrt_cif0_rf_freq = -1; /* 1-bit RF reference frequency */
+static int hf_vrt_cif0_rf_freq_offset = -1; /* 1-bit RF reference frequency offset */
+static int hf_vrt_cif0_if_band_offset = -1; /* 1-bit IF band offset */
+static int hf_vrt_cif0_ref_level = -1; /* 1-bit reference level */
+static int hf_vrt_cif0_gain = -1; /* 1-bit gain */
+static int hf_vrt_cif0_over_range_count = -1; /* 1-bit over-range count */
+static int hf_vrt_cif0_sample_rate = -1; /* 1-bit sample rate */
+static int hf_vrt_cif0_timestamp_adjust = -1; /* 1-bit timestamp adjustment */
+static int hf_vrt_cif0_timestamp_cal = -1; /* 1-bit timestamp calibration time */
+static int hf_vrt_cif0_temperature = -1; /* 1-bit temperature */
+static int hf_vrt_cif0_device_id = -1; /* 1-bit device identifier */
+static int hf_vrt_cif0_state_event = -1; /* 1-bit state/event indicators */
+static int hf_vrt_cif0_signal_data_format = -1; /* 1-bit signal data packet payload format */
+static int hf_vrt_cif0_gps = -1; /* 1-bit formatted GPS */
+static int hf_vrt_cif0_ins = -1; /* 1-bit formatted INS */
+static int hf_vrt_cif0_ecef_ephemeris = -1; /* 1-bit ECEF ephemeris */
+static int hf_vrt_cif0_rel_ephemeris = -1; /* 1-bit relative ephemeris */
+static int hf_vrt_cif0_ephemeris_ref_id = -1; /* 1-bit ephemeris ref ID */
+static int hf_vrt_cif0_gps_ascii = -1; /* 1-bit GPS ASCII */
+static int hf_vrt_cif0_context_assoc_lists = -1; /* 1-bit context association lists */
+static int hf_vrt_cif0_cif7 = -1; /* 1-bit CIF7 */
+static int hf_vrt_cif0_cif6 = -1; /* 1-bit CIF6 */
+static int hf_vrt_cif0_cif5 = -1; /* 1-bit CIF5 */
+static int hf_vrt_cif0_cif4 = -1; /* 1-bit CIF4 */
+static int hf_vrt_cif0_cif3 = -1; /* 1-bit CIF3 */
+static int hf_vrt_cif0_cif2 = -1; /* 1-bit CIF2 */
+static int hf_vrt_cif0_cif1 = -1; /* 1-bit CIF1 */
+/* TODO: complete CIF1 support (have partial CIF1 support) */
+static int hf_vrt_cif1_phase_offset = -1; /* 1-bit phase offset */
+static int hf_vrt_cif1_polarization = -1; /* 1-bit polarization */
+static int hf_vrt_cif1_range = -1; /* 1-bit range (distance) */
+static int hf_vrt_cif1_aux_freq = -1; /* 1-bit aux frequency */
+static int hf_vrt_cif1_aux_bandwidth = -1; /* 1-bit aux bandwidth */
+static int hf_vrt_cif1_io32 = -1; /* 1-bit discrete I/O (32-bit) */
+static int hf_vrt_cif1_io64 = -1; /* 1-bit discrete I/O (64-bit) */
+static int hf_vrt_cif1_v49_spec = -1; /* 1-bit V49 spec compliance */
+static int hf_vrt_cif1_ver = -1; /* 1-bit version and build code */
+static int hf_vrt_context_ref_pt_id = -1; /* 32-bit reference point identifier */
+static int hf_vrt_context_bandwidth = -1; /* 64-bit bandwidth */
+static int hf_vrt_context_if_freq = -1; /* 64-bit IF reference frequency */
+static int hf_vrt_context_rf_freq = -1; /* 64-bit RF reference frequency */
+static int hf_vrt_context_rf_freq_offset = -1; /* 64-bit RF frequency offset */
+static int hf_vrt_context_if_band_offset = -1; /* 64-bit IF band offset */
+static int hf_vrt_context_ref_level = -1; /* 16-bit reference level */
+static int hf_vrt_context_gain_stage2 = -1; /* 16-bit gain stage 2 */
+static int hf_vrt_context_gain_stage1 = -1; /* 16-bit gain stage 1 */
+static int hf_vrt_context_over_range_count = -1; /* 32-bit over-range count */
+static int hf_vrt_context_sample_rate = -1; /* 64-bit sample rate */
+static int hf_vrt_context_timestamp_adjust = -1; /* 64-bit timestamp adjustment */
+static int hf_vrt_context_timestamp_cal = -1; /* 32-bit timestamp calibration */
+static int hf_vrt_context_temperature = -1; /* 16-bit device temperature */
+static int hf_vrt_context_device_id_oui = -1; /* 24-bit device ID OUI */
+static int hf_vrt_context_device_id_code = -1; /* 16-bit device ID code */
+static int hf_vrt_context_state_event_en_cal_time = -1; /* 1-bit enable calibrated time */
+static int hf_vrt_context_state_event_en_valid_data = -1; /* 1-bit enable valid data */
+static int hf_vrt_context_state_event_en_ref_lock = -1; /* 1-bit enable reference lock */
+static int hf_vrt_context_state_event_en_agc = -1; /* 1-bit enable AGC/MGC */
+static int hf_vrt_context_state_event_en_detected_sig = -1; /* 1-bit enable detected signal */
+static int hf_vrt_context_state_event_en_spectral_inv = -1; /* 1-bit enable spectral inversion */
+static int hf_vrt_context_state_event_en_over_range = -1; /* 1-bit enable over-range */
+static int hf_vrt_context_state_event_en_sample_loss = -1; /* 1-bit enable sample loss */
+static int hf_vrt_context_state_event_cal_time = -1; /* 1-bit enable calibrated time */
+static int hf_vrt_context_state_event_valid_data = -1; /* 1-bit enable valid data */
+static int hf_vrt_context_state_event_ref_lock = -1; /* 1-bit enable reference lock */
+static int hf_vrt_context_state_event_agc = -1; /* 1-bit enable AGC/MGC */
+static int hf_vrt_context_state_event_detected_sig = -1; /* 1-bit enable detected signal */
+static int hf_vrt_context_state_event_spectral_inv = -1; /* 1-bit enable spectral inversion */
+static int hf_vrt_context_state_event_over_range = -1; /* 1-bit enable over-range */
+static int hf_vrt_context_state_event_sample_loss = -1; /* 1-bit enable sample loss */
+static int hf_vrt_context_state_event_user = -1; /* 8-bit user-defined */
+static int hf_vrt_context_signal_data_format_packing = -1; /* 1-bit signal data format packing */
+static int hf_vrt_context_signal_data_format_type = -1; /* 2-bit real/complex type */
+static int hf_vrt_context_signal_data_format_item = -1; /* 5-bit data item format */
+static int hf_vrt_context_signal_data_format_repeat = -1; /* 1-bit sample-component repeat indicator */
+static int hf_vrt_context_signal_data_format_event_size = -1; /* 3-bit event-tag size */
+static int hf_vrt_context_signal_data_format_channel_size = -1; /* 4-bit channel-tag size */
+static int hf_vrt_context_signal_data_format_fraction_size = -1; /* 4-bit data item fraction size */
+static int hf_vrt_context_signal_data_format_packing_size = -1; /* 6-bit item packing field size */
+static int hf_vrt_context_signal_data_format_item_size = -1; /* 6-bit data item size */
+static int hf_vrt_context_signal_data_format_repeat_count = -1; /* 16-bit repeat count */
+static int hf_vrt_context_signal_data_format_vector_size = -1; /* 16-bit vector size */
+static formatted_gps_ins_fields hf_vrt_context_gps = DEFAULT_FORMATTED_GPS_INS_FIELDS; /* struct for formatted GPS */
+static formatted_gps_ins_fields hf_vrt_context_ins = DEFAULT_FORMATTED_GPS_INS_FIELDS; /* struct for formatted INS */
+static ephemeris_fields hf_vrt_context_ecef_ephemeris = DEFAULT_EPHEMERIS_FIELDS; /* struct for ECEF ephemeris */
+static ephemeris_fields hf_vrt_context_rel_ephemeris = DEFAULT_EPHEMERIS_FIELDS; /* struct for relative ephemeris */
+static int hf_vrt_context_ephemeris_ref_id = -1; /* 32-bit ephemeris reference identifier */
+static int hf_vrt_context_gps_ascii_oui; /* 24-bit GPS/INS manufacturer OUI */
+static int hf_vrt_context_gps_ascii_size; /* 32-bit number of words */
+static int hf_vrt_context_gps_ascii_data = -1; /* Variable GPS ASCII data */
+static int hf_vrt_context_assoc_lists_src_size; /* 32-bit source list size */
+static int hf_vrt_context_assoc_lists_sys_size; /* 32-bit system list size */
+static int hf_vrt_context_assoc_lists_vec_size; /* 32-bit vector-component list size */
+static int hf_vrt_context_assoc_lists_a; /* 1-bit "A" bit (asynchronous-channel tag list present) */
+static int hf_vrt_context_assoc_lists_asy_size; /* 32-bit asynchronous-channel list size */
+static int hf_vrt_context_assoc_lists_src_data; /* Variable source context association list */
+static int hf_vrt_context_assoc_lists_sys_data; /* Variable system context association list */
+static int hf_vrt_context_assoc_lists_vec_data; /* Variable vector-component context association list */
+static int hf_vrt_context_assoc_lists_asy_data; /* Variable asynchronous-channel context association list */
+static int hf_vrt_context_assoc_lists_asy_tag_data; /* Variable asynchronous-channel tag list */
+static int hf_vrt_context_phase_offset = -1; /* 16-bit phase offset */
+static int hf_vrt_context_pol_tilt = -1; /* 16-bit polarization tilt angle */
+static int hf_vrt_context_pol_ellipticity = -1; /* 16-bit polarization ellipticity angle */
+static int hf_vrt_context_range = -1; /* 32-bit range (distance) */
+static int hf_vrt_context_aux_freq = -1; /* 64-bit aux frequency */
+static int hf_vrt_context_aux_bandwidth = -1; /* 64-bit aux bandwidth */
+static int hf_vrt_context_io32 = -1; /* 32-bit discrete I/O */
+static int hf_vrt_context_io64 = -1; /* 64-bit discrete I/O */
+static int hf_vrt_context_v49_spec = -1; /* 32-bit V49 spec compliance */
+static int hf_vrt_context_ver_year = -1; /* 7-bit year */
+static int hf_vrt_context_ver_day = -1; /* 9-bit day */
+static int hf_vrt_context_ver_rev = -1; /* 6-bit revision */
+static int hf_vrt_context_ver_user = -1; /* 10-bit user defined */
+static int hf_vrt_ts_int = -1; /* 32-bit integer timestamp (opt.) */
+static int hf_vrt_ts_frac_picosecond = -1; /* 64-bit fractional timestamp (opt.) */
+static int hf_vrt_ts_frac_sample = -1; /* 64-bit fractional timestamp (opt.) */
+static int hf_vrt_data = -1; /* data */
+static int hf_vrt_trailer = -1; /* 32-bit trailer (opt.) */
+static int hf_vrt_trailer_enables = -1; /* trailer indicator enables */
+static int hf_vrt_trailer_ind = -1; /* trailer indicators */
+static int hf_vrt_trailer_e = -1; /* ass con pac cnt enable */
+static int hf_vrt_trailer_acpc = -1; /* associated context packet count */
+static int hf_vrt_trailer_en_caltime = -1; /* calibrated time indicator */
+static int hf_vrt_trailer_en_valid = -1; /* valid data ind */
+static int hf_vrt_trailer_en_reflock = -1; /* reference locked ind */
+static int hf_vrt_trailer_en_agc = -1; /* AGC/MGC enabled ind */
+static int hf_vrt_trailer_en_sig = -1; /* signal detected ind */
+static int hf_vrt_trailer_en_inv = -1; /* spectral inversion ind */
+static int hf_vrt_trailer_en_overrng = -1; /* overrange indicator */
+static int hf_vrt_trailer_en_sampleloss = -1; /* sample loss indicator */
+static int hf_vrt_trailer_en_user0 = -1; /* User indicator 0 */
+static int hf_vrt_trailer_en_user1 = -1; /* User indicator 1 */
+static int hf_vrt_trailer_en_user2 = -1; /* User indicator 2 */
+static int hf_vrt_trailer_en_user3 = -1; /* User indicator 3 */
+static int hf_vrt_trailer_ind_caltime = -1; /* calibrated time indicator */
+static int hf_vrt_trailer_ind_valid = -1; /* valid data ind */
+static int hf_vrt_trailer_ind_reflock = -1; /* reference locked ind */
+static int hf_vrt_trailer_ind_agc = -1; /* AGC/MGC enabled ind */
+static int hf_vrt_trailer_ind_sig = -1; /* signal detected ind */
+static int hf_vrt_trailer_ind_inv = -1; /* spectral inversion ind */
+static int hf_vrt_trailer_ind_overrng = -1; /* overrange indicator */
+static int hf_vrt_trailer_ind_sampleloss = -1; /* sample loss indicator */
+static int hf_vrt_trailer_ind_user0 = -1; /* User indicator 0 */
+static int hf_vrt_trailer_ind_user1 = -1; /* User indicator 1 */
+static int hf_vrt_trailer_ind_user2 = -1; /* User indicator 2 */
+static int hf_vrt_trailer_ind_user3 = -1; /* User indicator 3 */
+
+/* fixed sizes (in bytes) of context packet CIF field bits */
+static int context_size_cif0[32] = { 0, 4, 4, 4, 4, 4, 4, 4, 8, 8, 4, 52, 52, 44, 44, 8,
+ 4, 8, 4, 4, 8, 8, 4, 4, 4, 8, 8, 8, 8, 8, 4, 0 };
+static int context_size_cif1[32] = { 0, 8, 4, 4, 4, 8, 4, 0, 0, 0, 52, 0, 0, 8, 4, 8,
+ 4, 4, 4, 4, 4, 0, 0, 0, 4, 4, 4, 4, 0, 4, 4, 4 };
+
+/* subtree state variables */
+static gint ett_vrt = -1;
+static gint ett_header = -1;
+static gint ett_trailer = -1;
+static gint ett_indicators = -1;
+static gint ett_ind_enables = -1;
+static gint ett_cid = -1;
+static gint ett_cif0 = -1;
+static gint ett_cif1 = -1;
+static gint ett_gain = -1;
+static gint ett_device_id = -1;
+static gint ett_state_event = -1;
+static gint ett_signal_data_format = -1;
+static gint ett_gps = -1;
+static gint ett_ins = -1;
+static gint ett_ecef_ephem = -1;
+static gint ett_rel_ephem = -1;
+static gint ett_gps_ascii = -1;
+static gint ett_assoc_lists = -1;
+static gint ett_pol = -1;
+static gint ett_ver = -1;
+
+/* constants (unit conversion) */
+static const double FEMTOSEC_PER_SEC = 1e-15;
+static const double RADIX_CELSIUS = 1.0/64.0;
+static const double RADIX_DECIBEL = 1.0/128.0;
+static const double RADIX_DECIBEL_MILLIWATT = 1.0/128.0;
+static const double RADIX_DEGREES = 1.0/4194304.0;
+static const double RADIX_HERTZ = 1.0/1048576.0;
+static const double RADIX_METER = 1.0/32.0;
+static const double RADIX_METER_UNSIGNED = 1.0/64.0;
+static const double RADIX_METERS_PER_SECOND = 1.0/65536.0;
+static const double RADIX_RADIAN_PHASE = 1.0/128.0;
+static const double RADIX_RADIAN_POL = 1.0/8192.0;
+
+/* constants (tree index) */
+static const int ETT_IDX_GAIN = 8;
+static const int ETT_IDX_DEVICE_ID = 9;
+static const int ETT_IDX_STATE_EVENT = 10;
+static const int ETT_IDX_SIGNAL_DATA_FORMAT = 11;
+static const int ETT_IDX_GPS = 12;
+static const int ETT_IDX_INS = 13;
+static const int ETT_IDX_ECEF_EPHEM = 14;
+static const int ETT_IDX_REL_EPHEM = 15;
+static const int ETT_IDX_GPS_ASCII = 16;
+static const int ETT_IDX_ASSOC_LISTS = 17;
+static const int ETT_IDX_POL = 18;
+static const int ETT_IDX_VER = 19;
+
+static const value_string packet_types[] = {
+ {0x00, "IF data packet without stream ID"},
+ {0x01, "IF data packet with stream ID"},
+ {0x02, "Extension data packet without stream ID"},
+ {0x03, "Extension data packet with stream ID"},
+ {0x04, "IF context packet"},
+ {0x05, "Extension context packet"},
+ {0, NULL}
+};
+
+static const value_string tsi_types[] = {
+ {0x00, "No integer-seconds timestamp field included"},
+ {0x01, "Coordinated Universal Time (UTC)"},
+ {0x02, "GPS time"},
+ {0x03, "Other"},
+ {0, NULL}
+};
+
+static const value_string tsf_types[] = {
+ {0x00, "No fractional-seconds timestamp field included"},
+ {0x01, "Sample count timestamp"},
+ {0x02, "Real time (picoseconds) timestamp"},
+ {0x03, "Free running count timestamp"},
+ {0, NULL}
+};
+
+static const value_string tsm_types[] = {
+ {0x00, "Precise timestamp resolution"},
+ {0x01, "General timestamp resolution"},
+ {0, NULL}
+};
+
+static const value_string packing_method[] = {
+ {0x00, "Processing efficient"},
+ {0x01, "Link efficient"},
+ {0, NULL}
+};
+
+static const value_string data_sample_type[] = {
+ {0x00, "Real"},
+ {0x01, "Complex, Cartesian"},
+ {0x02, "Complex, polar"},
+ {0, NULL}
+};
+
+static const value_string data_item_format[] = {
+ {0x00, "Signed fixed-point"},
+ {0x01, "Signed VRT, 1-bit exponent"},
+ {0x02, "Signed VRT, 2-bit exponent"},
+ {0x03, "Signed VRT, 3-bit exponent"},
+ {0x04, "Signed VRT, 4-bit exponent"},
+ {0x05, "Signed VRT, 5-bit exponent"},
+ {0x06, "Signed VRT, 6-bit exponent"},
+ {0x07, "Signed fixed-point non-normalized"},
+ {0x0D, "IEEE-754 half-precision floating-point"},
+ {0x0E, "IEEE-754 single-precision floating-point"},
+ {0x0F, "IEEE-754 double-precision floating-point"},
+ {0x10, "Unsigned fixed-point"},
+ {0x11, "Unsigned VRT, 1-bit exponent"},
+ {0x12, "Unsigned VRT, 2-bit exponent"},
+ {0x13, "Unsigned VRT, 3-bit exponent"},
+ {0x14, "Unsigned VRT, 4-bit exponent"},
+ {0x15, "Unsigned VRT, 5-bit exponent"},
+ {0x16, "Unsigned VRT, 6-bit exponent"},
+ {0x17, "Unsigned fixed-point non-normalized"},
+ {0, NULL}
+};
+
+static const value_string standard_version_codes[] = {
+ {0x01, "Implements V49.0"},
+ {0x02, "Implements V49.1"},
+ {0x03, "Implements V49A"},
+ {0x04, "Implements V49.2"},
+ {0, NULL}
+};
+
+static int * const enable_hfs[] = {
+ &hf_vrt_trailer_en_user3,
+ &hf_vrt_trailer_en_user2,
+ &hf_vrt_trailer_en_user1,
+ &hf_vrt_trailer_en_user0,
+ &hf_vrt_trailer_en_sampleloss,
+ &hf_vrt_trailer_en_overrng,
+ &hf_vrt_trailer_en_inv,
+ &hf_vrt_trailer_en_sig,
+ &hf_vrt_trailer_en_agc,
+ &hf_vrt_trailer_en_reflock,
+ &hf_vrt_trailer_en_valid,
+ &hf_vrt_trailer_en_caltime
+};
+
+static int * const ind_hfs[] = {
+ &hf_vrt_trailer_ind_user3,
+ &hf_vrt_trailer_ind_user2,
+ &hf_vrt_trailer_ind_user1,
+ &hf_vrt_trailer_ind_user0,
+ &hf_vrt_trailer_ind_sampleloss,
+ &hf_vrt_trailer_ind_overrng,
+ &hf_vrt_trailer_ind_inv,
+ &hf_vrt_trailer_ind_sig,
+ &hf_vrt_trailer_ind_agc,
+ &hf_vrt_trailer_ind_reflock,
+ &hf_vrt_trailer_ind_valid,
+ &hf_vrt_trailer_ind_caltime
+};
+
+static void dissect_header(tvbuff_t *tvb, proto_tree *tree, int type, int offset);
+static void dissect_trailer(tvbuff_t *tvb, proto_tree *tree, int offset);
+static void dissect_cid(tvbuff_t *tvb, proto_tree *tree, int offset);
+static int dissect_context(tvbuff_t *tvb, proto_tree *tree, int offset);
+static int dissect_context_as_cif(tvbuff_t *tvb, proto_tree *tree, int offset, uint32_t cif, complex_dissector_t
+ *complex_fptr, int **item_ptr, const int *size_ptr, int stop);
+static int dissect_context_array_of_records(proto_tree *tree _U_, tvbuff_t *tvb, int offset);
+static int dissect_context_assoc_lists(proto_tree *tree, tvbuff_t *tvb, int offset);
+static int dissect_context_cif0(proto_tree *tree, tvbuff_t *tvb, int offset);
+static int dissect_context_cif1(proto_tree *tree, tvbuff_t *tvb, int offset);
+static int dissect_context_device_id(proto_tree *tree, tvbuff_t *tvb, int offset);
+static int dissect_context_ecef_ephemeris(proto_tree *tree, tvbuff_t *tvb, int offset);
+static void dissect_context_ephemeris(const ephemeris_fields *s, proto_tree *tree, tvbuff_t *tvb, int offset);
+static int dissect_context_gain(proto_tree *tree, tvbuff_t *tvb, int offset);
+static int dissect_context_gps(proto_tree *tree, tvbuff_t *tvb, int offset);
+static int dissect_context_gps_ascii(proto_tree *tree, tvbuff_t *tvb, int offset);
+static int dissect_context_ins(proto_tree *tree, tvbuff_t *tvb, int offset);
+static int dissect_context_phase_offset(proto_tree *tree, tvbuff_t *tvb, int offset);
+static int dissect_context_polarization(proto_tree *tree, tvbuff_t *tvb, int offset);
+static int dissect_context_ref_level(proto_tree *tree, tvbuff_t *tvb, int offset);
+static int dissect_context_rel_ephemeris(proto_tree *tree, tvbuff_t *tvb, int offset);
+static int dissect_context_signal_data_format(proto_tree *tree, tvbuff_t *tvb, int offset);
+static int dissect_context_state_event(proto_tree *tree, tvbuff_t *tvb, int offset);
+static int dissect_context_temperature(proto_tree *tree, tvbuff_t *tvb, int offset);
+static int dissect_context_ver(proto_tree *tree, tvbuff_t *tvb, int offset);
+static const char* get_engr_prefix(double *val);
+
+/* context simple field dissector function pointer array (mutually exclusive with complex below) */
+static int* hf_vrt_context_cif0[32] = { NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
+ NULL, &hf_vrt_context_ephemeris_ref_id, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
+ &hf_vrt_context_timestamp_cal, &hf_vrt_context_timestamp_adjust, &hf_vrt_context_sample_rate,
+ &hf_vrt_context_over_range_count, NULL, NULL, &hf_vrt_context_if_band_offset,
+ &hf_vrt_context_rf_freq_offset, &hf_vrt_context_rf_freq, &hf_vrt_context_if_freq,
+ &hf_vrt_context_bandwidth, &hf_vrt_context_ref_pt_id, NULL };
+
+static int* hf_vrt_context_cif1[32] = { NULL, NULL, NULL, &hf_vrt_context_v49_spec, NULL,
+ &hf_vrt_context_io64, &hf_vrt_context_io32, NULL, NULL, NULL, NULL, NULL, NULL,
+ &hf_vrt_context_aux_bandwidth, NULL, &hf_vrt_context_aux_freq, NULL, NULL, NULL, NULL, NULL,
+ NULL, NULL, NULL, &hf_vrt_context_range, NULL, NULL, NULL, NULL, NULL, NULL, NULL };
+
+
+/* context complex field dissector function pointer array */
+static complex_dissector_t complex_dissector_cif0[32] = {
+ NULL, dissect_context_cif1, NULL, NULL, NULL, NULL, NULL, NULL, dissect_context_assoc_lists,
+ dissect_context_gps_ascii, NULL, dissect_context_rel_ephemeris, dissect_context_ecef_ephemeris,
+ dissect_context_ins, dissect_context_gps, dissect_context_signal_data_format,
+ dissect_context_state_event, dissect_context_device_id, dissect_context_temperature, NULL,
+ NULL, NULL, NULL, dissect_context_gain, dissect_context_ref_level, NULL, NULL, NULL, NULL,
+ NULL, NULL, NULL };
+
+/* partial CIF1 support */
+static complex_dissector_t complex_dissector_cif1[32] = {
+ NULL, NULL, dissect_context_ver, NULL, NULL, NULL, NULL, dissect_context_array_of_records,
+ NULL, dissect_context_array_of_records, NULL, dissect_context_array_of_records, NULL, NULL,
+ NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
+ dissect_context_array_of_records, NULL,
+ dissect_context_polarization, dissect_context_phase_offset };
+
+
+static int dissect_vrt(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, void* data _U_)
+{
+ int offset = 0;
+ guint8 type;
+
+ col_set_str(pinfo->cinfo, COL_PROTOCOL, "VITA 49");
+ col_clear(pinfo->cinfo,COL_INFO);
+
+ /* HACK to support UHD's weird header offset on data packets. */
+ if (vrt_use_ettus_uhd_header_format && tvb_get_guint8(tvb, 0) == 0)
+ offset += 4;
+
+ /* get packet type */
+ type = tvb_get_guint8(tvb, offset) >> 4;
+ col_add_str(pinfo->cinfo, COL_INFO, val_to_str(type, packet_types, "Reserved packet type (0x%02x)"));
+
+ if (tree) { /* we're being asked for details */
+ guint8 sidflag;
+ guint8 cidflag;
+ guint8 tflag;
+ guint8 tsitype;
+ guint8 tsftype;
+ guint16 len;
+ guint16 nsamps;
+
+ proto_tree *vrt_tree;
+ proto_item *ti;
+
+ /* get SID, CID, T flags and TSI, TSF types */
+ sidflag = (((type & 0x01) != 0) || (type == 4)) ? 1 : 0;
+ cidflag = (tvb_get_guint8(tvb, offset) >> 3) & 0x01;
+ /* tflag is in data packets but not context packets */
+ tflag = (tvb_get_guint8(tvb, offset) >> 2) & 0x01;
+ if (type == 4)
+ tflag = 0; /* this should be unnecessary but we do it just in case */
+ /* tsmflag is in context packets but not data packets
+ tsmflag = (tvb_get_guint8(tvb, offset) >> 0) & 0x01; */
+ tsitype = (tvb_get_guint8(tvb, offset+1) >> 6) & 0x03;
+ tsftype = (tvb_get_guint8(tvb, offset+1) >> 4) & 0x03;
+ len = tvb_get_ntohs(tvb, offset+2);
+
+ nsamps = len - 1; /* (Before adjusting word count for optional fields) */
+
+ ti = proto_tree_add_item(tree, proto_vrt, tvb, offset, -1, ENC_NA);
+ vrt_tree = proto_item_add_subtree(ti, ett_vrt);
+
+ dissect_header(tvb, vrt_tree, type, offset);
+ offset += 4;
+
+ /* header's done! if SID (last bit of type), put the stream ID here */
+ if (sidflag) {
+ proto_tree_add_item(vrt_tree, hf_vrt_sid, tvb, offset, 4, ENC_BIG_ENDIAN);
+ nsamps -= 1;
+ offset += 4;
+
+ }
+
+ /* if there's a class ID (cidflag), put the class ID here */
+ if (cidflag) {
+ dissect_cid(tvb, vrt_tree, offset);
+ nsamps -= 2;
+ offset += 8;
+ }
+
+ /* if TSI and/or TSF, populate those here */
+ if (tsitype != 0) {
+ proto_tree_add_item(vrt_tree, hf_vrt_ts_int, tvb, offset, 4, ENC_BIG_ENDIAN);
+ nsamps -= 1;
+ offset += 4;
+ }
+ if (tsftype != 0) {
+ if (tsftype == 1 || tsftype == 3) {
+ proto_tree_add_item(vrt_tree, hf_vrt_ts_frac_sample, tvb, offset, 8, ENC_BIG_ENDIAN);
+ } else if (tsftype == 2) {
+ proto_tree_add_item(vrt_tree, hf_vrt_ts_frac_picosecond, tvb, offset, 8, ENC_BIG_ENDIAN);
+ }
+ nsamps -= 2;
+ offset += 8;
+ }
+
+ if (tflag) {
+ nsamps -= 1;
+ }
+
+ /* now we've got either a context packet or a data packet */
+ if (type == 4) {
+ /* parse context packet */
+ int num_v49_words = dissect_context(tvb, vrt_tree, offset);
+ nsamps -= num_v49_words;
+ offset += 4*num_v49_words;
+ }
+
+ /* we're into the data */
+ if (nsamps != 0) {
+ proto_tree_add_item(vrt_tree, hf_vrt_data, tvb, offset, nsamps*4, ENC_NA);
+ }
+
+ offset += nsamps*4;
+
+ if (tflag) {
+ dissect_trailer(tvb, vrt_tree, offset);
+ }
+ }
+ return tvb_captured_length(tvb);
+}
+
+static void dissect_header(tvbuff_t *tvb, proto_tree *tree, int type, int offset)
+{
+ proto_item *hdr_item;
+ proto_tree *hdr_tree;
+
+ hdr_item = proto_tree_add_item(tree, hf_vrt_header, tvb, offset, 4, ENC_BIG_ENDIAN);
+
+ hdr_tree = proto_item_add_subtree(hdr_item, ett_header);
+ proto_tree_add_item(hdr_tree, hf_vrt_type, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(hdr_tree, hf_vrt_cidflag, tvb, offset, 1, ENC_BIG_ENDIAN);
+ if (type == 4) {
+ proto_tree_add_item(hdr_tree, hf_vrt_tsmflag, tvb, offset, 1, ENC_BIG_ENDIAN);
+ } else {
+ proto_tree_add_item(hdr_tree, hf_vrt_tflag, tvb, offset, 1, ENC_BIG_ENDIAN);
+ }
+ offset += 1;
+ proto_tree_add_item(hdr_tree, hf_vrt_tsi, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(hdr_tree, hf_vrt_tsf, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(hdr_tree, hf_vrt_seq, tvb, offset, 1, ENC_BIG_ENDIAN);
+ offset += 1;
+ proto_tree_add_item(hdr_tree, hf_vrt_len, tvb, offset, 2, ENC_BIG_ENDIAN);
+}
+
+static void dissect_trailer(tvbuff_t *tvb, proto_tree *tree, int offset)
+{
+ proto_item *enable_item, *ind_item, *trailer_item;
+ proto_tree *enable_tree;
+ proto_tree *ind_tree;
+ proto_tree *trailer_tree;
+ guint16 en_bits;
+ gint16 i;
+
+ trailer_item = proto_tree_add_item(tree, hf_vrt_trailer, tvb, offset, 4, ENC_BIG_ENDIAN);
+ trailer_tree = proto_item_add_subtree(trailer_item, ett_trailer);
+
+ /* grab the indicator enables and the indicators;
+ only display enables, indicators which are enabled */
+ enable_item = proto_tree_add_item(trailer_tree, hf_vrt_trailer_enables, tvb, offset, 2, ENC_BIG_ENDIAN);
+ ind_item = proto_tree_add_item(trailer_tree, hf_vrt_trailer_ind, tvb, offset + 1, 2, ENC_BIG_ENDIAN);
+ /* grab enable bits */
+ en_bits = (tvb_get_ntohs(tvb, offset) & 0xFFF0) >> 4;
+
+ /* if there's any enables, start trees for enable bits and for indicators
+ only enables and indicators which are enabled get printed. */
+ if (en_bits) {
+ enable_tree = proto_item_add_subtree(enable_item, ett_ind_enables);
+ ind_tree = proto_item_add_subtree(ind_item, ett_indicators);
+ for (i = 11; i >= 0; i--) {
+ if (en_bits & (1<<i)) {
+ /* XXX: Display needs to be improved ... */
+ proto_tree_add_item(enable_tree, *enable_hfs[i], tvb, offset, 2, ENC_BIG_ENDIAN);
+ proto_tree_add_item(ind_tree, *ind_hfs[i], tvb, offset+1, 2, ENC_BIG_ENDIAN);
+ }
+ }
+ }
+ offset += 3;
+ proto_tree_add_item(trailer_tree, hf_vrt_trailer_e, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(trailer_tree, hf_vrt_trailer_acpc, tvb, offset, 1, ENC_BIG_ENDIAN);
+}
+
+static void dissect_cid(tvbuff_t *tvb, proto_tree *tree, int offset)
+{
+ proto_item *cid_item;
+ proto_tree *cid_tree;
+
+ cid_item = proto_tree_add_item(tree, hf_vrt_cid, tvb, offset, 8, ENC_BIG_ENDIAN);
+ cid_tree = proto_item_add_subtree(cid_item, ett_cid);
+
+ offset += 1;
+ proto_tree_add_item(cid_tree, hf_vrt_cid_oui, tvb, offset, 3, ENC_BIG_ENDIAN);
+ offset += 3;
+ proto_tree_add_item(cid_tree, hf_vrt_cid_icc, tvb, offset, 2, ENC_BIG_ENDIAN);
+ offset += 2;
+ proto_tree_add_item(cid_tree, hf_vrt_cid_pcc, tvb, offset, 2, ENC_BIG_ENDIAN);
+}
+
+static int dissect_context(tvbuff_t *tvb, proto_tree *tree, int offset)
+{
+ uint32_t cif[8] = {0, 0, 0, 0, 0, 0, 0, 0};
+ int offset_start = offset;
+
+ cif[0] = tvb_get_ntohl(tvb, offset);
+ dissect_context_cif0(tree, tvb, offset);
+ offset += 4;
+ // CIF1-CIF7 bit fields come next with CIF1 first
+ for (int i = 1; i < 8; i++) {
+ if (cif[0] & (1 << i)) {
+ if (complex_dissector_cif0[i] != NULL) {
+ (*complex_dissector_cif0[i])(tree, tvb, offset);
+ } else {
+ proto_tree_add_item(tree, hf_vrt_cif[i], tvb, offset, 4, ENC_BIG_ENDIAN);
+ }
+ cif[i] = tvb_get_ntohl(tvb, offset);
+ offset += 4;
+ }
+ }
+
+ // decode CIF0 fields
+ offset = dissect_context_as_cif(tvb, tree, offset, cif[0], complex_dissector_cif0, hf_vrt_context_cif0,
+ context_size_cif0, 7);
+ // finally other CIFs (only CIF1 for now)
+ if (cif[0] & (1 << 1)) {
+ offset = dissect_context_as_cif(tvb, tree, offset, cif[1], complex_dissector_cif1, hf_vrt_context_cif1,
+ context_size_cif1, 0);
+ }
+
+ // return how many VITA-49 words were processed
+ return (offset - offset_start)/4;
+}
+
+static int dissect_context_as_cif(tvbuff_t *tvb, proto_tree *tree, int offset, uint32_t cif,
+ complex_dissector_t *complex_fptr, int **item_ptr, const int *size_ptr, int stop) {
+ for (int i = 31; i > stop; i--) {
+ if (cif & (1u << i)) {
+ if (complex_fptr[i] != NULL) {
+ // a complex dissector returns the variable part of field length (in bytes)
+ offset += (*complex_fptr[i])(tree, tvb, offset);
+ } else if (item_ptr[i] != NULL) {
+ proto_tree_add_item(tree, *item_ptr[i], tvb, offset, size_ptr[i], ENC_BIG_ENDIAN);
+ }
+ // add fixed part of field length (in bytes)
+ offset += size_ptr[i];
+ }
+ }
+
+ return offset;
+}
+
+static int dissect_context_array_of_records(proto_tree *tree _U_, tvbuff_t *tvb, int offset) {
+ // This is a placeholder that does not populate a proto tree, but computes & returns the
+ // variable field length so subsequent field indexing is correct.
+ return tvb_get_ntohl(tvb, offset)*4;
+}
+
+static int dissect_context_assoc_lists(proto_tree *tree, tvbuff_t *tvb, int offset) {
+ // compute number of variable words in field
+ guint32 word1 = tvb_get_ntohl(tvb, offset);
+ guint32 src_size = (word1 >> 16) & 0x01FF;
+ guint32 sys_size = word1 & 0x01FF;
+ guint32 word2 = tvb_get_ntohl(tvb, offset + 4);
+ guint32 vec_size = word2 >> 16;
+ gboolean a_bit = (word2 & 0x8000) != 0;
+ guint32 asy_size = word2 & 0x7FFF;
+ guint32 num_words = src_size + sys_size + vec_size + asy_size + a_bit*asy_size;
+
+ proto_tree *assoc_tree = proto_tree_add_subtree(tree, tvb, offset, 8 + num_words*4, ETT_IDX_ASSOC_LISTS, NULL,
+ "Context association lists");
+ proto_tree_add_item(assoc_tree, hf_vrt_context_assoc_lists_src_size, tvb, offset, 2, ENC_BIG_ENDIAN);
+ proto_tree_add_item(assoc_tree, hf_vrt_context_assoc_lists_sys_size, tvb, offset + 2, 2, ENC_BIG_ENDIAN);
+ proto_tree_add_item(assoc_tree, hf_vrt_context_assoc_lists_vec_size, tvb, offset + 4, 2, ENC_BIG_ENDIAN);
+ proto_tree_add_item(assoc_tree, hf_vrt_context_assoc_lists_a, tvb, offset + 6, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(assoc_tree, hf_vrt_context_assoc_lists_asy_size, tvb, offset + 6, 2, ENC_BIG_ENDIAN);
+ offset += 8;
+
+ if (src_size > 0) {
+ proto_tree_add_item(assoc_tree, hf_vrt_context_assoc_lists_src_data, tvb, offset, src_size*4, ENC_NA);
+ offset += src_size*4;
+ }
+
+ if (sys_size > 0) {
+ proto_tree_add_item(assoc_tree, hf_vrt_context_assoc_lists_sys_data, tvb, offset, sys_size*4, ENC_NA);
+ offset += sys_size*4;
+ }
+
+ if (vec_size > 0) {
+ proto_tree_add_item(assoc_tree, hf_vrt_context_assoc_lists_vec_data, tvb, offset, vec_size*4, ENC_NA);
+ offset += vec_size*4;
+ }
+
+ if (asy_size > 0) {
+ proto_tree_add_item(assoc_tree, hf_vrt_context_assoc_lists_asy_data, tvb, offset, asy_size*4, ENC_NA);
+ offset += asy_size*4;
+ if (a_bit) {
+ proto_tree_add_item(assoc_tree, hf_vrt_context_assoc_lists_asy_tag_data, tvb, offset, asy_size*4, ENC_NA);
+ }
+ }
+
+ return num_words*4;
+}
+
+static int dissect_context_cif0(proto_tree *tree, tvbuff_t *tvb, int offset) {
+ proto_item *cif0_item;
+ proto_tree *cif0_tree;
+
+ cif0_item = proto_tree_add_item(tree, hf_vrt_cif[0], tvb, offset, 4, ENC_BIG_ENDIAN);
+ cif0_tree = proto_item_add_subtree(cif0_item, ett_cif0);
+ proto_tree_add_item(cif0_tree, hf_vrt_cif0_change_flag, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif0_tree, hf_vrt_cif0_ref_pt_id, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif0_tree, hf_vrt_cif0_bandwidth, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif0_tree, hf_vrt_cif0_if_freq, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif0_tree, hf_vrt_cif0_rf_freq, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif0_tree, hf_vrt_cif0_rf_freq_offset, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif0_tree, hf_vrt_cif0_if_band_offset, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif0_tree, hf_vrt_cif0_ref_level, tvb, offset, 1, ENC_BIG_ENDIAN);
+ offset += 1;
+ proto_tree_add_item(cif0_tree, hf_vrt_cif0_gain, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif0_tree, hf_vrt_cif0_over_range_count, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif0_tree, hf_vrt_cif0_sample_rate, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif0_tree, hf_vrt_cif0_timestamp_adjust, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif0_tree, hf_vrt_cif0_timestamp_cal, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif0_tree, hf_vrt_cif0_temperature, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif0_tree, hf_vrt_cif0_device_id, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif0_tree, hf_vrt_cif0_state_event, tvb, offset, 1, ENC_BIG_ENDIAN);
+ offset += 1;
+ proto_tree_add_item(cif0_tree, hf_vrt_cif0_signal_data_format, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif0_tree, hf_vrt_cif0_gps, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif0_tree, hf_vrt_cif0_ins, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif0_tree, hf_vrt_cif0_ecef_ephemeris, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif0_tree, hf_vrt_cif0_rel_ephemeris, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif0_tree, hf_vrt_cif0_ephemeris_ref_id, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif0_tree, hf_vrt_cif0_gps_ascii, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif0_tree, hf_vrt_cif0_context_assoc_lists, tvb, offset, 1, ENC_BIG_ENDIAN);
+ offset += 1;
+ proto_tree_add_item(cif0_tree, hf_vrt_cif0_cif7, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif0_tree, hf_vrt_cif0_cif6, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif0_tree, hf_vrt_cif0_cif5, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif0_tree, hf_vrt_cif0_cif4, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif0_tree, hf_vrt_cif0_cif3, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif0_tree, hf_vrt_cif0_cif2, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif0_tree, hf_vrt_cif0_cif1, tvb, offset, 1, ENC_BIG_ENDIAN);
+ return 0;
+}
+
+static int dissect_context_cif1(proto_tree *tree, tvbuff_t *tvb, int offset) {
+ proto_item *cif1_item = proto_tree_add_item(tree, hf_vrt_cif[1], tvb, offset, 4, ENC_BIG_ENDIAN);
+ proto_tree *cif1_tree = proto_item_add_subtree(cif1_item, ett_cif1);
+ proto_tree_add_item(cif1_tree, hf_vrt_cif1_phase_offset, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif1_tree, hf_vrt_cif1_polarization, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif1_tree, hf_vrt_cif1_range, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif1_tree, hf_vrt_cif1_aux_freq, tvb, offset + 2, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif1_tree, hf_vrt_cif1_aux_bandwidth, tvb, offset + 2, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif1_tree, hf_vrt_cif1_io32, tvb, offset + 3, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif1_tree, hf_vrt_cif1_io64, tvb, offset + 3, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif1_tree, hf_vrt_cif1_v49_spec, tvb, offset + 3, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(cif1_tree, hf_vrt_cif1_ver, tvb, offset + 3, 1, ENC_BIG_ENDIAN);
+ return 0;
+}
+
+static int dissect_context_device_id(proto_tree *tree, tvbuff_t *tvb, int offset) {
+ proto_tree *id_tree = proto_tree_add_subtree(tree, tvb, offset, 8, ETT_IDX_DEVICE_ID, NULL, "Device identifier");
+ proto_tree_add_item(id_tree, hf_vrt_context_device_id_oui, tvb, offset + 1, 3, ENC_BIG_ENDIAN);
+ proto_tree_add_item(id_tree, hf_vrt_context_device_id_code, tvb, offset + 6, 2, ENC_BIG_ENDIAN);
+ return 0;
+}
+
+static int dissect_context_ecef_ephemeris(proto_tree *tree, tvbuff_t *tvb, int offset) {
+ proto_tree *ecef_tree = proto_tree_add_subtree(tree, tvb, offset, 52, ETT_IDX_ECEF_EPHEM, NULL, "ECEF ephemeris");
+ dissect_context_ephemeris(&hf_vrt_context_ecef_ephemeris, ecef_tree, tvb, offset);
+ return 0;
+}
+
+static void dissect_context_ephemeris(const ephemeris_fields *s, proto_tree *tree, tvbuff_t *tvb, int offset) {
+ proto_tree_add_item(tree, s->tsi, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(tree, s->tsf, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(tree, s->oui, tvb, offset + 1, 3, ENC_BIG_ENDIAN);
+ proto_tree_add_item(tree, s->ts_int, tvb, offset + 4, 4, ENC_BIG_ENDIAN);
+
+ guint8 tsftype = tvb_get_guint8(tvb, offset) & 0x03;
+ if (tsftype == 1 || tsftype == 3) {
+ proto_tree_add_item(tree, s->ts_frac_sample, tvb, offset + 8, 8, ENC_BIG_ENDIAN);
+ } else if (tsftype == 2) {
+ proto_tree_add_item(tree, s->ts_picosecond, tvb, offset + 8, 8, ENC_BIG_ENDIAN);
+ }
+
+ proto_tree_add_item(tree, s->pos_x, tvb, offset + 16, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_item(tree, s->pos_y, tvb, offset + 20, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_item(tree, s->pos_z, tvb, offset + 24, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_item(tree, s->att_alpha, tvb, offset + 28, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_item(tree, s->att_beta, tvb, offset + 32, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_item(tree, s->att_phi, tvb, offset + 36, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_item(tree, s->vel_dx, tvb, offset + 40, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_item(tree, s->vel_dy, tvb, offset + 44, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_item(tree, s->vel_dz, tvb, offset + 48, 4, ENC_BIG_ENDIAN);
+}
+
+static void dissect_context_formatted_gps_ins(const formatted_gps_ins_fields *s, proto_tree *tree, tvbuff_t *tvb,
+ int offset) {
+ proto_tree_add_item(tree, s->tsi, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(tree, s->tsf, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(tree, s->oui, tvb, offset + 1, 3, ENC_BIG_ENDIAN);
+ proto_tree_add_item(tree, s->ts_int, tvb, offset + 4, 4, ENC_BIG_ENDIAN);
+
+ guint8 tsftype = tvb_get_guint8(tvb, offset) & 0x03;
+ if (tsftype == 1 || tsftype == 3) {
+ proto_tree_add_item(tree, s->ts_frac_sample, tvb, offset + 8, 8, ENC_BIG_ENDIAN);
+ } else if (tsftype == 2) {
+ proto_tree_add_item(tree, s->ts_picosecond, tvb, offset + 8, 8, ENC_BIG_ENDIAN);
+ }
+
+ proto_tree_add_item(tree, s->lat, tvb, offset + 16, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_item(tree, s->lon, tvb, offset + 20, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_item(tree, s->alt, tvb, offset + 24, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_item(tree, s->speed, tvb, offset + 28, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_item(tree, s->heading, tvb, offset + 32, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_item(tree, s->track, tvb, offset + 36, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_item(tree, s->mag_var, tvb, offset + 40, 4, ENC_BIG_ENDIAN);
+}
+
+static int dissect_context_gain(proto_tree *tree, tvbuff_t *tvb, int offset) {
+ proto_tree *gain_tree = proto_tree_add_subtree(tree, tvb, offset, 4, ETT_IDX_GAIN, NULL, "Gain");
+ proto_tree_add_item(gain_tree, hf_vrt_context_gain_stage2, tvb, offset, 2, ENC_BIG_ENDIAN);
+ proto_tree_add_item(gain_tree, hf_vrt_context_gain_stage1, tvb, offset + 2, 2, ENC_BIG_ENDIAN);
+ return 0;
+}
+
+static int dissect_context_gps(proto_tree *tree, tvbuff_t *tvb, int offset) {
+ proto_tree *gps_tree = proto_tree_add_subtree(tree, tvb, offset, 44, ETT_IDX_GPS, NULL, "Formatted GPS");
+ dissect_context_formatted_gps_ins(&hf_vrt_context_gps, gps_tree, tvb, offset);
+ return 0;
+}
+
+static int dissect_context_gps_ascii(proto_tree *tree, tvbuff_t *tvb, int offset) {
+ guint32 nword = tvb_get_ntohl(tvb, offset + 4);
+ proto_tree *gps_tree = proto_tree_add_subtree(tree, tvb, offset, 8 + nword*4, ETT_IDX_GPS_ASCII, NULL, "GPS ASCII");
+ proto_tree_add_item(gps_tree, hf_vrt_context_gps_ascii_oui, tvb, offset + 1, 3, ENC_BIG_ENDIAN);
+ proto_tree_add_item(gps_tree, hf_vrt_context_gps_ascii_size, tvb, offset + 4, 4, ENC_BIG_ENDIAN);
+
+ if (nword > 0) {
+ proto_tree_add_item(gps_tree, hf_vrt_context_gps_ascii_data, tvb, offset + 8, nword*4, ENC_NA);
+ }
+
+ return nword*4;
+}
+
+static int dissect_context_ins(proto_tree *tree, tvbuff_t *tvb, int offset) {
+ proto_tree *ins_tree = proto_tree_add_subtree(tree, tvb, offset, 44, ETT_IDX_INS, NULL, "Formatted INS");
+ dissect_context_formatted_gps_ins(&hf_vrt_context_ins, ins_tree, tvb, offset);
+ return 0;
+}
+
+static int dissect_context_phase_offset(proto_tree *tree, tvbuff_t *tvb, int offset) {
+ proto_tree_add_item(tree, hf_vrt_context_phase_offset, tvb, offset + 2, 2, ENC_BIG_ENDIAN);
+ return 0;
+}
+
+static int dissect_context_polarization(proto_tree *tree, tvbuff_t *tvb, int offset) {
+ proto_tree *pol_tree = proto_tree_add_subtree(tree, tvb, offset, 4, ETT_IDX_POL, NULL, "Polarization");
+ proto_tree_add_item(pol_tree, hf_vrt_context_pol_tilt, tvb, offset, 2, ENC_BIG_ENDIAN);
+ proto_tree_add_item(pol_tree, hf_vrt_context_pol_ellipticity, tvb, offset + 2, 2, ENC_BIG_ENDIAN);
+ return 0;
+}
+
+static int dissect_context_ref_level(proto_tree *tree, tvbuff_t *tvb, int offset) {
+ proto_tree_add_item(tree, hf_vrt_context_ref_level, tvb, offset + 2, 2, ENC_BIG_ENDIAN);
+ return 0;
+}
+
+static int dissect_context_rel_ephemeris(proto_tree *tree, tvbuff_t *tvb, int offset) {
+ proto_tree *rel_tree = proto_tree_add_subtree(tree, tvb, offset, 52, ETT_IDX_REL_EPHEM, NULL, "Relative ephemeris");
+ dissect_context_ephemeris(&hf_vrt_context_rel_ephemeris, rel_tree, tvb, offset);
+ return 0;
+}
+
+static int dissect_context_signal_data_format(proto_tree *tree, tvbuff_t *tvb, int offset) {
+ proto_tree *format_tree = proto_tree_add_subtree(tree, tvb, offset, 8, ETT_IDX_SIGNAL_DATA_FORMAT, NULL,
+ "Signal data packet payload format");
+ proto_tree_add_item(format_tree, hf_vrt_context_signal_data_format_packing, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(format_tree, hf_vrt_context_signal_data_format_type, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(format_tree, hf_vrt_context_signal_data_format_item, tvb, offset, 1, ENC_BIG_ENDIAN);
+ offset += 1;
+ proto_tree_add_item(format_tree, hf_vrt_context_signal_data_format_repeat, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(format_tree, hf_vrt_context_signal_data_format_event_size, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(format_tree, hf_vrt_context_signal_data_format_channel_size, tvb, offset, 1, ENC_BIG_ENDIAN);
+ offset += 1;
+ proto_tree_add_item(format_tree, hf_vrt_context_signal_data_format_fraction_size, tvb, offset, 2, ENC_BIG_ENDIAN);
+ proto_tree_add_item(format_tree, hf_vrt_context_signal_data_format_packing_size, tvb, offset, 2, ENC_BIG_ENDIAN);
+ proto_tree_add_item(format_tree, hf_vrt_context_signal_data_format_item_size, tvb, offset, 2, ENC_BIG_ENDIAN);
+ offset += 2;
+ proto_tree_add_item(format_tree, hf_vrt_context_signal_data_format_repeat_count, tvb, offset, 2, ENC_BIG_ENDIAN);
+ offset += 2;
+ proto_tree_add_item(format_tree, hf_vrt_context_signal_data_format_vector_size, tvb, offset, 2, ENC_BIG_ENDIAN);
+
+ return 0;
+}
+
+static int dissect_context_state_event(proto_tree *tree, tvbuff_t *tvb, int offset) {
+ proto_tree *state_event_tree = proto_tree_add_subtree(tree, tvb, offset, 4, ETT_IDX_STATE_EVENT, NULL,
+ "State and event indicators");
+ proto_tree_add_item(state_event_tree, hf_vrt_context_state_event_en_cal_time, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(state_event_tree, hf_vrt_context_state_event_en_valid_data, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(state_event_tree, hf_vrt_context_state_event_en_ref_lock, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(state_event_tree, hf_vrt_context_state_event_en_agc, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(state_event_tree, hf_vrt_context_state_event_en_detected_sig, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(state_event_tree, hf_vrt_context_state_event_en_spectral_inv, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(state_event_tree, hf_vrt_context_state_event_en_over_range, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(state_event_tree, hf_vrt_context_state_event_en_sample_loss, tvb, offset, 1, ENC_BIG_ENDIAN);
+
+ proto_tree_add_item(state_event_tree, hf_vrt_context_state_event_cal_time, tvb, offset + 1, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(state_event_tree, hf_vrt_context_state_event_valid_data, tvb, offset + 1, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(state_event_tree, hf_vrt_context_state_event_ref_lock, tvb, offset + 1, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(state_event_tree, hf_vrt_context_state_event_agc, tvb, offset + 1, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(state_event_tree, hf_vrt_context_state_event_detected_sig, tvb, offset + 2, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(state_event_tree, hf_vrt_context_state_event_spectral_inv, tvb, offset + 2, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(state_event_tree, hf_vrt_context_state_event_over_range, tvb, offset + 2, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(state_event_tree, hf_vrt_context_state_event_sample_loss, tvb, offset + 2, 1, ENC_BIG_ENDIAN);
+
+ proto_tree_add_item(state_event_tree, hf_vrt_context_state_event_user, tvb, offset + 3, 1, ENC_BIG_ENDIAN);
+ return 0;
+}
+
+static int dissect_context_temperature(proto_tree *tree, tvbuff_t *tvb, int offset) {
+ proto_tree_add_item(tree, hf_vrt_context_temperature, tvb, offset + 2, 2, ENC_BIG_ENDIAN);
+ return 0;
+}
+
+static int dissect_context_ver(proto_tree *tree, tvbuff_t *tvb, int offset) {
+ proto_tree *ver_tree = proto_tree_add_subtree(tree, tvb, offset, 4, ETT_IDX_VER, NULL,
+ "Version and build code");
+ proto_tree_add_item(ver_tree, hf_vrt_context_ver_year, tvb, offset, 2, ENC_BIG_ENDIAN);
+ proto_tree_add_item(ver_tree, hf_vrt_context_ver_day, tvb, offset, 2, ENC_BIG_ENDIAN);
+ proto_tree_add_item(ver_tree, hf_vrt_context_ver_rev, tvb, offset + 2, 2, ENC_BIG_ENDIAN);
+ proto_tree_add_item(ver_tree, hf_vrt_context_ver_user, tvb, offset + 2, 2, ENC_BIG_ENDIAN);
+ return 0;
+}
+
+static void format_celsius(char *str, int16_t val) {
+ snprintf(str, ITEM_LABEL_LENGTH, "%f °C", (double)val*RADIX_CELSIUS);
+}
+
+static void format_decibel(char *str, int16_t val) {
+ snprintf(str, ITEM_LABEL_LENGTH, "%f dB", (double)val*RADIX_DECIBEL);
+}
+
+static void format_decibel_milliwatt(char *str, int16_t val) {
+ snprintf(str, ITEM_LABEL_LENGTH, "%f dBm", (double)val*RADIX_DECIBEL_MILLIWATT);
+}
+
+static void format_degrees(char *str, int32_t val) {
+ snprintf(str, ITEM_LABEL_LENGTH, "%f degrees", (double)val*RADIX_DEGREES);
+}
+
+static void format_hertz(char *str, int64_t val) {
+ double val_f64 = (double)val*RADIX_HERTZ;
+ const char *prefix = get_engr_prefix(&val_f64);
+ snprintf(str, ITEM_LABEL_LENGTH, "%f %sHz", val_f64, prefix);
+}
+
+static void format_meter(char *str, int32_t val) {
+ double val_f64 = (double)val*RADIX_METER;
+ const char *prefix = get_engr_prefix(&val_f64);
+ snprintf(str, ITEM_LABEL_LENGTH, "%f %sm", val_f64, prefix);
+}
+
+static void format_meter_unsigned(char *str, uint32_t val) {
+ double val_f64 = (double)val*RADIX_METER_UNSIGNED;
+ const char *prefix = get_engr_prefix(&val_f64);
+ snprintf(str, ITEM_LABEL_LENGTH, "%f %sm", val_f64, prefix);
+}
+
+static void format_meters_per_second(char *str, int32_t val) {
+ double val_f64 = (double)val*RADIX_METERS_PER_SECOND;
+ const char *prefix = get_engr_prefix(&val_f64);
+ snprintf(str, ITEM_LABEL_LENGTH, "%f %sm/s", val_f64, prefix);
+}
+
+static void format_radian_phase(char *str, int16_t val) {
+ snprintf(str, ITEM_LABEL_LENGTH, "%f rad", (double)val*RADIX_RADIAN_PHASE);
+}
+
+static void format_radian_pol(char *str, int16_t val) {
+ snprintf(str, ITEM_LABEL_LENGTH, "%f rad", (double)val*RADIX_RADIAN_POL);
+}
+
+static void format_second(char *str, int64_t val) {
+ double val_f64 = (double)val*FEMTOSEC_PER_SEC;
+ const char *prefix = get_engr_prefix(&val_f64);
+ snprintf(str, ITEM_LABEL_LENGTH, "%f %ss", val_f64, prefix);
+}
+
+static const char* get_engr_prefix(double *val) {
+ const char* prefix_str = "";
+ int32_t exp = (int32_t)floor(log10(fabs(*val))/(double)3.0)*3;
+
+ switch (exp) {
+ case -15:
+ prefix_str = "f";
+ *val *= 1e15;
+ break;
+ case -12:
+ prefix_str = "p";
+ *val *= 1e12;
+ break;
+ case -9:
+ prefix_str = "n";
+ *val *= 1e9;
+ break;
+ case -6:
+ prefix_str = "µ";
+ *val *= 1e6;
+ break;
+ case -3:
+ prefix_str = "m";
+ *val *= 1e3;
+ break;
+ case 3:
+ prefix_str = "k";
+ *val *= 1e-3;
+ break;
+ case 6:
+ prefix_str = "M";
+ *val *= 1e-6;
+ break;
+ case 9:
+ prefix_str = "G";
+ *val *= 1e-9;
+ break;
+ case 12:
+ prefix_str = "T";
+ *val *= 1e-12;
+ break;
+ }
+
+ return prefix_str;
+}
+
+void
+proto_register_vrt(void)
+{
+ module_t *vrt_module;
+
+ static hf_register_info hf[] = {
+ { &hf_vrt_header,
+ { "VRT header", "vrt.hdr",
+ FT_UINT32, BASE_HEX,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_type,
+ { "Packet type", "vrt.type",
+ FT_UINT8, BASE_DEC,
+ VALS(packet_types), 0xF0,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cidflag,
+ { "Class ID included", "vrt.cidflag",
+ FT_BOOLEAN, 8,
+ NULL, 0x08,
+ NULL, HFILL }
+ },
+ { &hf_vrt_tflag,
+ { "Trailer included", "vrt.tflag",
+ FT_BOOLEAN, 8,
+ NULL, 0x04,
+ NULL, HFILL }
+ },
+ { &hf_vrt_tsmflag,
+ { "Timestamp mode", "vrt.tsmflag",
+ FT_UINT8, BASE_DEC,
+ VALS(tsm_types), 0x01,
+ NULL, HFILL }
+ },
+ { &hf_vrt_tsi,
+ { "Integer timestamp type", "vrt.tsi",
+ FT_UINT8, BASE_DEC,
+ VALS(tsi_types), 0xC0,
+ NULL, HFILL }
+ },
+ { &hf_vrt_tsf,
+ { "Fractional timestamp type", "vrt.tsf",
+ FT_UINT8, BASE_DEC,
+ VALS(tsf_types), 0x30,
+ NULL, HFILL }
+ },
+ { &hf_vrt_seq,
+ { "Sequence number", "vrt.seq",
+ FT_UINT8, BASE_DEC,
+ NULL, 0x0F,
+ NULL, HFILL }
+ },
+ { &hf_vrt_len,
+ { "Length", "vrt.len",
+ FT_UINT16, BASE_DEC,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_ts_int,
+ { "Integer timestamp", "vrt.ts_int",
+ FT_UINT32, BASE_DEC,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_ts_frac_sample,
+ { "Fractional timestamp (samples)", "vrt.ts_frac_sample",
+ FT_UINT64, BASE_DEC,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_ts_frac_picosecond,
+ { "Fractional timestamp (picoseconds)", "vrt.ts_frac_picosecond",
+ FT_UINT64, BASE_DEC,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_sid,
+ { "Stream ID", "vrt.sid",
+ FT_UINT32, BASE_HEX,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cid,
+ { "Class ID", "vrt.cid",
+ FT_UINT64, BASE_HEX,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif[0],
+ { "CIF0", "vrt.cif0",
+ FT_UINT32, BASE_HEX,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif0_change_flag,
+ { "Context field change indicator", "vrt.cif0.change",
+ FT_BOOLEAN, 8,
+ NULL, 0x80,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif0_ref_pt_id,
+ { "Reference point identifier", "vrt.cif0.refptid",
+ FT_BOOLEAN, 8,
+ NULL, 0x40,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif0_bandwidth,
+ { "Bandwidth", "vrt.cif0.bw",
+ FT_BOOLEAN, 8,
+ NULL, 0x20,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif0_if_freq,
+ { "IF reference frequency", "vrt.cif0.iffreq",
+ FT_BOOLEAN, 8,
+ NULL, 0x10,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif0_rf_freq,
+ { "RF reference frequency", "vrt.cif0.rffreq",
+ FT_BOOLEAN, 8,
+ NULL, 0x08,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif0_rf_freq_offset,
+ { "RF reference frequency offset", "vrt.cif0.rffreqoffset",
+ FT_BOOLEAN, 8,
+ NULL, 0x04,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif0_if_band_offset,
+ { "IF band offset", "vrt.cif0.ifbandoffset",
+ FT_BOOLEAN, 8,
+ NULL, 0x02,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif0_ref_level,
+ { "Reference level", "vrt.cif0.reflevel",
+ FT_BOOLEAN, 8,
+ NULL, 0x01,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif0_gain,
+ { "Gain", "vrt.cif0.gain",
+ FT_BOOLEAN, 8,
+ NULL, 0x80,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif0_over_range_count,
+ { "Over-range count", "vrt.cif0.overrangecount",
+ FT_BOOLEAN, 8,
+ NULL, 0x40,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif0_sample_rate,
+ { "Sample rate", "vrt.cif0.samplerate",
+ FT_BOOLEAN, 8,
+ NULL, 0x20,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif0_timestamp_adjust,
+ { "Timestamp adjustment", "vrt.cif0.timestampadjust",
+ FT_BOOLEAN, 8,
+ NULL, 0x10,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif0_timestamp_cal,
+ { "Timestamp calibration time", "vrt.cif0.timestampcal",
+ FT_BOOLEAN, 8,
+ NULL, 0x08,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif0_temperature,
+ { "Temperature", "vrt.cif0.temperature",
+ FT_BOOLEAN, 8,
+ NULL, 0x04,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif0_device_id,
+ { "Device identifier", "vrt.cif0.deviceid",
+ FT_BOOLEAN, 8,
+ NULL, 0x02,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif0_state_event,
+ { "State/event indicators", "vrt.cif0.stateevent",
+ FT_BOOLEAN, 8,
+ NULL, 0x01,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif0_signal_data_format,
+ { "Signal data format", "vrt.cif0.signaldataformat",
+ FT_BOOLEAN, 8,
+ NULL, 0x80,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif0_gps,
+ { "Formatted GPS", "vrt.cif0.gps",
+ FT_BOOLEAN, 8,
+ NULL, 0x40,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif0_ins,
+ { "Formatted INS", "vrt.cif0.ins",
+ FT_BOOLEAN, 8,
+ NULL, 0x20,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif0_ecef_ephemeris,
+ { "ECEF ephemeris", "vrt.cif0.ecefephem",
+ FT_BOOLEAN, 8,
+ NULL, 0x10,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif0_rel_ephemeris,
+ { "Relative ephemeris", "vrt.cif0.relephem",
+ FT_BOOLEAN, 8,
+ NULL, 0x08,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif0_ephemeris_ref_id,
+ { "Ephemeris ref ID", "vrt.cif0.ephemrefid",
+ FT_BOOLEAN, 8,
+ NULL, 0x04,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif0_gps_ascii,
+ { "GPS ASCII", "vrt.cif0.gpsascii",
+ FT_BOOLEAN, 8,
+ NULL, 0x02,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif0_context_assoc_lists,
+ { "Context association lists", "vrt.cif0.assoclists",
+ FT_BOOLEAN, 8,
+ NULL, 0x01,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif0_cif7,
+ { "CIF7", "vrt.cif0.cif7",
+ FT_BOOLEAN, 8,
+ NULL, 0x80,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif0_cif6,
+ { "CIF6", "vrt.cif0.cif6",
+ FT_BOOLEAN, 8,
+ NULL, 0x40,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif0_cif5,
+ { "CIF5", "vrt.cif0.cif5",
+ FT_BOOLEAN, 8,
+ NULL, 0x20,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif0_cif4,
+ { "CIF4", "vrt.cif0.cif4",
+ FT_BOOLEAN, 8,
+ NULL, 0x10,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif0_cif3,
+ { "CIF3", "vrt.cif0.cif3",
+ FT_BOOLEAN, 8,
+ NULL, 0x08,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif0_cif2,
+ { "CIF2", "vrt.cif0.cif2",
+ FT_BOOLEAN, 8,
+ NULL, 0x04,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif0_cif1,
+ { "CIF1", "vrt.cif0.cif1",
+ FT_BOOLEAN, 8,
+ NULL, 0x02,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif1_phase_offset,
+ { "Phase offset", "vrt.cif1.phaseoffset",
+ FT_BOOLEAN, 8,
+ NULL, 0x80,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif1_polarization,
+ { "Polarization", "vrt.cif1.polarization",
+ FT_BOOLEAN, 8,
+ NULL, 0x40,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif1_range,
+ { "Range (distance)", "vrt.cif1.range",
+ FT_BOOLEAN, 8,
+ NULL, 0x01,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif1_aux_freq,
+ { "Aux frequency", "vrt.cif1.auxfreq",
+ FT_BOOLEAN, 8,
+ NULL, 0x80,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif1_aux_bandwidth,
+ { "Aux bandwidth", "vrt.cif1.auxbw",
+ FT_BOOLEAN, 8,
+ NULL, 0x20,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif1_io32,
+ { "Discrete I/O (32-bit)", "vrt.cif1.io32",
+ FT_BOOLEAN, 8,
+ NULL, 0x40,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif1_io64,
+ { "Discrete I/O (64-bit)", "vrt.cif1.io64",
+ FT_BOOLEAN, 8,
+ NULL, 0x20,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif1_v49_spec,
+ { "V49 spec compliance", "vrt.cif1.v49spec",
+ FT_BOOLEAN, 8,
+ NULL, 0x08,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif1_ver,
+ { "Version and build code", "vrt.cif1.ver",
+ FT_BOOLEAN, 8,
+ NULL, 0x04,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif[1],
+ { "CIF1", "vrt.cif1",
+ FT_UINT32, BASE_HEX,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif[2],
+ { "CIF2", "vrt.cif2",
+ FT_UINT32, BASE_HEX,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif[3],
+ { "CIF3", "vrt.cif3",
+ FT_UINT32, BASE_HEX,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif[4],
+ { "CIF4", "vrt.cif4",
+ FT_UINT32, BASE_HEX,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif[5],
+ { "CIF5", "vrt.cif5",
+ FT_UINT32, BASE_HEX,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif[6],
+ { "CIF6", "vrt.cif6",
+ FT_UINT32, BASE_HEX,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cif[7],
+ { "CIF7", "vrt.cif7",
+ FT_UINT32, BASE_HEX,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ref_pt_id,
+ { "Reference point identifier", "vrt.context.refptid",
+ FT_UINT32, BASE_DEC,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_bandwidth,
+ { "Bandwidth", "vrt.context.bw",
+ FT_INT64, BASE_CUSTOM,
+ CF_FUNC(format_hertz), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_if_freq,
+ { "IF reference frequency", "vrt.context.iffreq",
+ FT_INT64, BASE_CUSTOM,
+ CF_FUNC(format_hertz), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_rf_freq,
+ { "RF reference frequency", "vrt.context.rffreq",
+ FT_INT64, BASE_CUSTOM,
+ CF_FUNC(format_hertz), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_rf_freq_offset,
+ { "RF reference frequency offset", "vrt.context.rffreqoffset",
+ FT_INT64, BASE_CUSTOM,
+ CF_FUNC(format_hertz), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_if_band_offset,
+ { "IF band offset", "vrt.context.ifbandoffset",
+ FT_INT64, BASE_CUSTOM,
+ CF_FUNC(format_hertz), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ref_level,
+ { "Reference level", "vrt.context.reflevel",
+ FT_INT16, BASE_CUSTOM,
+ CF_FUNC(format_decibel_milliwatt), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_gain_stage2,
+ { "Stage 2", "vrt.context.gain.stage2",
+ FT_INT16, BASE_CUSTOM,
+ CF_FUNC(format_decibel), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_gain_stage1,
+ { "Stage 1", "vrt.context.gain.stage1",
+ FT_INT16, BASE_CUSTOM,
+ CF_FUNC(format_decibel), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_over_range_count,
+ { "Over-range count", "vrt.context.overrangecount",
+ FT_UINT32, BASE_DEC,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_sample_rate,
+ { "Sample rate", "vrt.context.samplerate",
+ FT_INT64, BASE_CUSTOM,
+ CF_FUNC(format_hertz), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_timestamp_adjust,
+ { "Timestamp adjustment", "vrt.context.timestampadjust",
+ FT_INT64, BASE_CUSTOM,
+ CF_FUNC(format_second), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_timestamp_cal,
+ { "Timestamp calibration", "vrt.context.timestampcal",
+ FT_UINT32, BASE_DEC,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_temperature,
+ { "Device temperature", "vrt.context.temperature",
+ FT_INT16, BASE_CUSTOM,
+ CF_FUNC(format_celsius), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_device_id_oui,
+ { "Manufacturer OUI", "vrt.context.deviceid.oui",
+ FT_UINT24, BASE_HEX,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_device_id_code,
+ { "Device code", "vrt.context.deviceid.code",
+ FT_UINT16, BASE_DEC,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_state_event_en_cal_time,
+ { "Calibrated time enable", "vrt.context.stateevent.caltime.en",
+ FT_BOOLEAN, 8,
+ NULL, 0x80,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_state_event_en_valid_data,
+ { "Valid data enable", "vrt.context.stateevent.validdata.en",
+ FT_BOOLEAN, 8,
+ NULL, 0x40,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_state_event_en_ref_lock,
+ { "Reference lock enable", "vrt.context.stateevent.reflock.en",
+ FT_BOOLEAN, 8,
+ NULL, 0x20,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_state_event_en_agc,
+ { "AGC/MGC enable", "vrt.context.stateevent.agc.en",
+ FT_BOOLEAN, 8,
+ NULL, 0x10,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_state_event_en_detected_sig,
+ { "Detected signal enable", "vrt.context.stateevent.detectedsignal.en",
+ FT_BOOLEAN, 8,
+ NULL, 0x08,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_state_event_en_spectral_inv,
+ { "Spectral inversion enable", "vrt.context.stateevent.spectralinv.en",
+ FT_BOOLEAN, 8,
+ NULL, 0x04,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_state_event_en_over_range,
+ { "Over-range enable", "vrt.context.stateevent.overrange.en",
+ FT_BOOLEAN, 8,
+ NULL, 0x02,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_state_event_en_sample_loss,
+ { "Sample loss enable", "vrt.cif0.context.sampleloss.en",
+ FT_BOOLEAN, 8,
+ NULL, 0x01,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_state_event_cal_time,
+ { "Calibrated time indicator", "vrt.context.stateevent.caltime.val",
+ FT_BOOLEAN, 8,
+ NULL, 0x08,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_state_event_valid_data,
+ { "Valid data indicator", "vrt.context.stateevent.validdata.val",
+ FT_BOOLEAN, 8,
+ NULL, 0x04,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_state_event_ref_lock,
+ { "Reference lock indicator", "vrt.context.stateevent.reflock.val",
+ FT_BOOLEAN, 8,
+ NULL, 0x02,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_state_event_agc,
+ { "AGC/MGC indicator", "vrt.context.stateevent.agc.val",
+ FT_BOOLEAN, 8,
+ NULL, 0x01,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_state_event_detected_sig,
+ { "Detected signal indicator", "vrt.context.stateevent.detectedsignal.val",
+ FT_BOOLEAN, 8,
+ NULL, 0x80,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_state_event_spectral_inv,
+ { "Spectral inversion indicator", "vrt.context.stateevent.spectralinv.val",
+ FT_BOOLEAN, 8,
+ NULL, 0x40,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_state_event_over_range,
+ { "Over-range indicator", "vrt.context.stateevent.overrange.val",
+ FT_BOOLEAN, 8,
+ NULL, 0x20,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_state_event_sample_loss,
+ { "Sample loss indicator", "vrt.context.stateevent.sampleloss.val",
+ FT_BOOLEAN, 8,
+ NULL, 0x10,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_state_event_user,
+ { "User-defined", "vrt.context.stateevent.user",
+ FT_UINT8, BASE_HEX,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_signal_data_format_packing,
+ { "Packing method", "vrt.context.signaldataformat.packing",
+ FT_UINT8, BASE_DEC,
+ VALS(packing_method), 0x80,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_signal_data_format_type,
+ { "Real/complex type", "vrt.context.signaldataformat.realcomplex",
+ FT_UINT8, BASE_DEC,
+ VALS(data_sample_type), 0x60,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_signal_data_format_item,
+ { "Data item format", "vrt.context.signaldataformat.format",
+ FT_UINT8, BASE_DEC,
+ VALS(data_item_format), 0x1F,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_signal_data_format_repeat,
+ { "Sample-component repeat indicator", "vrt.context.signaldataformat.repeat",
+ FT_BOOLEAN, 8,
+ NULL, 0x80,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_signal_data_format_event_size,
+ { "Event-tag size", "vrt.context.signaldataformat.eventsize",
+ FT_UINT8, BASE_DEC,
+ NULL, 0x70,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_signal_data_format_channel_size,
+ { "Channel-tag size", "vrt.context.signaldataformat.channelsize",
+ FT_UINT8, BASE_DEC,
+ NULL, 0x0F,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_signal_data_format_fraction_size,
+ { "Data item fraction size", "vrt.context.signaldataformat.fractionsize",
+ FT_UINT16, BASE_DEC,
+ NULL, 0xF000,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_signal_data_format_packing_size,
+ { "Item packing field size", "vrt.context.signaldataformat.packingsize",
+ FT_UINT16, BASE_DEC,
+ NULL, 0x0FC0,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_signal_data_format_item_size,
+ { "Data item size", "vrt.context.signaldataformat.itemsize",
+ FT_UINT16, BASE_DEC,
+ NULL, 0x003F,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_signal_data_format_repeat_count,
+ { "Repeat count", "vrt.context.signaldataformat.repeatcount",
+ FT_UINT16, BASE_DEC,
+ NULL, 0x0,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_signal_data_format_vector_size,
+ { "Vector size", "vrt.context.signaldataformat.vectorsize",
+ FT_UINT16, BASE_DEC,
+ NULL, 0,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_gps.tsi,
+ { "Integer timestamp type", "vrt.context.gps.tsi",
+ FT_UINT8, BASE_DEC,
+ VALS(tsi_types), 0x0C,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_gps.tsf,
+ { "Fractional timestamp type", "vrt.context.gps.tsf",
+ FT_UINT8, BASE_DEC,
+ VALS(tsf_types), 0x03,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_gps.oui,
+ { "Manufacturer OUI", "vrt.context.gps.oui",
+ FT_UINT24, BASE_HEX,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_gps.ts_int,
+ { "Integer timestamp of position fix", "vrt.context.gps.ts_int",
+ FT_UINT32, BASE_DEC,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_gps.ts_frac_sample,
+ { "Fractional timestamp (samples)", "vrt.context.gps.ts_frac_sample",
+ FT_UINT64, BASE_DEC,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_gps.ts_picosecond,
+ { "Fractional timestamp (picoseconds)", "vrt.context.gps.ts_frac_picosecond",
+ FT_UINT64, BASE_DEC,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_gps.lat,
+ { "Latitude", "vrt.context.gps.lat",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_degrees), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_gps.lon,
+ { "Longitude", "vrt.context.gps.lon",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_degrees), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_gps.alt,
+ { "Altitude", "vrt.context.gps.alt",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_meter), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_gps.speed,
+ { "Speed over ground", "vrt.context.gps.speed",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_meters_per_second), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_gps.heading,
+ { "Heading angle", "vrt.context.gps.heading",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_degrees), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_gps.track,
+ { "Track angle", "vrt.context.gps.track",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_degrees), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_gps.mag_var,
+ { "Magnetic variation", "vrt.context.gps.mag_var",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_degrees), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ins.tsi,
+ { "Integer timestamp type", "vrt.context.ins.tsi",
+ FT_UINT8, BASE_DEC,
+ VALS(tsi_types), 0x0C,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ins.tsf,
+ { "Fractional timestamp type", "vrt.context.ins.tsf",
+ FT_UINT8, BASE_DEC,
+ VALS(tsf_types), 0x03,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ins.oui,
+ { "Manufacturer OUI", "vrt.context.ins.oui",
+ FT_UINT24, BASE_HEX,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ins.ts_int,
+ { "Integer timestamp of position fix", "vrt.context.ins.ts_int",
+ FT_UINT32, BASE_DEC,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ins.ts_frac_sample,
+ { "Fractional timestamp (samples)", "vrt.context.ins.ts_frac_sample",
+ FT_UINT64, BASE_DEC,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ins.ts_picosecond,
+ { "Fractional timestamp (picoseconds)", "vrt.context.ins.ts_frac_picosecond",
+ FT_UINT64, BASE_DEC,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ins.lat,
+ { "Latitude", "vrt.context.ins.lat",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_degrees), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ins.lon,
+ { "Longitude", "vrt.context.ins.lon",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_degrees), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ins.alt,
+ { "Altitude", "vrt.context.ins.alt",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_meter), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ins.speed,
+ { "Speed over ground", "vrt.context.ins.speed",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_meters_per_second), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ins.heading,
+ { "Heading angle", "vrt.context.ins.heading",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_degrees), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ins.track,
+ { "Track angle", "vrt.context.ins.track",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_degrees), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ins.mag_var,
+ { "Magnetic variation", "vrt.context.ins.mag_var",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_degrees), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ecef_ephemeris.tsi,
+ { "Integer timestamp type", "vrt.context.ecefephem.tsi",
+ FT_UINT8, BASE_DEC,
+ VALS(tsi_types), 0x0C,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ecef_ephemeris.tsf,
+ { "Fractional timestamp type", "vrt.context.ecefephem.tsf",
+ FT_UINT8, BASE_DEC,
+ VALS(tsf_types), 0x03,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ecef_ephemeris.oui,
+ { "Manufacturer OUI", "vrt.context.ecefephem.oui",
+ FT_UINT24, BASE_HEX,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ecef_ephemeris.ts_int,
+ { "Integer timestamp of position fix", "vrt.context.ecefephem.ts_int",
+ FT_UINT32, BASE_DEC,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ecef_ephemeris.ts_frac_sample,
+ { "Fractional timestamp (samples)", "vrt.context.ecefephem.ts_frac_sample",
+ FT_UINT64, BASE_DEC,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ecef_ephemeris.ts_picosecond,
+ { "Fractional timestamp (picoseconds)", "vrt.context.ecefephem.ts_frac_picosecond",
+ FT_UINT64, BASE_DEC,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ecef_ephemeris.pos_x,
+ { "Position X", "vrt.context.ecefephem.posx",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_meter), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ecef_ephemeris.pos_y,
+ { "Position Y", "vrt.context.ecefephem.posy",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_meter), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ecef_ephemeris.pos_z,
+ { "Position Z", "vrt.context.ecefephem.posz",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_meter), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ecef_ephemeris.att_alpha,
+ { "Attitude alpha (α)", "vrt.context.ecefephem.attalpha",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_degrees), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ecef_ephemeris.att_beta,
+ { "Attitude beta (β)", "vrt.context.ecefephem.attbeta",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_degrees), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ecef_ephemeris.att_phi,
+ { "Attitude phi (φ)", "vrt.context.ecefephem.attphi",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_degrees), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ecef_ephemeris.vel_dx,
+ { "Velocity dX", "vrt.context.ecefephem.veldx",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_meters_per_second), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ecef_ephemeris.vel_dy,
+ { "Velocity dY", "vrt.context.ecefephem.veldy",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_meters_per_second), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ecef_ephemeris.vel_dz,
+ { "Velocity dZ", "vrt.context.ecefephem.veldz",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_meters_per_second), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_rel_ephemeris.tsi,
+ { "Integer timestamp type", "vrt.context.relephem.tsi",
+ FT_UINT8, BASE_DEC,
+ VALS(tsi_types), 0x0C,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_rel_ephemeris.tsf,
+ { "Fractional timestamp type", "vrt.context.relephem.tsf",
+ FT_UINT8, BASE_DEC,
+ VALS(tsf_types), 0x03,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_rel_ephemeris.oui,
+ { "Manufacturer OUI", "vrt.context.relephem.oui",
+ FT_UINT24, BASE_HEX,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_rel_ephemeris.ts_int,
+ { "Integer timestamp of position fix", "vrt.context.relephem.ts_int",
+ FT_UINT32, BASE_DEC,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_rel_ephemeris.ts_frac_sample,
+ { "Fractional timestamp (samples)", "vrt.context.relephem.ts_frac_sample",
+ FT_UINT64, BASE_DEC,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_rel_ephemeris.ts_picosecond,
+ { "Fractional timestamp (picoseconds)", "vrt.context.relephem.ts_frac_picosecond",
+ FT_UINT64, BASE_DEC,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_rel_ephemeris.pos_x,
+ { "Position X", "vrt.context.relephem.posx",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_meter), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_rel_ephemeris.pos_y,
+ { "Position Y", "vrt.context.relephem.posy",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_meter), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_rel_ephemeris.pos_z,
+ { "Position Z", "vrt.context.relephem.posz",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_meter), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_rel_ephemeris.att_alpha,
+ { "Attitude alpha (α)", "vrt.context.relephem.attalpha",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_degrees), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_rel_ephemeris.att_beta,
+ { "Attitude beta (β)", "vrt.context.relephem.attbeta",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_degrees), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_rel_ephemeris.att_phi,
+ { "Attitude phi (φ)", "vrt.context.relephem.attphi",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_degrees), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_rel_ephemeris.vel_dx,
+ { "Velocity dX", "vrt.context.relephem.veldx",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_meters_per_second), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_rel_ephemeris.vel_dy,
+ { "Velocity dY", "vrt.context.relephem.veldy",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_meters_per_second), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_rel_ephemeris.vel_dz,
+ { "Velocity dZ", "vrt.context.relephem.veldz",
+ FT_INT32, BASE_CUSTOM,
+ CF_FUNC(format_meters_per_second), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ephemeris_ref_id,
+ { "Ephemeris reference identifier", "vrt.context.ephemrefid",
+ FT_UINT32, BASE_DEC,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_gps_ascii_oui,
+ { "Manufacturer OUI", "vrt.context.gpsascii.oui",
+ FT_UINT24, BASE_HEX,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_gps_ascii_size,
+ { "Number of words", "vrt.context.gpsascii.size",
+ FT_UINT32, BASE_DEC,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_gps_ascii_data,
+ { "Data", "vrt.context.gpsascii.data",
+ FT_BYTES, BASE_NONE,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_assoc_lists_src_size,
+ { "Source list size", "vrt.context.assoclists.src.size",
+ FT_UINT16, BASE_DEC,
+ NULL, 0x01FF,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_assoc_lists_sys_size,
+ { "System list size", "vrt.context.assoclists.sys.size",
+ FT_UINT16, BASE_DEC,
+ NULL, 0x01FF,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_assoc_lists_vec_size,
+ { "Vector-component list size", "vrt.context.assoclists.vec.size",
+ FT_UINT16, BASE_DEC,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_assoc_lists_a,
+ { "A bit (asynchronous-channel tag list present)", "vrt.context.assoclists.a",
+ FT_BOOLEAN, 8,
+ NULL, 0x80,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_assoc_lists_asy_size,
+ { "Asynchronous-channel list size", "vrt.context.assoclists.asy.size",
+ FT_UINT16, BASE_DEC,
+ NULL, 0x7FFF,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_assoc_lists_src_data,
+ { "Source context association list", "vrt.context.assoclists.src.data",
+ FT_BYTES, BASE_NONE,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_assoc_lists_sys_data,
+ { "System context association list", "vrt.context.assoclists.sys.data",
+ FT_BYTES, BASE_NONE,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_assoc_lists_vec_data,
+ { "Vector-component context association list", "vrt.context.assoclists.vec.data",
+ FT_BYTES, BASE_NONE,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_assoc_lists_asy_data,
+ { "Asynchronous-channel context association list", "vrt.context.assoclists.asy.data",
+ FT_BYTES, BASE_NONE,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_assoc_lists_asy_tag_data,
+ { "Asynchronous-channel tag list", "vrt.context.assoclists.asy.tagdata",
+ FT_BYTES, BASE_NONE,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_phase_offset,
+ { "Phase offset", "vrt.context.phaseoffset",
+ FT_INT16, BASE_CUSTOM,
+ CF_FUNC(format_radian_phase), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_pol_tilt,
+ { "Tilt angle (θ)", "vrt.context.polarization.tilt",
+ FT_INT16, BASE_CUSTOM,
+ CF_FUNC(format_radian_pol), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_pol_ellipticity,
+ { "Ellipticity angle (χ)", "vrt.context.polarization.ellipticity",
+ FT_INT16, BASE_CUSTOM,
+ CF_FUNC(format_radian_pol), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_range,
+ { "Range (distance)", "vrt.context.range",
+ FT_UINT32, BASE_CUSTOM,
+ CF_FUNC(format_meter_unsigned), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_aux_freq,
+ { "Aux frequency", "vrt.context.auxfreq",
+ FT_INT64, BASE_CUSTOM,
+ CF_FUNC(format_hertz), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_aux_bandwidth,
+ { "Aux bandwidth", "vrt.context.auxbw",
+ FT_INT64, BASE_CUSTOM,
+ CF_FUNC(format_hertz), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_io32,
+ { "Discrete I/O (32-bit)", "vrt.context.io32",
+ FT_UINT32, BASE_HEX,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_io64,
+ { "Discrete I/O (64-bit)", "vrt.context.io64",
+ FT_UINT64, BASE_HEX,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_v49_spec,
+ { "V49 spec compliance", "vrt.context.v49spec",
+ FT_UINT32, BASE_HEX,
+ VALS(standard_version_codes), 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ver_year,
+ { "Year", "vrt.context.ver.year",
+ FT_UINT16, BASE_DEC,
+ NULL, 0xFE00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ver_day,
+ { "Day", "vrt.context.ver.day",
+ FT_UINT16, BASE_DEC,
+ NULL, 0x01FF,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ver_rev,
+ { "Revision", "vrt.context.ver.rev",
+ FT_UINT16, BASE_DEC,
+ NULL, 0xFC00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_context_ver_user,
+ { "User defined", "vrt.context.ver.user",
+ FT_UINT16, BASE_DEC,
+ NULL, 0x03FF,
+ NULL, HFILL }
+ },
+ { &hf_vrt_data,
+ { "Data", "vrt.data",
+ FT_BYTES, BASE_NONE,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_trailer,
+ { "Trailer", "vrt.trailer",
+ FT_UINT32, BASE_HEX,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_trailer_enables,
+ { "Indicator enable bits", "vrt.enables",
+ FT_UINT16, BASE_HEX,
+ NULL, 0xFFF0,
+ NULL, HFILL }
+ },
+ { &hf_vrt_trailer_ind,
+ { "Indicator bits", "vrt.indicators",
+ FT_UINT16, BASE_HEX,
+ NULL, 0x0FFF,
+ NULL, HFILL }
+ },
+ { &hf_vrt_trailer_e,
+ { "Associated context packet count enabled", "vrt.e",
+ FT_BOOLEAN, 8,
+ NULL, 0x80,
+ NULL, HFILL }
+ },
+ { &hf_vrt_trailer_acpc,
+ { "Associated context packet count", "vrt.acpc",
+ FT_UINT8, BASE_DEC,
+ NULL, 0x7F,
+ NULL, HFILL }
+ },
+ { &hf_vrt_trailer_ind_caltime,
+ { "Calibrated time indicator", "vrt.caltime",
+ FT_BOOLEAN, 16,
+ NULL, 0x0800,
+ NULL, HFILL }
+ },
+ { &hf_vrt_trailer_ind_valid,
+ { "Valid signal indicator", "vrt.valid",
+ FT_BOOLEAN, 16,
+ NULL, 0x0400,
+ NULL, HFILL }
+ },
+ { &hf_vrt_trailer_ind_reflock,
+ { "Reference lock indicator", "vrt.reflock",
+ FT_BOOLEAN, 16,
+ NULL, 0x0200,
+ NULL, HFILL }
+ },
+ { &hf_vrt_trailer_ind_agc,
+ { "AGC/MGC indicator", "vrt.agc",
+ FT_BOOLEAN, 16,
+ NULL, 0x0100,
+ NULL, HFILL }
+ },
+ { &hf_vrt_trailer_ind_sig,
+ { "Signal detected indicator", "vrt.sig",
+ FT_BOOLEAN, 16,
+ NULL, 0x0080,
+ NULL, HFILL }
+ },
+ { &hf_vrt_trailer_ind_inv,
+ { "Spectral inversion indicator", "vrt.inv",
+ FT_BOOLEAN, 16,
+ NULL, 0x0040,
+ NULL, HFILL }
+ },
+ { &hf_vrt_trailer_ind_overrng,
+ { "Overrange indicator", "vrt.overrng",
+ FT_BOOLEAN, 16,
+ NULL, 0x0020,
+ NULL, HFILL }
+ },
+ { &hf_vrt_trailer_ind_sampleloss,
+ { "Lost sample indicator", "vrt.sampleloss",
+ FT_BOOLEAN, 16,
+ NULL, 0x0010,
+ NULL, HFILL }
+ },
+ { &hf_vrt_trailer_ind_user0,
+ { "User indicator 0", "vrt.user0",
+ FT_BOOLEAN, 16,
+ NULL, 0x0008,
+ NULL, HFILL }
+ },
+ { &hf_vrt_trailer_ind_user1,
+ { "User indicator 1", "vrt.user1",
+ FT_BOOLEAN, 16,
+ NULL, 0x0004,
+ NULL, HFILL }
+ },
+ { &hf_vrt_trailer_ind_user2,
+ { "User indicator 2", "vrt.user2",
+ FT_BOOLEAN, 16,
+ NULL, 0x0002,
+ NULL, HFILL }
+ },
+ { &hf_vrt_trailer_ind_user3,
+ { "User indicator 3", "vrt.user3",
+ FT_BOOLEAN, 16,
+ NULL, 0x0001,
+ NULL, HFILL }
+ },
+ { &hf_vrt_trailer_en_caltime,
+ { "Calibrated time indicator enable", "vrt.caltime_en",
+ FT_BOOLEAN, 16,
+ NULL, 0x8000,
+ NULL, HFILL }
+ },
+ { &hf_vrt_trailer_en_valid,
+ { "Valid signal indicator enable", "vrt.valid_en",
+ FT_BOOLEAN, 16,
+ NULL, 0x4000,
+ NULL, HFILL }
+ },
+ { &hf_vrt_trailer_en_reflock,
+ { "Reference lock indicator enable", "vrt.reflock_en",
+ FT_BOOLEAN, 16,
+ NULL, 0x2000,
+ NULL, HFILL }
+ },
+ { &hf_vrt_trailer_en_agc,
+ { "AGC/MGC indicator enable", "vrt.agc_en",
+ FT_BOOLEAN, 16,
+ NULL, 0x1000,
+ NULL, HFILL }
+ },
+ { &hf_vrt_trailer_en_sig,
+ { "Signal detected indicator enable", "vrt.sig_en",
+ FT_BOOLEAN, 16,
+ NULL, 0x0800,
+ NULL, HFILL }
+ },
+ { &hf_vrt_trailer_en_inv,
+ { "Spectral inversion indicator enable", "vrt.inv_en",
+ FT_BOOLEAN, 16,
+ NULL, 0x0400,
+ NULL, HFILL }
+ },
+ { &hf_vrt_trailer_en_overrng,
+ { "Overrange indicator enable", "vrt.overrng_en",
+ FT_BOOLEAN, 16,
+ NULL, 0x0200,
+ NULL, HFILL }
+ },
+ { &hf_vrt_trailer_en_sampleloss,
+ { "Lost sample indicator enable", "vrt.sampleloss_en",
+ FT_BOOLEAN, 16,
+ NULL, 0x0100,
+ NULL, HFILL }
+ },
+ { &hf_vrt_trailer_en_user0,
+ { "User indicator 0 enable", "vrt.user0_en",
+ FT_BOOLEAN, 16,
+ NULL, 0x0080,
+ NULL, HFILL }
+ },
+ { &hf_vrt_trailer_en_user1,
+ { "User indicator 1 enable", "vrt.user1_en",
+ FT_BOOLEAN, 16,
+ NULL, 0x0040,
+ NULL, HFILL }
+ },
+ { &hf_vrt_trailer_en_user2,
+ { "User indicator 2 enable", "vrt.user2_en",
+ FT_BOOLEAN, 16,
+ NULL, 0x0020,
+ NULL, HFILL }
+ },
+ { &hf_vrt_trailer_en_user3,
+ { "User indicator 3 enable", "vrt.user3_en",
+ FT_BOOLEAN, 16,
+ NULL, 0x0010,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cid_oui,
+ { "Class ID Organizationally Unique ID", "vrt.oui",
+ FT_UINT24, BASE_HEX,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cid_icc,
+ { "Class ID Information Class Code", "vrt.icc",
+ FT_UINT16, BASE_DEC,
+ NULL, 0x00,
+ NULL, HFILL }
+ },
+ { &hf_vrt_cid_pcc,
+ { "Class ID Packet Class Code", "vrt.pcc",
+ FT_UINT16, BASE_DEC,
+ NULL, 0x00,
+ NULL, HFILL }
+ }
+ };
+
+ // update ETT_IDX_* as new items added to track indices
+ static gint *ett[] = {
+ &ett_vrt,
+ &ett_header,
+ &ett_trailer,
+ &ett_indicators,
+ &ett_ind_enables,
+ &ett_cid,
+ &ett_cif0,
+ &ett_cif1,
+ &ett_gain, // ETT_IDX_GAIN
+ &ett_device_id, // ETT_IDX_DEVICE_ID
+ &ett_state_event, // ETT_IDX_STATE_EVENT
+ &ett_signal_data_format, // ETT_IDX_SIGNAL_DATA_FORMAT
+ &ett_gps, // ETT_IDX_GPS
+ &ett_ins, // ETT_IDX_INS
+ &ett_ecef_ephem, // ETT_IDX_ECEF_EPHEM
+ &ett_rel_ephem, // ETT_IDX_REL_EPHEM
+ &ett_gps_ascii, // ETT_IDX_GPS_ASCII
+ &ett_assoc_lists, // ETT_IDX_ASSOC_LISTS
+ &ett_pol, // ETT_IDX_POL
+ &ett_ver, // ETT_IDX_VER
+ };
+
+ proto_vrt = proto_register_protocol ("VITA 49 radio transport protocol", "VITA 49", "vrt");
+
+ proto_register_field_array(proto_vrt, hf, array_length(hf));
+ proto_register_subtree_array(ett, array_length(ett));
+
+ vrt_handle = register_dissector("vrt", dissect_vrt, proto_vrt);
+
+ vrt_module = prefs_register_protocol(proto_vrt, NULL);
+ prefs_register_bool_preference(vrt_module, "ettus_uhd_header_format",
+ "Use Ettus UHD header format",
+ "Activate workaround for weird Ettus UHD header offset on data packets",
+ &vrt_use_ettus_uhd_header_format);
+}
+
+void
+proto_reg_handoff_vrt(void)
+{
+ dissector_add_uint_with_preference("udp.port", VITA_49_PORT, vrt_handle);
+
+ dissector_add_string("rtp_dyn_payload_type","VITA 49", vrt_handle);
+ dissector_add_uint_range_with_preference("rtp.pt", "", vrt_handle);
+}
+
+/*
+ * Editor modelines - https://www.wireshark.org/tools/modelines.html
+ *
+ * Local variables:
+ * c-basic-offset: 4
+ * tab-width: 8
+ * indent-tabs-mode: nil
+ * End:
+ *
+ * vi: set shiftwidth=4 tabstop=8 expandtab:
+ * :indentSize=4:tabSize=8:noTabs=true:
+ */