summaryrefslogtreecommitdiffstats
path: root/epan/dissectors/file-tiff.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/file-tiff.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/file-tiff.c')
-rw-r--r--epan/dissectors/file-tiff.c1168
1 files changed, 1168 insertions, 0 deletions
diff --git a/epan/dissectors/file-tiff.c b/epan/dissectors/file-tiff.c
new file mode 100644
index 00000000..ac5f4212
--- /dev/null
+++ b/epan/dissectors/file-tiff.c
@@ -0,0 +1,1168 @@
+/* file-tiff.c
+ *
+ * Routines for image/tiff dissection
+ * Copyright 2021, Daniel Dulaney.
+ *
+ * Wireshark - Network traffic analyzer
+ * By Gerald Combs <gerald@wireshark.org>
+ * Copyright 1998 Gerald Combs
+ *
+ * The TIFF 6 specification can be found at:
+ * https://www.adobe.io/content/dam/udp/en/open/standards/tiff/TIFF6.pdf
+ *
+ * SPDX-License-Identifier: GPL-2.0-or-later
+ */
+
+#include "config.h"
+#include <epan/packet.h>
+#include <epan/expert.h>
+
+void proto_reg_handoff_tiff(void);
+void proto_register_tiff(void);
+
+static int proto_tiff = -1;
+
+// Header fields
+static int hf_tiff_header_endianness = -1;
+static int hf_tiff_header_magic = -1;
+static int hf_tiff_header_lead_ifd = -1;
+
+// IFD fields
+static int hf_tiff_ifd_count = -1;
+static int hf_tiff_ifd_next = -1;
+
+// Entry fields
+static int hf_tiff_entry_tag = -1;
+static int hf_tiff_entry_type = -1;
+static int hf_tiff_entry_count = -1;
+static int hf_tiff_entry_offset = -1;
+static int hf_tiff_entry_unknown = -1;
+
+// Expert fields
+static expert_field ei_tiff_unknown_tag = EI_INIT;
+static expert_field ei_tiff_bad_entry = EI_INIT;
+static expert_field ei_tiff_zero_denom = EI_INIT;
+
+
+static gint ett_tiff = -1;
+static gint ett_ifd = -1;
+static gint ett_t6 = -1;
+
+#define TIFF_TAG_NEW_SUBFILE_TYPE 254
+// Fields TBD
+
+#define TIFF_TAG_SUBFILE_TYPE 255
+// Fields TBD
+
+#define TIFF_TAG_IMAGE_WIDTH 256
+static int hf_tiff_image_width = -1;
+
+#define TIFF_TAG_IMAGE_LENGTH 257
+static int hf_tiff_image_length = -1;
+
+#define TIFF_TAG_BITS_PER_SAMPLE 258
+static int hf_tiff_bits_per_sample = -1;
+
+#define TIFF_TAG_COMPRESSION 259
+static int hf_tiff_compression = -1;
+
+#define TIFF_TAG_PHOTOMETRIC_INTERPRETATION 262
+static int hf_tiff_photometric_interp = -1;
+
+#define TIFF_TAG_THRESHHOLDING 263
+static int hf_tiff_threshholding = -1;
+
+#define TIFF_TAG_CELL_WIDTH 264
+static int hf_tiff_cell_width = -1;
+
+#define TIFF_TAG_CELL_LENGTH 265
+static int hf_tiff_cell_length = -1;
+
+#define TIFF_TAG_FILL_ORDER 266
+static int hf_tiff_fill_order = -1;
+
+#define TIFF_TAG_DOCUMENT_NAME 269
+static int hf_tiff_document_name = -1;
+
+#define TIFF_TAG_IMAGE_DESCRIPTION 270
+static int hf_tiff_image_description = -1;
+
+#define TIFF_TAG_MAKE 271
+static int hf_tiff_make = -1;
+
+#define TIFF_TAG_MODEL 272
+static int hf_tiff_model = -1;
+
+#define TIFF_TAG_STRIP_OFFSETS 273
+static int hf_tiff_strip_offset = -1;
+
+#define TIFF_TAG_ORIENTATION 274
+static int hf_tiff_orientation = -1;
+
+#define TIFF_TAG_SAMPLES_PER_PIXEL 277
+static int hf_tiff_samples_per_pixel = -1;
+
+#define TIFF_TAG_ROWS_PER_STRIP 278
+static int hf_tiff_rows_per_strip = -1;
+
+#define TIFF_TAG_STRIP_BYTE_COUNTS 279
+static int hf_tiff_strip_byte_count = -1;
+
+#define TIFF_TAG_MIN_SAMPLE_VALUE 280
+// Fields TBD
+
+#define TIFF_TAG_MAX_SAMPLE_VALUE 281
+// Fields TBD
+
+#define TIFF_TAG_X_RESOLUTION 282
+static int hf_tiff_x_res_numer = -1;
+static int hf_tiff_x_res_denom = -1;
+static int hf_tiff_x_res_approx = -1;
+
+#define TIFF_TAG_Y_RESOLUTION 283
+static int hf_tiff_y_res_numer = -1;
+static int hf_tiff_y_res_denom = -1;
+static int hf_tiff_y_res_approx = -1;
+
+#define TIFF_TAG_PLANAR_CONFIGURATION 284
+static int hf_tiff_planar_configuration = -1;
+
+#define TIFF_TAG_PAGE_NAME 285
+static int hf_tiff_page_name = -1;
+
+#define TIFF_TAG_X_POSITION 286
+// Fields TBD
+
+#define TIFF_TAG_Y_POSITION 287
+// Fields TBD
+
+#define TIFF_TAG_FREE_OFFSETS 288
+// Fields TBD
+
+#define TIFF_TAG_FREE_BYTE_COUNTS 289
+// Fields TBD
+
+#define TIFF_TAG_GRAY_RESPONSE_UNIT 290
+static int hf_tiff_gray_response_unit = -1;
+
+#define TIFF_TAG_GRAY_RESPONSE_CURVE 291
+// Fields TBD
+
+#define TIFF_TAG_T4_OPTIONS 292
+// Fields TBD
+
+#define TIFF_TAG_T6_OPTIONS 293
+static int hf_tiff_t6_options = -1;
+static int hf_tiff_t6_unused = -1;
+static int hf_tiff_t6_allow_uncompresed = -1;
+
+#define TIFF_TAG_RESOLUTION_UNIT 296
+static int hf_tiff_resolution_unit = -1;
+
+#define TIFF_TAG_PAGE_NUMBER 297
+// Fields TBD
+
+#define TIFF_TAG_TRANSFER_FUNCTION 301
+// Fields TBD
+
+#define TIFF_TAG_SOFTWARE 305
+static int hf_tiff_software = -1;
+
+#define TIFF_TAG_DATE_TIME 306
+static int hf_tiff_date_time = -1;
+
+#define TIFF_TAG_ARTIST 315
+static int hf_tiff_artist = -1;
+
+#define TIFF_TAG_HOST_COMPUTER 316
+static int hf_tiff_host_computer = -1;
+
+#define TIFF_TAG_PREDICTOR 317
+static int hf_tiff_predictor = -1;
+
+#define TIFF_TAG_WHITE_POINT 318
+// Fields TBD
+
+#define TIFF_TAG_PRIMARY_CHROMATICITIES 319
+// Fields TBD
+
+#define TIFF_TAG_COLOR_MAP 320
+// Fields TBD
+
+#define TIFF_TAG_HALFTONE_HINTS 321
+// Fields TBD
+
+#define TIFF_TAG_TILE_WIDTH 322
+static int hf_tiff_tile_width = -1;
+
+#define TIFF_TAG_TILE_LENGTH 323
+static int hf_tiff_tile_length = -1;
+
+#define TIFF_TAG_TILE_OFFSETS 324
+// Fields TBD
+
+#define TIFF_TAG_TILE_BYTE_COUNTS 325
+// Fields TBD
+
+#define TIFF_TAG_INK_SET 332
+static int hf_tiff_ink_set = -1;
+
+#define TIFF_TAG_INK_NAMES 333
+// Fields TBD
+
+#define TIFF_TAG_NUMBER_OF_INKS 334
+static int hf_tiff_number_of_inks = -1;
+
+#define TIFF_TAG_DOT_RANGE 336
+// Fields TBD
+
+#define TIFF_TAG_TARGET_PRINTER 337
+static int hf_tiff_target_printer = -1;
+
+#define TIFF_TAG_EXTRA_SAMPLES 338
+// Fields TBD
+
+#define TIFF_TAG_SAMPLE_FORMAT 339
+// Fields TBD
+
+#define TIFF_TAG_S_MIN_SAMPLE_VALUE 340
+// Fields TBD
+
+#define TIFF_TAG_S_MAX_SAMPLE_VALUE 341
+// Fields TBD
+
+#define TIFF_TAG_TRANSFER_RANGE 342
+// Fields TBD
+
+#define TIFF_TAG_JPEG_PROC 512
+// Fields TBD
+
+#define TIFF_TAG_JPEG_INTERCHANGE_FORMAT 513
+// Fields TBD
+
+#define TIFF_TAG_JPEG_INTERCHANGE_FORMAT_LENGTH 514
+// Fields TBD
+
+#define TIFF_TAG_JPEG_RESTART_INTERVAL 515
+// Fields TBD
+
+#define TIFF_TAG_JPEG_LOSSLESS_PREDICTORS 517
+// Fields TBD
+
+#define TIFF_TAG_JPEG_POINT_TRANSFORMS 518
+// Fields TBD
+
+#define TIFF_TAG_JPEG_Q_TABLES 519
+// Fields TBD
+
+#define TIFF_TAG_JPEG_DC_TABLES 520
+// Fields TBD
+
+#define TIFF_TAG_JPEG_AC_TABLES 521
+// Fields TBD
+
+#define TIFF_TAG_YCBCR_COEFFICIENTS 529
+// Fields TBD
+
+#define TIFF_TAG_YCBCR_SUBSAMPLING 530
+// Fields TBD
+
+#define TIFF_TAG_YCBCR_POSITIONING 531
+// Fields TBD
+
+#define TIFF_TAG_REFERENCE_BLACK_WHITE 532
+// Fields TBD
+
+#define TIFF_TAG_COPYRIGHT 0x8298
+static int hf_tiff_copyright = -1;
+
+static const value_string tiff_endianness_names[] = {
+ { 0x4949, "Little-Endian" },
+ { 0x4D4D, "Big-Endian" },
+ { 0, NULL },
+};
+
+static const value_string tiff_tag_names[] = {
+ { TIFF_TAG_NEW_SUBFILE_TYPE, "New Subfile Type" },
+ { TIFF_TAG_SUBFILE_TYPE, "Subfile Type" },
+ { TIFF_TAG_IMAGE_WIDTH, "Image Width" },
+ { TIFF_TAG_IMAGE_LENGTH, "Image Length" },
+ { TIFF_TAG_BITS_PER_SAMPLE, "Bits Per Sample" },
+ { TIFF_TAG_COMPRESSION, "Compression" },
+ { TIFF_TAG_PHOTOMETRIC_INTERPRETATION, "Photometric Interpretation" },
+ { TIFF_TAG_THRESHHOLDING, "Threshholding" },
+ { TIFF_TAG_CELL_WIDTH, "Cell Width" },
+ { TIFF_TAG_CELL_LENGTH, "Cell Length" },
+ { TIFF_TAG_FILL_ORDER, "Fill Order" },
+ { TIFF_TAG_DOCUMENT_NAME, "Document Name" },
+ { TIFF_TAG_IMAGE_DESCRIPTION, "Image Description" },
+ { TIFF_TAG_MAKE, "Make" },
+ { TIFF_TAG_MODEL, "Model" },
+ { TIFF_TAG_STRIP_OFFSETS, "Strip Offsets" },
+ { TIFF_TAG_ORIENTATION, "Orientation" },
+ { TIFF_TAG_SAMPLES_PER_PIXEL, "Samples Per Pixel" },
+ { TIFF_TAG_ROWS_PER_STRIP, "Rows Per Strip" },
+ { TIFF_TAG_STRIP_BYTE_COUNTS, "Strip Byte Counts" },
+ { TIFF_TAG_MIN_SAMPLE_VALUE, "Min Sample Value" },
+ { TIFF_TAG_MAX_SAMPLE_VALUE, "Max Sample Value" },
+ { TIFF_TAG_X_RESOLUTION, "X Resolution" },
+ { TIFF_TAG_Y_RESOLUTION, "Y Resolution" },
+ { TIFF_TAG_PLANAR_CONFIGURATION, "Planar Configuration" },
+ { TIFF_TAG_PAGE_NAME, "Page Name" },
+ { TIFF_TAG_X_POSITION, "X Position" },
+ { TIFF_TAG_Y_POSITION, "Y Position" },
+ { TIFF_TAG_FREE_OFFSETS, "Free Offsets" },
+ { TIFF_TAG_FREE_BYTE_COUNTS, "Free Byte Counts" },
+ { TIFF_TAG_GRAY_RESPONSE_UNIT, "Gray Response Unit" },
+ { TIFF_TAG_GRAY_RESPONSE_CURVE, "Gray Response Curve" },
+ { TIFF_TAG_T4_OPTIONS, "T4 Options" },
+ { TIFF_TAG_T6_OPTIONS, "T6 Options" },
+ { TIFF_TAG_RESOLUTION_UNIT, "Resolution Unit" },
+ { TIFF_TAG_PAGE_NUMBER, "Page Number" },
+ { TIFF_TAG_TRANSFER_FUNCTION, "Transfer Function" },
+ { TIFF_TAG_SOFTWARE, "Software" },
+ { TIFF_TAG_DATE_TIME, "Date Time" },
+ { TIFF_TAG_ARTIST, "Artist" },
+ { TIFF_TAG_HOST_COMPUTER, "Host Computer" },
+ { TIFF_TAG_PREDICTOR, "Predictor" },
+ { TIFF_TAG_WHITE_POINT, "White Point" },
+ { TIFF_TAG_PRIMARY_CHROMATICITIES, "Primary Chromaticities" },
+ { TIFF_TAG_COLOR_MAP, "Color Map" },
+ { TIFF_TAG_HALFTONE_HINTS, "Halftone Hints" },
+ { TIFF_TAG_TILE_WIDTH, "Tile Width" },
+ { TIFF_TAG_TILE_LENGTH, "Tile Length" },
+ { TIFF_TAG_TILE_OFFSETS, "Tile Offsets" },
+ { TIFF_TAG_TILE_BYTE_COUNTS, "Tile Byte Counts" },
+ { TIFF_TAG_INK_SET, "Ink Set" },
+ { TIFF_TAG_INK_NAMES, "Ink Names" },
+ { TIFF_TAG_NUMBER_OF_INKS, "Number Of Inks" },
+ { TIFF_TAG_DOT_RANGE, "Dot Range" },
+ { TIFF_TAG_TARGET_PRINTER, "Target Printer" },
+ { TIFF_TAG_EXTRA_SAMPLES, "Extra Samples" },
+ { TIFF_TAG_SAMPLE_FORMAT, "Sample Format" },
+ { TIFF_TAG_S_MIN_SAMPLE_VALUE, "S Min Sample Value" },
+ { TIFF_TAG_S_MAX_SAMPLE_VALUE, "S Max Sample Value" },
+ { TIFF_TAG_TRANSFER_RANGE, "Transfer Range" },
+ { TIFF_TAG_JPEG_PROC, "JPEG Proc" },
+ { TIFF_TAG_JPEG_INTERCHANGE_FORMAT, "JPEG Interchange Format" },
+ { TIFF_TAG_JPEG_INTERCHANGE_FORMAT_LENGTH, "JPEG Interchange Format Length" },
+ { TIFF_TAG_JPEG_RESTART_INTERVAL, "JPEG Restart Interval" },
+ { TIFF_TAG_JPEG_LOSSLESS_PREDICTORS, "JPEG Lossless Predictors" },
+ { TIFF_TAG_JPEG_POINT_TRANSFORMS, "JPEG Point Transforms" },
+ { TIFF_TAG_JPEG_Q_TABLES, "JPEG Q Tables" },
+ { TIFF_TAG_JPEG_DC_TABLES, "JPEG DC Tables" },
+ { TIFF_TAG_JPEG_AC_TABLES, "JPEG AC Tables" },
+ { TIFF_TAG_YCBCR_COEFFICIENTS, "YCbCr Coefficients" },
+ { TIFF_TAG_YCBCR_SUBSAMPLING, "YCbCr Subsampling" },
+ { TIFF_TAG_YCBCR_POSITIONING, "YCbCr Positioning" },
+ { TIFF_TAG_REFERENCE_BLACK_WHITE, "Reference Black White" },
+ { TIFF_TAG_COPYRIGHT, "Copyright" },
+ { 0, NULL },
+};
+
+#define TIFF_TYPE_BYTE 1
+#define TIFF_TYPE_ASCII 2
+#define TIFF_TYPE_SHORT 3
+#define TIFF_TYPE_LONG 4
+#define TIFF_TYPE_RATIONAL 5
+#define TIFF_TYPE_SBYTE 6
+#define TIFF_TYPE_UNDEFINED 7
+#define TIFF_TYPE_SSHORT 8
+#define TIFF_TYPE_SLONG 9
+#define TIFF_TYPE_SRATIONAL 10
+#define TIFF_TYPE_FLOAT 11
+#define TIFF_TYPE_DOUBLE 12
+
+static const value_string tiff_type_names[] = {
+ { TIFF_TYPE_BYTE, "Byte" },
+ { TIFF_TYPE_ASCII, "ASCII" },
+ { TIFF_TYPE_SHORT, "Unsigned Short" },
+ { TIFF_TYPE_LONG, "Unsigned Long" },
+ { TIFF_TYPE_RATIONAL, "Rational" },
+ { TIFF_TYPE_SBYTE, "Signed Byte" },
+ { TIFF_TYPE_UNDEFINED, "Undefined" },
+ { TIFF_TYPE_SSHORT, "Signed Short" },
+ { TIFF_TYPE_SLONG, "Signed Long" },
+ { TIFF_TYPE_SRATIONAL, "Signed Rational" },
+ { TIFF_TYPE_FLOAT, "Float" },
+ { TIFF_TYPE_DOUBLE, "Double" },
+ { 0, NULL },
+};
+
+static const value_string tiff_compression_names[] = {
+ { 1, "Uncompressed" },
+ { 2, "CITT 1D" },
+ { 3, "Group 3 Fax" },
+ { 4, "Group 4 Fax" },
+ { 5, "LZW" },
+ { 6, "JPEG" },
+ { 32773, "PackBits" },
+ { 0, NULL },
+};
+
+static const value_string tiff_photometric_interp_names[] = {
+ { 0, "White is Zero" },
+ { 1, "Black is Zero" },
+ { 2, "RGB" },
+ { 3, "RGB Palette" },
+ { 4, "Transparency Mask" },
+ { 5, "CMYK" },
+ { 6, "YCbCr" },
+ /* N.B. no value for 7 (see Appendix A) */
+ { 8, "CIELab" },
+ { 0, NULL },
+};
+
+static const value_string tiff_threshholding_names[] = {
+ { 0, "None" },
+ { 1, "Ordered" },
+ { 2, "Randomized" },
+ { 0, NULL },
+};
+
+static const value_string tiff_fill_order_names[] = {
+ { 1, "High-order first" },
+ { 2, "Low-order first" },
+ { 0, NULL },
+};
+
+static const value_string tiff_orientation_names[] = {
+ { 1, "Origin at Top-Left, Horizontal Rows" },
+ { 2, "Origin at Top-Right, Horizontal Rows" },
+ { 3, "Origin at Bottom-Right, Horizontal Rows" },
+ { 4, "Origin at Bottom-Left, Horizontal Rows" },
+ { 5, "Origin at Top-Left, Vertical Rows" },
+ { 6, "Origin at Top-Right, Vertical Rows" },
+ { 7, "Origin at Bottom-Right, Vertical Rows" },
+ { 8, "Origin at Bottom-Left, Vertical Rows" },
+ { 0, NULL },
+};
+
+static const value_string tiff_planar_configuration_names[] = {
+ { 1, "Chunky" },
+ { 2, "Planar" },
+ { 0, NULL },
+};
+
+static const value_string tiff_gray_response_unit_names[] = {
+ { 1, "Tenths" },
+ { 2, "Hundredths" },
+ { 3, "Thousandths" },
+ { 4, "Ten-thousandths" },
+ { 5, "Hundred-thousandths" },
+ { 0, NULL },
+};
+
+static const value_string tiff_allow_uncompressed_names[] = {
+ { 0, "Not Allowed" },
+ { 1, "Allowed" },
+ { 0, NULL },
+};
+
+static const value_string tiff_resolution_unit_names[] = {
+ { 1, "None" },
+ { 2, "Inch" },
+ { 3, "Centimeter" },
+ { 0, NULL },
+};
+
+static const value_string tiff_predictor_names[] = {
+ { 1, "No Predictor" },
+ { 2, "Horizontal Differencing" },
+ { 0, NULL },
+};
+
+static const value_string tiff_ink_set_names[] = {
+ { 1, "CMYK" },
+ { 2, "Not CMYK" },
+ { 0, NULL },
+};
+
+// Return the length of the given data type.
+//
+// If the type isn't known, return -1.
+static gint
+tiff_type_len(const guint16 type) {
+ switch (type) {
+ case TIFF_TYPE_BYTE: return 1;
+ case TIFF_TYPE_ASCII: return 1;
+ case TIFF_TYPE_SHORT: return 2;
+ case TIFF_TYPE_LONG: return 4;
+ case TIFF_TYPE_RATIONAL: return 8;
+ case TIFF_TYPE_SBYTE: return 1;
+ case TIFF_TYPE_UNDEFINED: return 1;
+ case TIFF_TYPE_SSHORT: return 2;
+ case TIFF_TYPE_SLONG: return 4;
+ case TIFF_TYPE_SRATIONAL: return 8;
+ case TIFF_TYPE_FLOAT: return 4;
+ case TIFF_TYPE_DOUBLE: return 8;
+
+ default: return -1;
+ }
+}
+
+// Return the length of the given array of data.
+//
+// If the type isn't known, return -1.
+static gint
+tiff_data_len(const guint16 type, const guint32 count) {
+ const gint field = tiff_type_len(type);
+ if (field < 0) return -1;
+ else return field * count;
+}
+
+static void
+dissect_tiff_tag_unknown(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, guint32 offset, guint16 type, guint32 count, gint encoding _U_)
+{
+ const gint len = tiff_data_len(type, count);
+
+ expert_add_info(pinfo, tree, &ei_tiff_unknown_tag);
+
+ guint32 item_offset;
+ if (len <= 0) {
+ // If we can't determine the length, that's an issue
+ expert_add_info_format(pinfo, tree, &ei_tiff_bad_entry, "Could not determine length of entry");
+ return;
+ } else if (len <= 4) {
+ // If the length is <= 4, the item is located directly at the offset
+ item_offset = offset;
+ } else {
+ // If the length is >4, the offset is a pointer indicating where the item is located
+ proto_tree_add_item_ret_uint(tree, hf_tiff_entry_offset, tvb, offset, 4, encoding, &item_offset);
+ }
+
+ proto_tree_add_item(tree, hf_tiff_entry_unknown, tvb, item_offset, len, encoding);
+}
+
+static void
+dissect_tiff_single_uint(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, guint32 offset, guint16 type, guint32 count, gint encoding, int hfindex) {
+ if (count != 1) {
+ expert_add_info_format(pinfo, tree, &ei_tiff_bad_entry, "Expected a single item; found %d items", count);
+ return;
+ }
+
+ if (type == TIFF_TYPE_BYTE) {
+ proto_tree_add_item(tree, hfindex, tvb, offset, 1, encoding);
+ } else if (type == TIFF_TYPE_SHORT) {
+ proto_tree_add_item(tree, hfindex, tvb, offset, 2, encoding);
+ } else if (type == TIFF_TYPE_LONG) {
+ proto_tree_add_item(tree, hfindex, tvb, offset, 4, encoding);
+ } else {
+ expert_add_info_format(pinfo, tree, &ei_tiff_bad_entry, "Expected an unsigned integer, found type %s", val_to_str_const(type, tiff_type_names, "Unknown"));
+ }
+}
+
+static void
+dissect_tiff_array_uint(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, guint32 offset, guint16 type, guint32 count, gint encoding, int hfindex) {
+ if (!(type == TIFF_TYPE_BYTE || type == TIFF_TYPE_SHORT || type == TIFF_TYPE_LONG)) {
+ expert_add_info_format(pinfo, tree, &ei_tiff_bad_entry, "Expected an unsigned integer, found type %s", val_to_str_const(type, tiff_type_names, "Unknown"));
+ return;
+ }
+
+ if (count < 1) {
+ expert_add_info_format(pinfo, tree, &ei_tiff_bad_entry, "At least 1 item; found %d items", count);
+ return;
+ }
+
+ const gint item_len = tiff_type_len(type);
+ const gint len = tiff_data_len(type, count);
+
+ guint32 item_offset;
+ if (len <= 0 || item_len <= 0) {
+ // If we can't determine the length, that's an issue
+ expert_add_info_format(pinfo, tree, &ei_tiff_bad_entry, "Could not determine length of entry");
+ return;
+ } else if (len <= 4) {
+ // If the length is <= 4, the item is located directly at the offset
+ item_offset = offset;
+ } else {
+ // If the length is >4, the offset is a pointer indicating where the item is located
+ proto_tree_add_item_ret_uint(tree, hf_tiff_entry_offset, tvb, offset, 4, encoding, &item_offset);
+ }
+
+ // Add each item
+ for (guint32 i = 0; i < count; i++) {
+ proto_tree_add_item(tree, hfindex, tvb, item_offset + item_len * i, item_len, encoding);
+ }
+}
+
+static void
+dissect_tiff_single_string(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, guint offset, guint16 type, guint32 count, gint encoding, int hfindex) {
+ if (type != TIFF_TYPE_ASCII) {
+ expert_add_info_format(pinfo, tree, &ei_tiff_bad_entry, "Expected an ASCII string");
+ return;
+ }
+
+ guint32 item_offset;
+ if (count == 0) {
+ expert_add_info_format(pinfo, tree, &ei_tiff_bad_entry, "Expected at least one byte for an ASCII string; got zero");
+ return;
+ } else if (count <= 4) {
+ // If there are 4 or fewer bytes, the string is embedded in the pointer
+ item_offset = offset;
+ } else {
+ // If the length is >4, the offset is a pointer indicating where the item is located
+ proto_tree_add_item_ret_uint(tree, hf_tiff_entry_offset, tvb, offset, 4, encoding, &item_offset);
+ }
+
+ proto_tree_add_item(tree, hfindex, tvb, item_offset, count, ENC_ASCII);
+}
+
+static void
+dissect_tiff_single_urational(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, guint offset, guint16 type, guint32 count, gint encoding, int hfnumer, int hfdenom, int hfapprox) {
+ if (count != 1) {
+ expert_add_info_format(pinfo, tree, &ei_tiff_bad_entry, "Expected a single item; found %d items", count);
+ return;
+ }
+
+ if (type != TIFF_TYPE_RATIONAL) {
+ expert_add_info_format(pinfo, tree, &ei_tiff_bad_entry, "Expected an unsigned rational");
+ return;
+ }
+
+ guint32 item_offset;
+ proto_tree_add_item_ret_uint(tree, hf_tiff_entry_offset, tvb, offset, 4, encoding, &item_offset);
+
+ guint32 numer = 0;
+ guint32 denom = 0;
+ proto_tree_add_item_ret_uint(tree, hfnumer, tvb, item_offset, 4, encoding, &numer);
+ proto_item *denom_ti = proto_tree_add_item_ret_uint(tree, hfdenom, tvb, item_offset + 4, 4, encoding, &denom);
+
+ if (denom > 0) {
+ proto_item *approx_item = proto_tree_add_double(tree, hfapprox, tvb, item_offset, 8, (double)numer / (double)denom);
+ proto_item_set_generated(approx_item);
+ }
+ else {
+ expert_add_info_format(pinfo, denom_ti, &ei_tiff_zero_denom, "Denominator is zero");
+ }
+}
+
+static void
+dissect_tiff_t6_options(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, guint offset, guint16 type, guint32 count, gint encoding) {
+ if (count != 1) {
+ expert_add_info_format(pinfo, tree, &ei_tiff_bad_entry, "Expected a single item; found %d items", count);
+ return;
+ }
+
+ if (type != TIFF_TYPE_LONG) {
+ expert_add_info_format(pinfo, tree, &ei_tiff_bad_entry, "Expected an unsigned long");
+ return;
+ }
+
+ proto_item *t6_ti = proto_tree_add_item(tree, hf_tiff_t6_options, tvb, offset, 4, encoding);
+ proto_tree *t6_tree = proto_item_add_subtree(t6_ti, ett_t6);
+ proto_tree_add_item(t6_tree, hf_tiff_t6_unused, tvb, offset, 4, encoding);
+ proto_tree_add_item(t6_tree, hf_tiff_t6_allow_uncompresed, tvb, offset, 4, encoding);
+}
+
+static void
+dissect_tiff_entry(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, guint32 offset, gint encoding) {
+ const guint16 tag = tvb_get_guint16(tvb, offset, encoding);
+
+ proto_tree *entry_tree = proto_tree_add_subtree_format(tree, tvb, offset, 12, ett_ifd, NULL, "%s", val_to_str_const(tag, tiff_tag_names, "Unknown Entry"));
+
+ proto_tree_add_item(entry_tree, hf_tiff_entry_tag, tvb, offset, 2, encoding);
+
+ guint32 type = 0;
+ guint32 count = 0;
+ proto_tree_add_item_ret_uint(entry_tree, hf_tiff_entry_type, tvb, offset + 2, 2, encoding, &type);
+ proto_tree_add_item_ret_uint(entry_tree, hf_tiff_entry_count, tvb, offset + 4, 4, encoding, &count);
+
+ switch (tag) {
+ case TIFF_TAG_IMAGE_WIDTH:
+ dissect_tiff_single_uint(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_image_width);
+ break;
+ case TIFF_TAG_IMAGE_LENGTH:
+ dissect_tiff_single_uint(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_image_length);
+ break;
+ case TIFF_TAG_BITS_PER_SAMPLE:
+ dissect_tiff_array_uint(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_bits_per_sample);
+ break;
+ case TIFF_TAG_COMPRESSION:
+ dissect_tiff_single_uint(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_compression);
+ break;
+ case TIFF_TAG_PHOTOMETRIC_INTERPRETATION:
+ dissect_tiff_single_uint(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_photometric_interp);
+ break;
+ case TIFF_TAG_THRESHHOLDING:
+ dissect_tiff_single_uint(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_threshholding);
+ break;
+ case TIFF_TAG_CELL_WIDTH:
+ dissect_tiff_single_uint(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_cell_width);
+ break;
+ case TIFF_TAG_CELL_LENGTH:
+ dissect_tiff_single_uint(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_cell_length);
+ break;
+ case TIFF_TAG_FILL_ORDER:
+ dissect_tiff_single_uint(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_fill_order);
+ break;
+ case TIFF_TAG_DOCUMENT_NAME:
+ dissect_tiff_single_string(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_document_name);
+ break;
+ case TIFF_TAG_IMAGE_DESCRIPTION:
+ dissect_tiff_single_string(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_image_description);
+ break;
+ case TIFF_TAG_MAKE:
+ dissect_tiff_single_string(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_make);
+ break;
+ case TIFF_TAG_MODEL:
+ dissect_tiff_single_string(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_model);
+ break;
+ case TIFF_TAG_STRIP_OFFSETS:
+ dissect_tiff_array_uint(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_strip_offset);
+ break;
+ case TIFF_TAG_ORIENTATION:
+ dissect_tiff_single_uint(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_orientation);
+ break;
+ case TIFF_TAG_SAMPLES_PER_PIXEL:
+ dissect_tiff_single_uint(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_samples_per_pixel);
+ break;
+ case TIFF_TAG_ROWS_PER_STRIP:
+ dissect_tiff_single_uint(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_rows_per_strip);
+ break;
+ case TIFF_TAG_STRIP_BYTE_COUNTS:
+ dissect_tiff_array_uint(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_strip_byte_count);
+ break;
+ case TIFF_TAG_X_RESOLUTION:
+ dissect_tiff_single_urational(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_x_res_numer, hf_tiff_x_res_denom, hf_tiff_x_res_approx);
+ break;
+ case TIFF_TAG_Y_RESOLUTION:
+ dissect_tiff_single_urational(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_y_res_numer, hf_tiff_y_res_denom, hf_tiff_y_res_approx);
+ break;
+ case TIFF_TAG_PLANAR_CONFIGURATION:
+ dissect_tiff_single_uint(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_planar_configuration);
+ break;
+ case TIFF_TAG_PAGE_NAME:
+ dissect_tiff_single_string(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_page_name);
+ break;
+ case TIFF_TAG_GRAY_RESPONSE_UNIT:
+ dissect_tiff_single_uint(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_gray_response_unit);
+ break;
+ case TIFF_TAG_T6_OPTIONS:
+ dissect_tiff_t6_options(tvb, pinfo, entry_tree, offset + 8, type, count, encoding);
+ break;
+ case TIFF_TAG_RESOLUTION_UNIT:
+ dissect_tiff_single_uint(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_resolution_unit);
+ break;
+ case TIFF_TAG_SOFTWARE:
+ dissect_tiff_single_string(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_software);
+ break;
+ case TIFF_TAG_DATE_TIME:
+ dissect_tiff_single_string(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_date_time);
+ break;
+ case TIFF_TAG_ARTIST:
+ dissect_tiff_single_string(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_artist);
+ break;
+ case TIFF_TAG_HOST_COMPUTER:
+ dissect_tiff_single_string(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_host_computer);
+ break;
+ case TIFF_TAG_PREDICTOR:
+ dissect_tiff_single_uint(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_predictor);
+ break;
+ case TIFF_TAG_TILE_WIDTH:
+ dissect_tiff_single_uint(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_tile_width);
+ break;
+ case TIFF_TAG_TILE_LENGTH:
+ dissect_tiff_single_uint(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_tile_length);
+ break;
+ case TIFF_TAG_INK_SET:
+ dissect_tiff_single_uint(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_ink_set);
+ break;
+ case TIFF_TAG_NUMBER_OF_INKS:
+ dissect_tiff_single_uint(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_number_of_inks);
+ break;
+ case TIFF_TAG_TARGET_PRINTER:
+ dissect_tiff_single_string(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_target_printer);
+ break;
+ case TIFF_TAG_COPYRIGHT:
+ dissect_tiff_single_string(tvb, pinfo, entry_tree, offset + 8, type, count, encoding, hf_tiff_copyright);
+ break;
+ default:
+ dissect_tiff_tag_unknown(tvb, pinfo, entry_tree, offset + 8, type, count, encoding);
+ }
+}
+
+// Dissect an IFD with all of its fields, starting at the given offset
+//
+// Return the offset of the next IFD, or 0 if there isn't one
+static guint32
+dissect_tiff_ifd(tvbuff_t *tvb, packet_info *pinfo _U_, proto_tree *tree, guint32 offset, gint encoding) {
+ guint16 ifd_count = tvb_get_guint16(tvb, offset, encoding);
+ gint ifd_length = 2 + (ifd_count * 12) + 4;
+
+ proto_tree *ifd_tree = proto_tree_add_subtree(tree, tvb, offset, ifd_length, ett_ifd, NULL, "Image File Directory");
+
+ proto_tree_add_item(ifd_tree, hf_tiff_ifd_count, tvb, offset, 2, encoding);
+ offset += 2;
+
+ for (gint i = 0; i < ifd_count; i++) {
+ dissect_tiff_entry(tvb, pinfo, ifd_tree, offset, encoding);
+ offset += 12;
+ }
+
+ proto_tree_add_item(ifd_tree, hf_tiff_ifd_next, tvb, offset, 4, encoding);
+ guint32 ifd_next = tvb_get_guint32(tvb, offset, encoding);
+
+ return ifd_next;
+}
+
+static int
+dissect_tiff(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, void *data _U_) {
+ int encoding;
+
+ // Reject if we don't have enough room for the heuristics
+ if (tvb_captured_length(tvb) < 4) {
+ return 0;
+ }
+
+ // Figure out if we're big-endian or little endian
+ guint16 raw_encoding = tvb_get_ntohs(tvb, 0);
+ guint16 magic;
+ guint32 ifd_offset;
+ if (raw_encoding == 0x4949) {
+ encoding = ENC_LITTLE_ENDIAN;
+ } else if (raw_encoding == 0x4D4D) {
+ encoding = ENC_BIG_ENDIAN;
+ } else {
+ // If we don't recognize the endianness, abort with nothing decoded
+ return 0;
+ }
+
+ magic = tvb_get_guint16(tvb, 2, encoding);
+
+ // If the magic number isn't 42, abort with nothing decoded
+ if (magic != 42) {
+ return 0;
+ }
+
+ proto_item *ti = proto_tree_add_item(tree, proto_tiff, tvb, 0, -1, ENC_NA);
+ proto_tree *tiff_tree = proto_item_add_subtree(ti, ett_tiff);
+
+ // Dissect the rest of the header
+ proto_tree_add_item(tiff_tree, hf_tiff_header_endianness, tvb, 0, 2, encoding);
+ proto_tree_add_item(tiff_tree, hf_tiff_header_magic, tvb, 2, 2, encoding);
+ proto_tree_add_item_ret_uint(tiff_tree, hf_tiff_header_lead_ifd, tvb, 4, 4, encoding, &ifd_offset);
+
+ // Keep dissecting IFDs until the offset to the next one is zero
+ while (ifd_offset != 0) {
+ ifd_offset = dissect_tiff_ifd(tvb, pinfo, tiff_tree, ifd_offset, encoding);
+ }
+
+ return tvb_captured_length(tvb);
+}
+
+void
+proto_register_tiff(void)
+{
+ static hf_register_info hf[] = {
+ { &hf_tiff_header_endianness,
+ { "Endianness", "tiff.endianness",
+ FT_UINT16, BASE_HEX, VALS(tiff_endianness_names),
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_header_magic,
+ { "Magic", "tiff.magic",
+ FT_UINT16, BASE_HEX, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_header_lead_ifd,
+ { "Lead IFD Offset", "tiff.lead_ifd",
+ FT_UINT32, BASE_HEX, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_ifd_count,
+ { "Number of Entries", "tiff.ifd_count",
+ FT_UINT16, BASE_DEC, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_ifd_next,
+ { "Next IFD Offset", "tiff.next_ifd",
+ FT_UINT32, BASE_HEX, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_entry_tag,
+ { "Tag", "tiff.tag",
+ FT_UINT16, BASE_DEC, VALS(tiff_tag_names),
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_entry_type,
+ { "Type", "tiff.type",
+ FT_UINT16, BASE_DEC, VALS(tiff_type_names),
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_entry_count,
+ { "Count", "tiff.count",
+ FT_UINT32, BASE_DEC, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_entry_offset,
+ { "Offset", "tiff.offset",
+ FT_UINT32, BASE_DEC, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_entry_unknown,
+ { "Unknown Data", "tiff.unknown",
+ FT_BYTES, BASE_NONE, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_image_width,
+ { "Image Width", "tiff.image_width",
+ FT_UINT32, BASE_DEC, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_image_length,
+ { "Image Length", "tiff.image_length",
+ FT_UINT32, BASE_DEC, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_bits_per_sample,
+ { "Bits per Sample", "tiff.bits_per_sample",
+ FT_UINT16, BASE_DEC, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_compression,
+ { "Compression", "tiff.compression",
+ FT_UINT16, BASE_DEC, VALS(tiff_compression_names),
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_photometric_interp,
+ { "Photometric Interpretation", "tiff.photometric_interp",
+ FT_UINT16, BASE_DEC, VALS(tiff_photometric_interp_names),
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_threshholding,
+ { "Threshholding", "tiff.threshholding",
+ FT_UINT16, BASE_DEC, VALS(tiff_threshholding_names),
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_cell_width,
+ { "Cell Width", "tiff.cell_width",
+ FT_UINT16, BASE_DEC, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_cell_length,
+ { "Cell Length", "tiff.cell_length",
+ FT_UINT16, BASE_DEC, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_fill_order,
+ { "Fill Order", "tiff.fill_order",
+ FT_UINT16, BASE_DEC, VALS(tiff_fill_order_names),
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_document_name,
+ { "Document Name", "tiff.document_name",
+ FT_STRINGZ, BASE_NONE, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_image_description,
+ { "Image Description", "tiff.image_description",
+ FT_STRINGZ, BASE_NONE, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_make,
+ { "Make", "tiff.make",
+ FT_STRINGZ, BASE_NONE, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_model,
+ { "Model", "tiff.model",
+ FT_STRINGZ, BASE_NONE, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_strip_offset,
+ { "Strip Offset", "tiff.strip_offset",
+ FT_UINT32, BASE_DEC, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_orientation,
+ { "Orientation", "tiff.orientation",
+ FT_UINT16, BASE_DEC, VALS(tiff_orientation_names),
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_samples_per_pixel,
+ { "Samples per Pixel", "tiff.samples_per_pixel",
+ FT_UINT16, BASE_DEC, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_rows_per_strip,
+ { "Rows per Strip", "tiff.rows_per_strip",
+ FT_UINT16, BASE_DEC, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_strip_byte_count,
+ { "Strip Byte Count", "tiff.strip_byte_count",
+ FT_UINT32, BASE_DEC, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_x_res_numer,
+ { "X Resolution Numerator", "tiff.x_res_numer",
+ FT_UINT32, BASE_DEC, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_x_res_denom,
+ { "X Resolution Denominator", "tiff.x_res_denom",
+ FT_UINT32, BASE_DEC, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_x_res_approx,
+ { "X Resolution Approximation", "tiff.x_res_approx",
+ FT_DOUBLE, BASE_NONE, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_y_res_numer,
+ { "Y Resolution Numerator", "tiff.y_res_numer",
+ FT_UINT32, BASE_DEC, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_y_res_denom,
+ { "Y Resolution Denominator", "tiff.y_res_denom",
+ FT_UINT32, BASE_DEC, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_y_res_approx,
+ { "Y Resolution Approximation", "tiff.y_res_approx",
+ FT_DOUBLE, BASE_NONE, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_planar_configuration,
+ { "Planar Configuration", "tiff.planar_configuration",
+ FT_UINT16, BASE_DEC, VALS(tiff_planar_configuration_names),
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_page_name,
+ { "Page Name", "tiff.page_name",
+ FT_STRINGZ, BASE_NONE, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_gray_response_unit,
+ { "Gray Response Unit", "tiff.gray_response_unit",
+ FT_UINT16, BASE_DEC, VALS(tiff_gray_response_unit_names),
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_t6_options,
+ { "T6 Options", "tiff.t6",
+ FT_UINT32, BASE_HEX, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_t6_unused,
+ { "Unused", "tiff.t6.unused",
+ FT_UINT32, BASE_HEX, NULL,
+ 0xFFFFFFFD, NULL, HFILL }
+ },
+ { &hf_tiff_t6_allow_uncompresed,
+ { "Allow Uncompressed", "tiff.t6.allow_uncompressed",
+ FT_UINT32, BASE_HEX, VALS(tiff_allow_uncompressed_names),
+ 0x00000002, NULL, HFILL }
+ },
+ { &hf_tiff_resolution_unit,
+ { "Resolution Unit", "tiff.resolution_unit",
+ FT_UINT16, BASE_DEC, VALS(tiff_resolution_unit_names),
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_software,
+ { "Software", "tiff.software",
+ FT_STRINGZ, BASE_NONE, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_date_time,
+ { "Date/Time", "tiff.date_time",
+ FT_STRINGZ, BASE_NONE, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_artist,
+ { "Artist", "tiff.artist",
+ FT_STRINGZ, BASE_NONE, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_host_computer,
+ { "Host Computer", "tiff.host_computer",
+ FT_STRINGZ, BASE_NONE, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_predictor,
+ { "Predictor", "tiff.predictor",
+ FT_UINT16, BASE_DEC, VALS(tiff_predictor_names),
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_tile_width,
+ { "Tile Width", "tiff.tile_width",
+ FT_UINT32, BASE_DEC, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_tile_length,
+ { "Tile Width", "tiff.tile_length",
+ FT_UINT32, BASE_DEC, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_ink_set,
+ { "Ink Set", "tiff.ink_set",
+ FT_UINT16, BASE_DEC, VALS(tiff_ink_set_names),
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_number_of_inks,
+ { "Number of Inks", "tiff.number_of_inks",
+ FT_UINT16, BASE_DEC, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_target_printer,
+ { "Target Printer", "tiff.target_printer",
+ FT_STRINGZ, BASE_NONE, NULL,
+ 0x0, NULL, HFILL }
+ },
+ { &hf_tiff_copyright,
+ { "Copyright", "tiff.copyright",
+ FT_STRINGZ, BASE_NONE, NULL,
+ 0x0, NULL, HFILL }
+ }
+ };
+
+ static gint *ett[] = {
+ &ett_tiff,
+ &ett_ifd,
+ &ett_t6,
+ };
+
+ static ei_register_info ei[] = {
+ { &ei_tiff_unknown_tag,
+ { "tiff.unknown_tag", PI_UNDECODED, PI_NOTE,
+ "Unknown tag", EXPFILL }
+ },
+ { &ei_tiff_bad_entry,
+ { "tiff.bad_entry", PI_PROTOCOL, PI_WARN,
+ "Invalid entry contents", EXPFILL }
+ },
+ { &ei_tiff_zero_denom,
+ { "tiff.zero_denom", PI_PROTOCOL, PI_WARN,
+ "Zero denominator", EXPFILL }
+ },
+
+ };
+
+ proto_tiff = proto_register_protocol("Tagged Image File Format", "TIFF image", "tiff");
+
+ register_dissector("tiff", dissect_tiff, proto_tiff);
+ proto_register_field_array(proto_tiff, hf, array_length(hf));
+ proto_register_subtree_array(ett, array_length(ett));
+
+ expert_module_t *expert_tiff = expert_register_protocol(proto_tiff);
+ expert_register_field_array(expert_tiff, ei, array_length(ei));
+}
+
+static gboolean
+dissect_tiff_heur(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, void *data _U_)
+{
+ return dissect_tiff(tvb, pinfo, tree, NULL) > 0;
+}
+
+void
+proto_reg_handoff_tiff(void)
+{
+ dissector_handle_t tiff_handle = find_dissector("tiff");
+
+ // Register the TIFF media type
+ dissector_add_string("media_type", "image/tiff", tiff_handle);
+
+ // Register the TIFF heuristic dissector
+ heur_dissector_add("wtap_file", dissect_tiff_heur, "TIFF file", "tiff_wtap", proto_tiff, HEURISTIC_ENABLE);
+}