diff options
Diffstat (limited to 'tools/process-x11-xcb.pl')
-rwxr-xr-x | tools/process-x11-xcb.pl | 1946 |
1 files changed, 1946 insertions, 0 deletions
diff --git a/tools/process-x11-xcb.pl b/tools/process-x11-xcb.pl new file mode 100755 index 00000000..91dcf427 --- /dev/null +++ b/tools/process-x11-xcb.pl @@ -0,0 +1,1946 @@ +#!/usr/bin/perl +# +# Script to convert xcbproto and mesa protocol files for +# X11 dissector. Creates header files containing code to +# dissect X11 extensions. +# +# Instructions for using this script are in epan/dissectors/README.X11 +# +# Copyright 2008, 2009, 2013, 2014 Open Text Corporation <pharris[AT]opentext.com> +# +# Wireshark - Network traffic analyzer +# By Gerald Combs <gerald@wireshark.org> +# Copyright 1998 Gerald Combs +# +# SPDX-License-Identifier: GPL-2.0-or-later +# + +#TODO +# - support constructs that are legal in XCB, but don't appear to be used + +use 5.010; + +use warnings; +use strict; + +# given/when is going to be removed (and/or dramatically altered) +# in 5.20. Patches welcome. +# Patches even more welcome if they rewrite this whole thing in a +# language with a proper compatibility document, such as +# http://golang.org/doc/go1compat +no if $] >= 5.018, warnings => "experimental::smartmatch"; + +use IO::File; +use XML::Twig; + +use File::Spec; + +my $srcdir = shift; +die "'$srcdir' is not a directory" unless -d $srcdir; + +my @reslist = grep {!/xproto\.xml$/} glob File::Spec->catfile($srcdir, 'xcbproto', 'src', '*.xml'); +my @register; + +my $script_name = File::Spec->abs2rel ($0, $srcdir); + +my %basictype = ( + char => { size => 1, encoding => 'ENC_ASCII|ENC_NA', type => 'FT_STRING', base => 'BASE_NONE', get => 'tvb_get_guint8', list => 'listOfByte', }, + void => { size => 1, encoding => 'ENC_NA', type => 'FT_BYTES', base => 'BASE_NONE', get => 'tvb_get_guint8', list => 'listOfByte', }, + BYTE => { size => 1, encoding => 'ENC_NA', type => 'FT_BYTES', base => 'BASE_NONE', get => 'tvb_get_guint8', list => 'listOfByte', }, + CARD8 => { size => 1, encoding => 'byte_order', type => 'FT_UINT8', base => 'BASE_HEX_DEC', get => 'tvb_get_guint8', list => 'listOfByte', }, + CARD16 => { size => 2, encoding => 'byte_order', type => 'FT_UINT16', base => 'BASE_HEX_DEC', get => 'tvb_get_guint16', list => 'listOfCard16', }, + CARD32 => { size => 4, encoding => 'byte_order', type => 'FT_UINT32', base => 'BASE_HEX_DEC', get => 'tvb_get_guint32', list => 'listOfCard32', }, + CARD64 => { size => 8, encoding => 'byte_order', type => 'FT_UINT64', base => 'BASE_HEX_DEC', get => 'tvb_get_guint64', list => 'listOfCard64', }, + INT8 => { size => 1, encoding => 'byte_order', type => 'FT_INT8', base => 'BASE_DEC', get => 'tvb_get_guint8', list => 'listOfByte', }, + INT16 => { size => 2, encoding => 'byte_order', type => 'FT_INT16', base => 'BASE_DEC', get => 'tvb_get_guint16', list => 'listOfInt16', }, + INT32 => { size => 4, encoding => 'byte_order', type => 'FT_INT32', base => 'BASE_DEC', get => 'tvb_get_guint32', list => 'listOfInt32', }, + INT64 => { size => 8, encoding => 'byte_order', type => 'FT_INT64', base => 'BASE_DEC', get => 'tvb_get_guint64', list => 'listOfInt64', }, + float => { size => 4, encoding => 'byte_order', type => 'FT_FLOAT', base => 'BASE_NONE', get => 'tvb_get_ieee_float', list => 'listOfFloat', }, + double => { size => 8, encoding => 'byte_order', type => 'FT_DOUBLE', base => 'BASE_NONE', get => 'tvb_get_ieee_double', list => 'listOfDouble', }, + BOOL => { size => 1, encoding => 'byte_order', type => 'FT_BOOLEAN',base => 'BASE_NONE', get => 'tvb_get_guint8', list => 'listOfByte', }, +); + +my %simpletype; # Reset at the beginning of each extension +my %gltype; # No need to reset, since it's only used once + +my %struct = # Not reset; contains structures already defined. + # Also contains this black-list of structures never used by any + # extension (to avoid generating useless code). +( + # structures defined by xproto, but not used by any extension + 'xproto:CHAR2B' => 1, + 'xproto:ARC' => 1, + 'xproto:FORMAT' => 1, + 'xproto:VISUALTYPE' => 1, + 'xproto:DEPTH' => 1, + 'xproto:SCREEN' => 1, + 'xproto:SetupRequest' => 1, + 'xproto:SetupFailed' => 1, + 'xproto:SetupAuthenticate' => 1, + 'xproto:Setup' => 1, + 'xproto:TIMECOORD' => 1, + 'xproto:FONTPROP' => 1, + 'xproto:CHARINFO' => 1, + 'xproto:SEGMENT' => 1, + 'xproto:COLORITEM' => 1, + 'xproto:RGB' => 1, + 'xproto:HOST' => 1, + 'xproto:POINT' => 1, + + # structures defined by xinput, but never used (except by each other)(bug in xcb?) + 'xinput:KeyInfo' => 1, + 'xinput:ButtonInfo' => 1, + 'xinput:ValuatorInfo' => 1, + 'xinput:KbdFeedbackState' => 1, + 'xinput:PtrFeedbackState' => 1, + 'xinput:IntegerFeedbackState' => 1, + 'xinput:StringFeedbackState' => 1, + 'xinput:BellFeedbackState' => 1, + 'xinput:LedFeedbackState' => 1, + 'xinput:KbdFeedbackCtl' => 1, + 'xinput:PtrFeedbackCtl' => 1, + 'xinput:IntegerFeedbackCtl' => 1, + 'xinput:StringFeedbackCtl' => 1, + 'xinput:BellFeedbackCtl' => 1, + 'xinput:LedFeedbackCtl' => 1, + 'xinput:KeyState' => 1, + 'xinput:ButtonState' => 1, + 'xinput:ValuatorState' => 1, + 'xinput:DeviceResolutionState' => 1, + 'xinput:DeviceAbsCalibState' => 1, + 'xinput:DeviceAbsAreaState' => 1, + 'xinput:DeviceCoreState' => 1, + 'xinput:DeviceEnableState' => 1, + 'xinput:DeviceResolutionCtl' => 1, + 'xinput:DeviceAbsCalibCtl' => 1, + 'xinput:DeviceAbsAreaCtrl' => 1, + 'xinput:DeviceCoreCtrl' => 1, + 'xinput:DeviceEnableCtrl' => 1, + 'xinput:DeviceName' => 1, + 'xinput:AddMaster' => 1, + 'xinput:RemoveMaster' => 1, + 'xinput:AttachSlave' => 1, + 'xinput:DetachSlave' => 1, + 'xinput:ButtonClass' => 1, + 'xinput:KeyClass' => 1, + 'xinput:ScrollClass' => 1, + 'xinput:TouchClass' => 1, + 'xinput:ValuatorClass' => 1, + + # structures defined by xv, but never used (bug in xcb?) + 'xv:Image' => 1, + + # structures defined by xkb, but never used (except by each other)(bug in xcb?) + 'xkb:Key' => 1, + 'xkb:Outline' => 1, + 'xkb:Overlay' => 1, + 'xkb:OverlayKey' => 1, + 'xkb:OverlayRow' => 1, + 'xkb:Row' => 1, + 'xkb:Shape' => 1, +); +my %enum; # Not reset; contains enums already defined. +my %enum_name; +my %type_name; +my $header; +my $extname; +my @incname; +my %request; +my %genericevent; +my %event; +my %reply; + +# Output files +my $impl; +my $reg; +my $decl; +my $error; + +# glRender sub-op output files +my $enum; + +# Mesa API definitions keep moving +my @mesas = ($srcdir . '/mesa/src/mapi/glapi/gen', # 2010-04-26 + $srcdir . '/mesa/src/mesa/glapi/gen', # 2010-02-22 + $srcdir . '/mesa/src/mesa/glapi'); # 2004-05-18 +my $mesadir = (grep { -d } @mesas)[0]; + +sub mesa_category { + my ($t, $elt) = @_; + $t->purge; +} + +#used to prevent duplication and sort enumerated values +my %mesa_enum_hash = (); + +sub mesa_enum { + my ($t, $elt) = @_; + my $name = $elt->att('name'); + my $value = $elt->att('value'); + my $hex_value = hex($value); #convert string to hex value to catch leading zeros + + #make sure value isn't already in the hash, to prevent duplication in value_string + if (!exists($mesa_enum_hash{$hex_value})) { + $mesa_enum_hash{$hex_value} = $name; + } + $t->purge; +} + +sub mesa_type { + my ($t, $elt) = @_; + + my $name = $elt->att('name'); + my $size = $elt->att('size'); + my $float = $elt->att('float'); + my $unsigned = $elt->att('unsigned'); + my $base; + + $t->purge; + + if($name eq 'enum') { + # enum does not have a direct X equivalent + $gltype{'GLenum'} = { size => 4, encoding => 'byte_order', type => 'FT_UINT32', base => 'BASE_HEX|BASE_EXT_STRING', + get => 'tvb_get_guint32', list => 'listOfCard32', + val => '&mesa_enum_ext', }; + return; + } + + $name = 'GL'.$name; + if (defined($float) && $float eq 'true') { + $base = 'float'; + $base = 'double' if ($size == 8); + } else { + $base = 'INT'; + if (defined($unsigned) && $unsigned eq 'true') { + $base = 'CARD'; + } + $base .= ($size * 8); + + $base = 'BOOL' if ($name eq 'bool'); + $base = 'BYTE' if ($name eq 'void'); + } + + $gltype{$name} = $basictype{$base}; +} + +sub registered_name($$) +{ + my $name = shift; + my $field = shift; + + return "hf_x11_$header"."_$name"."_$field"; +} + +sub mesa_function { + my ($t, $elt) = @_; + # rop == glRender sub-op + # sop == GLX minor opcode + my $glx = $elt->first_child('glx'); + unless(defined $glx) { $t->purge; return; } + + my $rop = $glx->att('rop'); + unless (defined $rop) { $t->purge; return; } + + # Ideally, we want the main name, not the alias name. + # Practically, we'd have to scan the file twice to find + # the functions that we want to skip. + my $alias = $elt->att('alias'); + if (defined $alias) { $t->purge; return; } + + my $name = $elt->att('name'); + $request{$rop} = $name; + + my $image; + + my $length = 0; + my @elements = $elt->children('param'); + + # Wireshark defines _U_ to mean "Unused" (compiler specific define) + if (!@elements) { + print $impl <<eot +static void mesa_$name(tvbuff_t *tvb _U_, int *offsetp _U_, proto_tree *t _U_, guint byte_order _U_, int length _U_) +{ +eot +; + } else { + print $impl <<eot +static void mesa_$name(tvbuff_t *tvb, int *offsetp, proto_tree *t, guint byte_order, int length _U_) +{ +eot +; + } + + my %type_param; + foreach my $e (@elements) { + # Detect count && variable_param + my $count = $e->att('count'); + my $variable_param = $e->att('variable_param'); + if (defined $count and defined $variable_param) { + $type_param{$variable_param} = 1; + } + } + foreach my $e (@elements) { + # Register field with wireshark + + my $type = $e->att('type'); + $type =~ s/^const //; + my $list; + $list = 1 if ($type =~ /\*$/); + $type =~ s/ \*$//; + + my $fieldname = $e->att('name'); + my $regname = registered_name($name, $fieldname); + + my $info = $gltype{$type}; + my $ft = $info->{'type'}; + my $base = $info->{'base'}; + my $val = $info->{'val'} // 'NULL'; + my $count = $e->att('count'); + my $variable_param = $e->att('variable_param'); + + if ($list and $count and $variable_param) { + print $decl "static int ${regname} = -1;\n"; + print $reg "{ &$regname, { \"$fieldname\", \"x11.glx.render.$name.$fieldname\", FT_NONE, BASE_NONE, NULL, 0, NULL, HFILL }},\n"; + print $decl "static int ${regname}_signed = -1;\n"; + print $reg "{ &${regname}_signed, { \"$fieldname\", \"x11.glx.render.$name.$fieldname\", FT_INT8, BASE_DEC, NULL, 0, NULL, HFILL }},\n"; + print $decl "static int ${regname}_unsigned = -1;\n"; + print $reg "{ &${regname}_unsigned, { \"$fieldname\", \"x11.glx.render.$name.$fieldname\", FT_UINT8, BASE_DEC, NULL, 0, NULL, HFILL }},\n"; + print $decl "static int ${regname}_item_card16 = -1;\n"; + print $reg "{ &${regname}_item_card16, { \"$fieldname\", \"x11.glx.render.$name.$fieldname\", FT_UINT16, BASE_DEC, NULL, 0, NULL, HFILL }},\n"; + print $decl "static int ${regname}_item_int16 = -1;\n"; + print $reg "{ &${regname}_item_int16, { \"$fieldname\", \"x11.glx.render.$name.$fieldname\", FT_INT16, BASE_DEC, NULL, 0, NULL, HFILL }},\n"; + print $decl "static int ${regname}_item_card32 = -1;\n"; + print $reg "{ &${regname}_item_card32, { \"$fieldname\", \"x11.glx.render.$name.$fieldname\", FT_UINT32, BASE_DEC, NULL, 0, NULL, HFILL }},\n"; + print $decl "static int ${regname}_item_int32 = -1;\n"; + print $reg "{ &${regname}_item_int32, { \"$fieldname\", \"x11.glx.render.$name.$fieldname\", FT_INT32, BASE_DEC, NULL, 0, NULL, HFILL }},\n"; + print $decl "static int ${regname}_item_float = -1;\n"; + print $reg "{ &${regname}_item_float, { \"$fieldname\", \"x11.glx.render.$name.$fieldname\", FT_FLOAT, BASE_NONE, NULL, 0, NULL, HFILL }},\n"; + } else { + print $decl "static int $regname = -1;\n"; + if ($list and $info->{'size'} > 1) { + print $reg "{ &$regname, { \"$fieldname\", \"x11.glx.render.$name.$fieldname.list\", FT_NONE, BASE_NONE, NULL, 0, NULL, HFILL }},\n"; + $regname .= '_item'; + print $decl "static int $regname = -1;\n"; + } + print $reg "{ &$regname, { \"$fieldname\", \"x11.glx.render.$name.$fieldname\", $ft, $base, $val, 0, NULL, HFILL }},\n"; + + if ($e->att('counter') or $type_param{$fieldname}) { + print $impl " int $fieldname;\n"; + } + } + + if ($list) { + if ($e->att('img_format')) { + $image = 1; + foreach my $wholename (('swap bytes', 'lsb first')) { + # Boolean values + my $varname = $wholename; + $varname =~ s/\s//g; + my $regname = registered_name($name, $varname); + print $decl "static int $regname = -1;\n"; + print $reg "{ &$regname, { \"$wholename\", \"x11.glx.render.$name.$varname\", FT_BOOLEAN, BASE_NONE, NULL, 0, NULL, HFILL }},\n"; + } + foreach my $wholename (('row length', 'skip rows', 'skip pixels', 'alignment')) { + # Integer values + my $varname = $wholename; + $varname =~ s/\s//g; + my $regname = registered_name($name, $varname); + print $decl "static int $regname = -1;\n"; + print $reg "{ &$regname, { \"$wholename\", \"x11.glx.render.$name.$varname\", FT_UINT32, BASE_HEX_DEC, NULL, 0, NULL, HFILL }},\n"; + } + } + } + } + + # The image requests have a few implicit elements first: + if ($image) { + foreach my $wholename (('swap bytes', 'lsb first')) { + # Boolean values + my $varname = $wholename; + $varname =~ s/\s//g; + my $regname = registered_name($name, $varname); + print $impl " proto_tree_add_item(t, $regname, tvb, *offsetp, 1, byte_order);\n"; + print $impl " *offsetp += 1;\n"; + $length += 1; + } + print $impl " proto_tree_add_item(t, hf_x11_unused, tvb, *offsetp, 2, ENC_NA);\n"; + print $impl " *offsetp += 2;\n"; + $length += 2; + foreach my $wholename (('row length', 'skip rows', 'skip pixels', 'alignment')) { + # Integer values + my $varname = $wholename; + $varname =~ s/\s//g; + my $regname = registered_name($name, $varname); + print $impl " proto_tree_add_item(t, $regname, tvb, *offsetp, 4, byte_order);\n"; + print $impl " *offsetp += 4;\n"; + $length += 4; + } + } + + foreach my $e (@elements) { + my $type = $e->att('type'); + $type =~ s/^const //; + my $list; + $list = 1 if ($type =~ /\*$/); + $type =~ s/ \*$//; + + my $fieldname = $e->att('name'); + my $regname = registered_name($name, $fieldname); + + my $info = $gltype{$type}; + my $ft = $info->{'type'}; + my $base = $info->{'base'}; + + if (!$list) { + my $size = $info->{'size'}; + my $encoding = $info->{'encoding'}; + my $get = $info->{'get'}; + + if ($e->att('counter') or $type_param{$fieldname}) { + if ($get ne "tvb_get_guint8") { + print $impl " $fieldname = $get(tvb, *offsetp, $encoding);\n"; + } else { + print $impl " $fieldname = $get(tvb, *offsetp);\n"; + } + } + print $impl " proto_tree_add_item(t, $regname, tvb, *offsetp, $size, $encoding);\n"; + print $impl " *offsetp += $size;\n"; + $length += $size; + } else { # list + my $list = $info->{'list'}; + my $count = $e->att('count'); + my $variable_param = $e->att('variable_param'); + + if (defined($count) && !defined($variable_param)) { + $regname .= ", $regname".'_item' if ($info->{'size'} > 1); + print $impl " $list(tvb, offsetp, t, $regname, $count, byte_order);\n"; + } else { + if (defined($count)) { + # Currently, only CallLists has both a count and a variable_param + # The XML contains a size description of all the possibilities + # for CallLists, but not a type description. Implement by hand, + # with the caveat that more types may need to be added in the + # future. + say $impl " switch($variable_param) {"; + say $impl " case 0x1400: /* BYTE */"; + say $impl " listOfByte(tvb, offsetp, t, ${regname}_signed, $count, byte_order);"; + say $impl " proto_tree_add_item(t, hf_x11_unused, tvb, *offsetp, (length - $length - $count), ENC_NA);"; + say $impl " *offsetp += (length - $length - $count);"; + say $impl " break;"; + say $impl " case 0x1401: /* UNSIGNED_BYTE */"; + say $impl " listOfByte(tvb, offsetp, t, ${regname}_unsigned, $count, byte_order);"; + say $impl " proto_tree_add_item(t, hf_x11_unused, tvb, *offsetp, (length - $length - $count), ENC_NA);"; + say $impl " *offsetp += (length - $length - $count);"; + say $impl " break;"; + say $impl " case 0x1402: /* SHORT */"; + say $impl " listOfInt16(tvb, offsetp, t, $regname, ${regname}_item_int16, $count, byte_order);"; + say $impl " proto_tree_add_item(t, hf_x11_unused, tvb, *offsetp, (length - $length - 2 * $count), ENC_NA);"; + say $impl " *offsetp += (length - $length - 2 * $count);"; + say $impl " break;"; + say $impl " case 0x1403: /* UNSIGNED_SHORT */"; + say $impl " listOfCard16(tvb, offsetp, t, $regname, ${regname}_item_card16, $count, byte_order);"; + say $impl " proto_tree_add_item(t, hf_x11_unused, tvb, *offsetp, (length - $length - 2 * $count), ENC_NA);"; + say $impl " *offsetp += (length - $length - 2 * $count);"; + say $impl " break;"; + say $impl " case 0x1404: /* INT */"; + say $impl " listOfInt32(tvb, offsetp, t, $regname, ${regname}_item_int32, $count, byte_order);"; + say $impl " break;"; + say $impl " case 0x1405: /* UNSIGNED_INT */"; + say $impl " listOfCard32(tvb, offsetp, t, $regname, ${regname}_item_card32, $count, byte_order);"; + say $impl " break;"; + say $impl " case 0x1406: /* FLOAT */"; + say $impl " listOfFloat(tvb, offsetp, t, $regname, ${regname}_item_float, $count, byte_order);"; + say $impl " break;"; + say $impl " case 0x1407: /* 2_BYTES */"; + say $impl " listOfCard16(tvb, offsetp, t, $regname, ${regname}_item_card16, $count, ENC_BIG_ENDIAN);"; + say $impl " proto_tree_add_item(t, hf_x11_unused, tvb, *offsetp, (length - $length - 2 * $count), ENC_NA);"; + say $impl " *offsetp += (length - $length - 2 * $count);"; + say $impl " break;"; + say $impl " case 0x1408: /* 3_BYTES */"; + say $impl " UNDECODED(3 * $count);"; + say $impl " proto_tree_add_item(t, hf_x11_unused, tvb, *offsetp, (length - $length - 3 * $count), ENC_NA);"; + say $impl " *offsetp += (length - $length - 3 * $count);"; + say $impl " break;"; + say $impl " case 0x1409: /* 4_BYTES */"; + say $impl " listOfCard32(tvb, offsetp, t, $regname, ${regname}_item_card32, $count, ENC_BIG_ENDIAN);"; + say $impl " break;"; + say $impl " case 0x140B: /* HALF_FLOAT */"; + say $impl " UNDECODED(2 * $count);"; + say $impl " proto_tree_add_item(t, hf_x11_unused, tvb, *offsetp, (length - $length - 2 * $count), ENC_NA);"; + say $impl " *offsetp += (length - $length - 2 * $count);"; + say $impl " break;"; + say $impl " default: /* Unknown */"; + say $impl " UNDECODED(length - $length);"; + say $impl " break;"; + say $impl " }"; + } else { + $regname .= ", $regname".'_item' if ($info->{'size'} > 1); + print $impl " $list(tvb, offsetp, t, $regname, (length - $length) / $gltype{$type}{'size'}, byte_order);\n"; + } + } + } + } + + print $impl "}\n\n"; + $t->purge; +} + +sub get_op($;$); +sub get_unop($;$); + +sub get_ref($$) +{ + my $elt = shift; + my $refref = shift; + my $rv; + + given($elt->name()) { + when ('fieldref') { + $rv = $elt->text(); + $refref->{$rv} = 1; + $rv = 'f_'.$rv; + } + when ('value') { $rv = $elt->text(); } + when ('op') { $rv = get_op($elt, $refref); } + when (['unop','popcount']) { $rv = get_unop($elt, $refref); } + default { die "Invalid op fragment: $_" } + } + return $rv; +} + +sub get_op($;$) { + my $op = shift; + my $refref = shift // {}; + + my @elements = $op->children(qr/fieldref|value|op|unop|popcount/); + (@elements == 2) or die ("Wrong number of children for 'op'\n"); + my $left; + my $right; + + $left = get_ref($elements[0], $refref); + $right = get_ref($elements[1], $refref); + + return "($left " . $op->att('op') . " $right)"; +} + +sub get_unop($;$) { + my $op = shift; + my $refref = shift // {}; + + my @elements = $op->children(qr/fieldref|value|op|unop|popcount/); + (@elements == 1) or die ("Wrong number of children for 'unop'\n"); + my $left; + + $left = get_ref($elements[0], $refref); + + given ($op->name()) { + when ('unop') { + return '(' . $op->att('op') . "$left)"; + } + when ('popcount') { + return "ws_count_ones($left)"; + } + default { die "Invalid unop element $op->name()\n"; } + } +} + +sub qualname { + my $name = shift; + $name = $incname[0].':'.$name unless $name =~ /:/; + return $name +} + +sub get_simple_info { + my $name = shift; + my $info = $basictype{$name}; + return $info if (defined $info); + $info = $simpletype{$name}; + return $info if (defined $info); + if (defined($type_name{$name})) { + return $simpletype{$type_name{$name}}; + } + return undef +} + +sub get_struct_info { + my $name = shift; + my $info = $struct{$name}; + return $info if (defined $info); + if (defined($type_name{$name})) { + return $struct{$type_name{$name}}; + } + return undef +} + +sub getinfo { + my $name = shift; + my $info = get_simple_info($name) // get_struct_info($name); + # If the script fails here search for $name in this script and remove it from the black list + die "$name is defined to be unused in process-x11-xcb.pl but is actually used!" if (defined($info) && $info == "1"); + return $info; +} + +sub dump_enum_values($) +{ + my $e = shift; + + defined($enum{$e}) or die("Enum $e not found"); + + my $enumname = "x11_enum_$e"; + return $enumname if (defined $enum{$e}{done}); + + say $enum 'static const value_string '.$enumname.'[] = {'; + + my $value = $enum{$e}{value}; + for my $val (sort { $a <=> $b } keys %$value) { + say $enum sprintf(" { %3d, \"%s\" },", $val, $$value{$val}); + } + say $enum sprintf(" { %3d, NULL },", 0); + say $enum '};'; + say $enum ''; + + $enum{$e}{done} = 1; + return $enumname; +} + +# Find all references, so we can declare only the minimum necessary +sub reference_elements($$); + +sub reference_elements($$) +{ + my $e = shift; + my $refref = shift; + + given ($e->name()) { + when ('switch') { + my $lentype = $e->first_child(); + if (defined $lentype) { + given ($lentype->name()) { + when ('fieldref') { $refref->{field}{$lentype->text()} = 1; } + when ('op') { get_op($lentype, $refref->{field}); } + } + } + + my @elements = $e->children(qr/(bit)?case/); + for my $case (@elements) { + my @sub_elements = $case->children(qr/list|switch/); + + foreach my $sub_e (@sub_elements) { + reference_elements($sub_e, $refref); + } + } + } + when ('list') { + my $type = $e->att('type'); + my $info = getinfo($type); + if (defined $info->{paramref}) { + for my $pref (keys %{$info->{paramref}}) { + $refref->{field}{$pref} = 1; + } + } + + my $lentype = $e->first_child(); + if (defined $lentype) { + given ($lentype->name()) { + when ('fieldref') { $refref->{field}{$lentype->text()} = 1; } + when ('op') { get_op($lentype, $refref->{field}); } + when (['unop','popcount']) { get_unop($lentype, $refref->{field}); } + when ('sumof') { $refref->{sumof}{$lentype->att('ref')} = 1; } + } + } else { + $refref->{field}{'length'} = 1; + $refref->{'length'} = 1; + } + } + } +} + +sub register_element($$$$;$) +{ + my $e = shift; + my $varpat = shift; + my $humanpat = shift; + my $refref = shift; + my $indent = shift // ' ' x 4; + + given ($e->name()) { + when ('pad') { return; } # Pad has no variables + when ('switch') { return; } # Switch defines varaibles in a tighter scope to avoid collisions + } + + # Register field with wireshark + + my $fieldname = $e->att('name'); + my $type = $e->att('type') or die ("Field $fieldname does not have a valid type\n"); + + my $regname = 'hf_x11_'.sprintf ($varpat, $fieldname); + my $humanname = 'x11.'.sprintf ($humanpat, $fieldname); + + my $info = getinfo($type); + my $ft = $info->{'type'} // 'FT_NONE'; + my $base = $info->{'base'} // 'BASE_NONE'; + my $vals = 'NULL'; + + my $enum = $e->att('enum') // $e->att('altenum'); + if (defined $enum) { + my $enumname = dump_enum_values($enum_name{$enum}); + $vals = "VALS($enumname)"; + + # Wireshark does not allow FT_BYTES, FT_BOOLEAN, or BASE_NONE to have an enum + $ft =~ s/FT_BYTES/FT_UINT8/; + $ft =~ s/FT_BOOLEAN/FT_UINT8/; + $base =~ s/BASE_NONE/BASE_DEC/; + } + + $enum = $e->att('mask'); + if (defined $enum) { + # Create subtree items: + defined($enum{$enum_name{$enum}}) or die("Enum $enum not found"); + + # Wireshark does not allow FT_BYTES or BASE_NONE to have an enum + $ft =~ s/FT_BYTES/FT_UINT8/; + $base =~ s/BASE_NONE/BASE_DEC/; + + my $bitsize = $info->{'size'} * 8; + + my $bit = $enum{$enum_name{$enum}}{bit}; + for my $val (sort { $a <=> $b } keys %$bit) { + my $itemname = $$bit{$val}; + my $item = $regname . '_mask_' . $itemname; + my $itemhuman = $humanname . '.' . $itemname; + my $bitshift = "1U << $val"; + + say $decl "static int $item = -1;"; + say $reg "{ &$item, { \"$itemname\", \"$itemhuman\", FT_BOOLEAN, $bitsize, NULL, $bitshift, NULL, HFILL }},"; + } + } + + print $decl "static int $regname = -1;\n"; + if ($e->name() eq 'list' and defined $info->{'size'} and $info->{'size'} > 1) { + print $reg "{ &$regname, { \"$fieldname\", \"$humanname.list\", FT_NONE, BASE_NONE, NULL, 0, NULL, HFILL }},\n"; + $regname .= '_item'; + print $decl "static int $regname = -1;\n"; + } + print $reg "{ &$regname, { \"$fieldname\", \"$humanname\", $ft, $base, $vals, 0, NULL, HFILL }},\n"; + + if ($refref->{sumof}{$fieldname}) { + print $impl $indent."int sumof_$fieldname = 0;\n"; + } + + if ($e->name() eq 'field') { + if ($refref->{field}{$fieldname} and get_simple_info($type)) { + # Pre-declare variable + if ($ft eq 'FT_FLOAT') { + print $impl $indent."gfloat f_$fieldname;\n"; + } elsif ($ft eq 'FT_DOUBLE') { + print $impl $indent."gdouble f_$fieldname;\n"; + } elsif ($ft eq 'FT_INT64' or $ft eq 'FT_UINT64') { + print $impl $indent."gint64 f_$fieldname;\n"; + } else { + print $impl $indent."int f_$fieldname;\n"; + } + } + } +} + +sub dissect_element($$$$$;$$); + +sub dissect_element($$$$$;$$) +{ + my $e = shift; + my $varpat = shift; + my $humanpat = shift; + my $length = shift; + my $refref = shift; + my $adjustlength = shift; + my $indent = shift // ' ' x 4; + + given ($e->name()) { + when ('pad') { + my $bytes = $e->att('bytes'); + my $align = $e->att('align'); + if (defined $bytes) { + print $impl $indent."proto_tree_add_item(t, hf_x11_unused, tvb, *offsetp, $bytes, ENC_NA);\n"; + print $impl $indent."*offsetp += $bytes;\n"; + $length += $bytes; + } else { + say $impl $indent.'if (*offsetp % '.$align.') {'; + say $impl $indent." proto_tree_add_item(t, hf_x11_unused, tvb, *offsetp, ($align - *offsetp % $align), ENC_NA);"; + say $impl $indent." *offsetp += ($align - *offsetp % $align);"; + say $impl $indent."}"; + if ($length % $align != 0) { + $length += $align - $length % $align; + } + if ($adjustlength) { + say $impl $indent.'length = ((length + '.($align-1).') & ~'.($align-1).');'; + } + } + } + when ('field') { + my $fieldname = $e->att('name'); + my $regname = 'hf_x11_'.sprintf ($varpat, $fieldname); + my $type = $e->att('type'); + + if (get_simple_info($type)) { + my $info = get_simple_info($type); + my $size = $info->{'size'}; + my $encoding = $info->{'encoding'}; + my $get = $info->{'get'}; + + if ($e->att('enum') // $e->att('altenum')) { + my $fieldsize = $size * 8; + print $impl $indent; + if ($refref->{field}{$fieldname}) { + print $impl "f_$fieldname = "; + } + say $impl "field$fieldsize(tvb, offsetp, t, $regname, byte_order);"; + } elsif ($e->att('mask')) { + if ($refref->{field}{$fieldname}) { + if ($get ne "tvb_get_guint8") { + say $impl $indent."f_$fieldname = $get(tvb, *offsetp, byte_order);"; + } else { + say $impl $indent."f_$fieldname = $get(tvb, *offsetp);"; + } + } + my $bitmask_field = $fieldname . "_bits"; + say $impl $indent."{"; + say $impl $indent." int* const $bitmask_field [] = {"; + my $bit = $enum{$enum_name{$e->att('mask')}}{bit}; + for my $val (sort { $a <=> $b } keys %$bit) { + my $item = $regname . '_mask_' . $$bit{$val}; + say $impl "$indent$indent&$item,"; + } + say $impl "$indent$indent" . "NULL"; + say $impl $indent." };"; + + say $impl $indent." proto_tree_add_bitmask(t, tvb, *offsetp, $regname, ett_x11_rectangle, $bitmask_field, $encoding);"; + say $impl $indent."}"; + say $impl $indent."*offsetp += $size;"; + } else { + if ($refref->{field}{$fieldname}) { + if ($get ne "tvb_get_guint8") { + say $impl $indent."f_$fieldname = $get(tvb, *offsetp, byte_order);"; + } else { + say $impl $indent."f_$fieldname = $get(tvb, *offsetp);"; + } + } + print $impl $indent."proto_tree_add_item(t, $regname, tvb, *offsetp, $size, $encoding);\n"; + print $impl $indent."*offsetp += $size;\n"; + } + $length += $size; + } elsif (get_struct_info($type)) { + # TODO: variable-lengths (when $info->{'size'} == 0 ) + my $info = get_struct_info($type); + $length += $info->{'size'}; + print $impl $indent."struct_$info->{'name'}(tvb, offsetp, t, byte_order, 1);\n"; + } else { + die ("Unrecognized type: $type\n"); + } + } + when ('list') { + my $fieldname = $e->att('name'); + my $regname = 'hf_x11_'.sprintf ($varpat, $fieldname); + my $type = $e->att('type'); + + my $info = getinfo($type); + my $lencalc; + my $lentype = $e->first_child(); + if (defined $info->{'size'}) { + $lencalc = "(length - $length) / $info->{'size'}"; + } else { + $lencalc = "(length - $length)"; + } + if (defined $lentype) { + given ($lentype->name()) { + when ('value') { $lencalc = $lentype->text(); } + when ('fieldref') { $lencalc = 'f_'.$lentype->text(); } + when ('paramref') { $lencalc = 'p_'.$lentype->text(); } + when ('op') { $lencalc = get_op($lentype); } + when (['unop','popcount']) { $lencalc = get_unop($lentype); } + when ('sumof') { $lencalc = 'sumof_'.$lentype->att('ref'); } + } + } + + if (get_simple_info($type)) { + my $list = $info->{'list'}; + my $size = $info->{'size'}; + $regname .= ", $regname".'_item' if ($size > 1); + + if ($refref->{sumof}{$fieldname}) { + my $get = $info->{'get'}; + say $impl $indent."{"; + say $impl $indent." int i;"; + say $impl $indent." for (i = 0; i < $lencalc; i++) {"; + if ($get ne "tvb_get_guint8") { + say $impl $indent." sumof_$fieldname += $get(tvb, *offsetp + i * $size, byte_order);"; + } else { + say $impl $indent." sumof_$fieldname += $get(tvb, *offsetp + i * $size);"; + } + say $impl $indent." }"; + say $impl $indent."}"; + } + + print $impl $indent."$list(tvb, offsetp, t, $regname, $lencalc, byte_order);\n"; + } elsif (get_struct_info($type)) { + my $si = get_struct_info($type); + my $prefs = ""; + foreach my $pref (sort keys %{$si->{paramref}}) { + $prefs .= ", f_$pref"; + } + + print $impl $indent."struct_$info->{'name'}(tvb, offsetp, t, byte_order, $lencalc$prefs);\n"; + } else { + # TODO: Fix unrecognized type. Comment out for now to generate dissector + # die ("Unrecognized type: $type\n"); + } + + if ($adjustlength && defined($lentype)) { + # Some requests end with a list of unspecified length + # Adjust the length field here so that the next $lencalc will be accurate + if (defined $info->{'size'}) { + say $impl $indent."length -= $lencalc * $info->{'size'};"; + } else { + say $impl $indent."length -= $lencalc * 1;"; + } + } + } + when ('switch') { + my $switchtype = $e->first_child() or die("Switch element not defined"); + + my $switchon = get_ref($switchtype, {}); + my @elements = $e->children(qr/(bit)?case/); + for my $case (@elements) { + my @refs = $case->children('enumref'); + my @test; + my $fieldname; + foreach my $ref (@refs) { + my $enum_ref = $ref->att('ref'); + my $field = $ref->text(); + $fieldname //= $field; # Use first named field + if ($case->name() eq 'bitcase') { + my $bit = $enum{$enum_name{$enum_ref}}{rbit}{$field}; + if (! defined($bit)) { + for my $foo (keys %{$enum{$enum_name{$enum_ref}}{rbit}}) { say "'$foo'"; } + die ("Field '$field' not found in '$enum_ref'"); + } + push @test , "$switchon & (1U << $bit)"; + } else { + my $val = $enum{$enum_name{$enum_ref}}{rvalue}{$field}; + if (! defined($val)) { + for my $foo (keys %{$enum{$enum_name{$enum_ref}}{rvalue}}) { say "'$foo'"; } + die ("Field '$field' not found in '$enum_ref'"); + } + push @test , "$switchon == $val"; + } + } + + if (@test > 1) { + # We have more than one conditional, add parentheses to them. + # We don't add parentheses to all the conditionals because + # clang complains about the extra parens if you do "if ((x == y))". + my @tests_with_parens; + foreach my $conditional (@test) { + push @tests_with_parens, "($conditional)"; + } + + @test = @tests_with_parens; + } + + my $list = join ' || ', @test; + say $impl $indent."if ($list) {"; + + my $vp = $varpat; + my $hp = $humanpat; + + $vp =~ s/%s/${fieldname}_%s/; + $hp =~ s/%s/${fieldname}.%s/; + + my @sub_elements = $case->children(qr/pad|field|list|switch/); + + my $subref = { field => {}, sumof => {} }; + foreach my $sub_e (@sub_elements) { + reference_elements($sub_e, $subref); + } + foreach my $sub_e (@sub_elements) { + register_element($sub_e, $vp, $hp, $subref, $indent . ' '); + } + foreach my $sub_e (@sub_elements) { + $length = dissect_element($sub_e, $vp, $hp, $length, $subref, $adjustlength, $indent . ' '); + } + + say $impl $indent."}"; + } + } + default { die "Unknown field type: $_\n"; } + } + return $length; +} + +sub struct { + my ($t, $elt) = @_; + my $name = $elt->att('name'); + my $qualname = qualname($name); + $type_name{$name} = $qualname; + + if (defined $struct{$qualname}) { + $t->purge; + return; + } + + my @elements = $elt->children(qr/pad|field|list|switch/); + + print(" - Struct $name\n"); + + $name = $qualname; + $name =~ s/:/_/; + + my %refs; + my %paramrefs; + my $size = 0; + my $dynamic = 0; + my $needi = 0; + # Find struct size + foreach my $e (@elements) { + my $count; + $count = 1; + given ($e->name()) { + when ('pad') { + my $bytes = $e->att('bytes'); + my $align = $e->att('align'); + if (defined $bytes) { + $size += $bytes; + next; + } + if (!$dynamic) { + if ($size % $align) { + $size += $align - $size % $align; + } + } + next; + } + when ('list') { + my $type = $e->att('type'); + my $info = getinfo($type); + + $needi = 1 if ($info->{'size'} == 0); + + my $value = $e->first_child(); + given($value->name()) { + when ('fieldref') { + $refs{$value->text()} = 1; + $count = 0; + $dynamic = 1; + } + when ('paramref') { + $paramrefs{$value->text()} = $value->att('type'); + $count = 0; + $dynamic = 1; + } + when ('op') { + get_op($value, \%refs); + $count = 0; + $dynamic = 1; + } + when (['unop','popcount']) { + get_unop($value, \%refs); + $count = 0; + $dynamic = 1; + } + when ('value') { + $count = $value->text(); + } + default { die("Invalid list size $_\n"); } + } + } + when ('field') { } + when ('switch') { + $dynamic = 1; + next; + } + default { die("unrecognized field: $_\n"); } + } + + my $type = $e->att('type'); + my $info = getinfo($type); + + $size += $info->{'size'} * $count; + } + + my $prefs = ""; + + if ($dynamic) { + $size = 0; + + foreach my $pref (sort keys %paramrefs) { + $prefs .= ", int p_$pref"; + } + + print $impl <<eot + +static int struct_size_$name(tvbuff_t *tvb _U_, int *offsetp _U_, guint byte_order _U_$prefs) +{ + int size = 0; +eot +; + say $impl ' int i, off;' if ($needi); + + foreach my $ref (sort keys %refs) { + say $impl " int f_$ref;"; + } + + foreach my $e (@elements) { + my $count; + $count = 1; + + my $type = $e->att('type') // ''; + my $info = getinfo($type); + + given ($e->name()) { + when ('pad') { + my $bytes = $e->att('bytes'); + my $align = $e->att('align'); + if (defined $bytes) { + $size += $bytes; + } else { + say $impl ' size = (size + '.($align-1).') & ~'.($align-1).';'; + } + } + when ('list') { + my $len = $e->first_child(); + my $infosize = $info->{'size'}; + my $sizemul; + + given ($len->name()) { + when ('op') { $sizemul = get_op($len, \%refs); } + when (['unop','popcount']) { $sizemul = get_unop($len, \%refs); } + when ('fieldref') { $sizemul = 'f_'.$len->text(); } + when ('paramref') { $sizemul = 'p_'.$len->text(); } + when ('value') { + if ($infosize) { + $size += $infosize * $len->text(); + } else { + $sizemul = $len->text(); + } + } + default { die "Invalid list size: $_\n"; } + } + if (defined $sizemul) { + if ($infosize) { + say $impl " size += $sizemul * $infosize;"; + } else { + say $impl " for (i = 0; i < $sizemul; i++) {"; + say $impl " off = (*offsetp) + size + $size;"; + say $impl " size += struct_size_$info->{name}(tvb, &off, byte_order);"; + say $impl ' }'; + } + } + } + when ('field') { + my $fname = $e->att('name'); + if (defined($refs{$fname})) { + my $get = $info->{'get'}; + if ($get ne "tvb_get_guint8") { + say $impl " f_$fname = $info->{'get'}(tvb, *offsetp + size + $size, byte_order);"; + } else { + say $impl " f_$fname = $info->{'get'}(tvb, *offsetp + size + $size);"; + } + } + $size += $info->{'size'}; + } + } + } + say $impl " return size + $size;"; + say $impl '}'; + $size = 0; # 0 means "dynamic calcuation required" + } + + print $decl "static int hf_x11_struct_$name = -1;\n"; + print $reg "{ &hf_x11_struct_$name, { \"$name\", \"x11.struct.$name\", FT_NONE, BASE_NONE, NULL, 0, NULL, HFILL }},\n"; + + print $impl <<eot + +static void struct_$name(tvbuff_t *tvb, int *offsetp, proto_tree *root, guint byte_order _U_, int count$prefs) +{ + int i; + for (i = 0; i < count; i++) { + proto_item *item; + proto_tree *t; +eot +; + + my $varpat = 'struct_'.$name.'_%s'; + my $humanpat = "struct.$name.%s"; + my $refs = { field => {}, sumof => {} }; + + foreach my $e (@elements) { + reference_elements($e, $refs); + } + foreach my $e (@elements) { + register_element($e, $varpat, $humanpat, $refs, " "); + } + + $prefs = ""; + foreach my $pref (sort keys %paramrefs) { + $prefs .= ", p_$pref"; + } + + my $sizecalc = $size; + $size or $sizecalc = "struct_size_$name(tvb, offsetp, byte_order$prefs)"; + + print $impl <<eot + + item = proto_tree_add_item(root, hf_x11_struct_$name, tvb, *offsetp, $sizecalc, ENC_NA); + t = proto_item_add_subtree(item, ett_x11_rectangle); +eot +; + my $length = 0; + foreach my $e (@elements) { + $length = dissect_element($e, $varpat, $humanpat, $length, $refs, 0, " "); + } + + print $impl " }\n}\n"; + $struct{$qualname} = { size => $size, name => $name, paramref => \%paramrefs }; + $t->purge; +} + +sub union { + # TODO proper dissection + # + # Right now, the only extension to use a union is randr. + # for now, punt. + my ($t, $elt) = @_; + my $name = $elt->att('name'); + my $qualname = qualname($name); + $type_name{$name} = $qualname; + + if (defined $struct{$qualname}) { + $t->purge; + return; + } + + my @elements = $elt->children(qr/field/); + my @sizes; + + print(" - Union $name\n"); + + $name = $qualname; + $name =~ s/:/_/; + + # Find union size + foreach my $e (@elements) { + my $type = $e->att('type'); + my $info = getinfo($type); + + $info->{'size'} > 0 or die ("Error: Union containing variable sized struct $type\n"); + push @sizes, $info->{'size'}; + } + @sizes = sort {$b <=> $a} @sizes; + my $size = $sizes[0]; + + print $decl "static int hf_x11_union_$name = -1;\n"; + print $reg "{ &hf_x11_union_$name, { \"$name\", \"x11.union.$name\", FT_NONE, BASE_NONE, NULL, 0, NULL, HFILL }},\n"; + + print $impl <<eot + +static void struct_$name(tvbuff_t *tvb, int *offsetp, proto_tree *root, guint byte_order, int count) +{ + int i; + int base = *offsetp; + for (i = 0; i < count; i++) { + proto_item *item; + proto_tree *t; +eot +; + + my $varpat = 'union_'.$name.'_%s'; + my $humanpat = "union.$name.%s"; + my $refs = { field => {}, sumof => {} }; + + foreach my $e (@elements) { + reference_elements($e, $refs); + } + foreach my $e (@elements) { + register_element($e, $varpat, $humanpat, $refs, " "); + } + + print $impl <<eot + item = proto_tree_add_item(root, hf_x11_union_$name, tvb, base, $size, ENC_NA); + t = proto_item_add_subtree(item, ett_x11_rectangle); + +eot +; + + foreach my $e (@elements) { + say $impl ' *offsetp = base;'; + dissect_element($e, $varpat, $humanpat, 0, $refs, 0, " "); + } + say $impl " base += $size;"; + say $impl ' }'; + say $impl ' *offsetp = base;'; + say $impl '}'; + + $struct{$qualname} = { size => $size, name => $name }; + $t->purge; +} + +sub enum { + my ($t, $elt) = @_; + my $name = $elt->att('name'); + my $fullname = $incname[0].'_'.$name; + + $enum_name{$name} = $fullname; + $enum_name{$incname[0].':'.$name} = $fullname; + + if (defined $enum{$fullname}) { + $t->purge; + return; + } + + my @elements = $elt->children('item'); + + print(" - Enum $name\n"); + + my $value = {}; + my $bit = {}; + my $rvalue = {}; + my $rbit = {}; + $enum{$fullname} = { value => $value, bit => $bit, rbit => $rbit, rvalue => $rvalue }; + + my $nextvalue = 0; + + foreach my $e (@elements) { + my $n = $e->att('name'); + my $valtype = $e->first_child(qr/value|bit/); + if (defined $valtype) { + my $val = int($valtype->text()); + given ($valtype->name()) { + when ('value') { + $$value{$val} = $n; + $$rvalue{$n} = $val; + $nextvalue = $val + 1; + + # Ugly hack to support (temporary, hopefully) ugly + # hack in xinput:ChangeDeviceProperty + # Register certain values as bits also + given ($val) { + when (8) { + $$bit{'3'} = $n; + $$rbit{$n} = 3; + } + when (16) { + $$bit{'4'} = $n; + $$rbit{$n} = 4; + } + when (32) { + $$bit{'5'} = $n; + $$rbit{$n} = 5; + } + } + } + when ('bit') { + $$bit{$val} = $n; + $$rbit{$n} = $val; + } + } + } else { + $$value{$nextvalue} = $n; + $nextvalue++; + } + } + + $t->purge; +} + +sub request { + my ($t, $elt) = @_; + my $name = $elt->att('name'); + + print(" - Request $name\n"); + $request{$elt->att('opcode')} = $name; + + my $length = 4; + my @elements = $elt->children(qr/pad|field|list|switch/); + + # Wireshark defines _U_ to mean "Unused" (compiler specific define) + if (!@elements) { + print $impl <<eot + +static void $header$name(tvbuff_t *tvb _U_, packet_info *pinfo _U_, int *offsetp _U_, proto_tree *t _U_, guint byte_order _U_, int length _U_) +{ +eot +; + } else { + print $impl <<eot + +static void $header$name(tvbuff_t *tvb, packet_info *pinfo _U_, int *offsetp, proto_tree *t, guint byte_order, int length _U_) +{ +eot +; + } + my $varpat = $header.'_'.$name.'_%s'; + my $humanpat = "$header.$name.%s"; + my $refs = { field => {}, sumof => {} }; + + foreach my $e (@elements) { + reference_elements($e, $refs); + } + foreach my $e (@elements) { + register_element($e, $varpat, $humanpat, $refs); + } + + foreach my $e (@elements) { + if ($e->name() eq 'list' && $name eq 'Render' && $e->att('name') eq 'data' && -e "$mesadir/gl_API.xml") { + # Special case: Use mesa-generated dissector for 'data' + print $impl " dispatch_glx_render(tvb, pinfo, offsetp, t, byte_order, (length - $length));\n"; + } else { + $length = dissect_element($e, $varpat, $humanpat, $length, $refs, 1); + } + } + + say $impl '}'; + + my $reply = $elt->first_child('reply'); + if ($reply) { + $reply{$elt->att('opcode')} = $name; + + $varpat = $header.'_'.$name.'_reply_%s'; + $humanpat = "$header.$name.reply.%s"; + + @elements = $reply->children(qr/pad|field|list|switch/); + + # Wireshark defines _U_ to mean "Unused" (compiler specific define) + if (!@elements) { + say $impl "static void $header$name"."_Reply(tvbuff_t *tvb _U_, packet_info *pinfo, int *offsetp _U_, proto_tree *t _U_, guint byte_order _U_)\n{"; + } else { + say $impl "static void $header$name"."_Reply(tvbuff_t *tvb, packet_info *pinfo, int *offsetp, proto_tree *t, guint byte_order)\n{"; + } + say $impl ' int sequence_number;' if (@elements); + + my $refs = { field => {}, sumof => {} }; + foreach my $e (@elements) { + reference_elements($e, $refs); + } + + say $impl ' int f_length;' if ($refs->{field}{'length'}); + say $impl ' int length;' if ($refs->{length}); + foreach my $e (@elements) { + register_element($e, $varpat, $humanpat, $refs); + } + + say $impl ''; + say $impl ' col_append_fstr(pinfo->cinfo, COL_INFO, "-'.$name.'");'; + say $impl ''; + say $impl ' REPLY(reply);'; + + my $first = 1; + my $length = 1; + foreach my $e (@elements) { + $length = dissect_element($e, $varpat, $humanpat, $length, $refs); + if ($first) { + $first = 0; + say $impl ' sequence_number = tvb_get_guint16(tvb, *offsetp, byte_order);'; + say $impl ' proto_tree_add_uint_format_value(t, hf_x11_reply_sequencenumber, tvb, *offsetp, 2, sequence_number,'; + say $impl ' "%d ('.$header.'-'.$name.')", sequence_number);'; + say $impl ' *offsetp += 2;'; + + if ($refs->{field}{length}) { + say $impl ' f_length = tvb_get_guint32(tvb, *offsetp, byte_order);'; + } + if ($refs->{length}) { + say $impl ' length = f_length * 4 + 32;'; + } + say $impl ' proto_tree_add_item(t, hf_x11_replylength, tvb, *offsetp, 4, byte_order);'; + say $impl ' *offsetp += 4;'; + + $length += 6; + } + } + + say $impl '}'; + } + $t->purge; +} + +sub defxid(@) { + my $name; + while ($name = shift) { + my $qualname = qualname($name); + $simpletype{$qualname} = { size => 4, encoding => 'byte_order', type => 'FT_UINT32', base => 'BASE_HEX', get => 'tvb_get_guint32', list => 'listOfCard32', }; + $type_name{$name} = $qualname; + } +} + +sub xidtype { + my ($t, $elt) = @_; + my $name = $elt->att('name'); + + defxid($name); + + $t->purge; +} + +sub typedef { + my ($t, $elt) = @_; + my $oldname = $elt->att('oldname'); + my $newname = $elt->att('newname'); + my $qualname = qualname($newname); + + # Duplicate the type + my $info = get_simple_info($oldname); + if ($info) { + $simpletype{$qualname} = $info; + } elsif ($info = get_struct_info($oldname)) { + $struct{$qualname} = $info; + } else { + die ("$oldname not found while attempting to typedef $newname\n"); + } + $type_name{$newname} = $qualname; + + $t->purge; +} + +sub error { + my ($t, $elt) = @_; + + my $number = $elt->att('number'); + if ($number >= 0) { + my $name = $elt->att('name'); + print $error " \"$header-$name\",\n"; + } + + $t->purge; +} + +sub event { + my ($t, $elt) = @_; + + my $number = $elt->att('number'); + $number or return; + + my $name = $elt->att('name'); + my $xge = $elt->att('xge'); + + if ($xge) { + $genericevent{$number} = $name; + } else { + $event{$number} = $name; + } + + my $length = 1; + my @elements = $elt->children(qr/pad|field|list|switch/); + + # Wireshark defines _U_ to mean "Unused" (compiler specific define) + if (!@elements) { + if ($xge) { + print $impl <<eot + +static void $header$name(tvbuff_t *tvb _U_, int length _U_, int *offsetp _U_, proto_tree *t _U_, guint byte_order _U_) +{ + } else { + print $impl <<eot + +static void $header$name(tvbuff_t *tvb _U_, int *offsetp _U_, proto_tree *t _U_, guint byte_order _U_) +{ +eot +; + } + } else { + if ($xge) { + $length = 10; + print $impl <<eot + +static void $header$name(tvbuff_t *tvb, int length _U_, int *offsetp, proto_tree *t, guint byte_order) +{ +eot +; + } else { + print $impl <<eot + +static void $header$name(tvbuff_t *tvb, int *offsetp, proto_tree *t, guint byte_order) +{ +eot +; + } + } + + my $varpat = $header.'_'.$name.'_%s'; + my $humanpat = "$header.$name.%s"; + my $refs = { field => {}, sumof => {} }; + + foreach my $e (@elements) { + reference_elements($e, $refs); + } + foreach my $e (@elements) { + register_element($e, $varpat, $humanpat, $refs); + } + + if ($xge) { + say $impl " proto_tree_add_uint_format_value(t, hf_x11_minor_opcode, tvb, *offsetp, 2, $number,"; + say $impl " \"$name ($number)\");"; + foreach my $e (@elements) { + $length = dissect_element($e, $varpat, $humanpat, $length, $refs); + } + } else { + my $first = 1; + foreach my $e (@elements) { + $length = dissect_element($e, $varpat, $humanpat, $length, $refs); + if ($first) { + $first = 0; + say $impl " CARD16(event_sequencenumber);"; + } + } + } + + say $impl "}\n"; + + $t->purge; +} + +sub include_start { + my ($t, $elt) = @_; + my $header = $elt->att('header'); + unshift @incname, $header; +} + +sub include_end { + shift @incname; +} + +sub include +{ + my ($t, $elt) = @_; + my $include = $elt->text(); + + print " - Import $include\n"; + my $xml = XML::Twig->new( + start_tag_handlers => { + 'xcb' => \&include_start, + }, + twig_roots => { + 'import' => \&include, + 'struct' => \&struct, + 'xidtype' => \&xidtype, + 'xidunion' => \&xidtype, + 'typedef' => \&typedef, + 'enum' => \&enum, + }, + end_tag_handlers => { + 'xcb' => \&include_end, + }); + $xml->parsefile("$srcdir/xcbproto/src/$include.xml") or die ("Cannot open $include.xml\n"); + + $t->purge; +} + + +sub xcb_start { + my ($t, $elt) = @_; + $header = $elt->att('header'); + $extname = ($elt->att('extension-name') or $header); + unshift @incname, $header; + + print("Extension $extname\n"); + + undef %request; + undef %genericevent; + undef %event; + undef %reply; + + %simpletype = (); + %enum_name = (); + %type_name = (); + + print $error "const char *$header"."_errors[] = {\n"; +} + +sub xcb { + my ($t, $elt) = @_; + + my $xextname = $elt->att('extension-xname'); + my $lookup_name = $header . "_extension_minor"; + my $error_name = $header . "_errors"; + my $event_name = $header . "_events"; + my $genevent_name = 'NULL'; + my $reply_name = $header . "_replies"; + + print $decl "static int hf_x11_$lookup_name = -1;\n\n"; + + print $impl "static const value_string $lookup_name"."[] = {\n"; + foreach my $req (sort {$a <=> $b} keys %request) { + print $impl " { $req, \"$request{$req}\" },\n"; + } + print $impl " { 0, NULL }\n"; + print $impl "};\n"; + + say $impl "const x11_event_info $event_name".'[] = {'; + foreach my $e (sort {$a <=> $b} keys %event) { + say $impl " { \"$header-$event{$e}\", $header$event{$e} },"; + } + say $impl ' { NULL, NULL }'; + say $impl '};'; + + if (%genericevent) { + $genevent_name = $header.'_generic_events'; + say $impl 'static const x11_generic_event_info '.$genevent_name.'[] = {'; + + for my $val (sort { $a <=> $b } keys %genericevent) { + say $impl sprintf(" { %3d, %s },", $val, $header.$genericevent{$val}); + } + say $impl sprintf(" { %3d, NULL },", 0); + say $impl '};'; + say $impl ''; + } + + print $impl "static x11_reply_info $reply_name"."[] = {\n"; + foreach my $e (sort {$a <=> $b} keys %reply) { + print $impl " { $e, $header$reply{$e}_Reply },\n"; + } + print $impl " { 0, NULL }\n"; + print $impl "};\n"; + + print $reg "{ &hf_x11_$lookup_name, { \"extension-minor\", \"x11.extension-minor\", FT_UINT8, BASE_DEC, VALS($lookup_name), 0, \"minor opcode\", HFILL }},\n\n"; + + print $impl <<eot + +static void dispatch_$header(tvbuff_t *tvb, packet_info *pinfo, int *offsetp, proto_tree *t, guint byte_order) +{ + int minor, length; + minor = CARD8($lookup_name); + length = REQUEST_LENGTH(); + + col_append_fstr(pinfo->cinfo, COL_INFO, "-%s", + val_to_str(minor, $lookup_name, + "<Unknown opcode %d>")); + switch (minor) { +eot + ; + + foreach my $req (sort {$a <=> $b} keys %request) { + print $impl " case $req:\n"; + print $impl " $header$request{$req}(tvb, pinfo, offsetp, t, byte_order, length);\n"; + print $impl " break;\n"; + } + say $impl " /* No need for a default case here, since Unknown is printed above,"; + say $impl " and UNDECODED() is taken care of by dissect_x11_request */"; + print $impl " }\n}\n"; + print $impl <<eot + +static void register_$header(void) +{ + set_handler("$xextname", dispatch_$header, $error_name, $event_name, $genevent_name, $reply_name); +} +eot + ; + + print $error " NULL\n};\n\n"; + + push @register, $header; +} + +sub find_version { + #my $git = `which git`; + #chomp($git); + #-x $git or return 'unknown'; + + my $lib = shift; + # this will generate an error on stderr if git isn't in our $PATH + # but that's OK. The version is still set to 'unknown' in that case + # and at least the operator could see it. + my $ver = `git --git-dir=$lib/.git describe --tags`; + $ver //= 'unknown'; + chomp $ver; + return $ver; +} + +sub add_generated_header { + my ($out, $using) = @_; + my $ver = find_version($using); + + $using = File::Spec->abs2rel ($using, $srcdir); + + print $out <<eot +/* Do not modify this file. */ +/* It was automatically generated by $script_name + using $using version $ver */ +eot + ; + + # Add license text + print $out <<eot +/* + * Copyright 2008, 2009, 2013, 2014 Open Text Corporation <pharris[AT]opentext.com> + * + * Wireshark - Network traffic analyzer + * By Gerald Combs <gerald[AT]wireshark.org> + * Copyright 1998 Gerald Combs + * + * SPDX-License-Identifier: GPL-2.0-or-later + */ + +eot + ; +} + +# initialize core X11 protocol +# Do this in the Makefile now +#system('./process-x11-fields.pl < x11-fields'); + +# Extension implementation +$impl = new IO::File "> $srcdir/x11-extension-implementation.h" + or die ("Cannot open $srcdir/x11-extension-implementation.h for writing\n"); +$error = new IO::File "> $srcdir/x11-extension-errors.h" + or die ("Cannot open $srcdir/x11-extension-errors.h for writing\n"); + +add_generated_header($impl, $srcdir . '/xcbproto'); +add_generated_header($error, $srcdir . '/xcbproto'); + +# Open the files generated by process-x11-fields.pl for appending +$reg = new IO::File ">> $srcdir/x11-register-info.h" + or die ("Cannot open $srcdir/x11-register-info.h for appending\n"); +$decl = new IO::File ">> $srcdir/x11-declarations.h" + or die ("Cannot open $srcdir/x11-declarations.h for appending\n"); + +print $reg "\n/* Generated by $script_name below this line */\n"; +print $decl "\n/* Generated by $script_name below this line */\n"; + +# Mesa for glRender +if (-e "$mesadir/gl_API.xml") { + $enum = new IO::File "> $srcdir/x11-glx-render-enum.h" + or die ("Cannot open $srcdir/x11-glx-render-enum.h for writing\n"); + add_generated_header($enum, $srcdir . '/mesa'); + print $enum "static const value_string mesa_enum[] = {\n"; + print $impl '#include "x11-glx-render-enum.h"'."\n\n"; + + print("Mesa glRender:\n"); + $header = "glx_render"; + + my $xml = XML::Twig->new( + start_tag_handlers => { + }, + twig_roots => { + 'category' => \&mesa_category, + 'enum' => \&mesa_enum, + 'type' => \&mesa_type, + 'function' => \&mesa_function, + }); + $xml->parsefile("$mesadir/gl_API.xml") or die ("Cannot open gl_API\n"); + + for my $enum_key ( sort {$a<=>$b} keys %mesa_enum_hash) { + say $enum sprintf(" { 0x%04x, \"%s\" },", $enum_key, $mesa_enum_hash{$enum_key}); + } + print $enum " { 0, NULL }\n"; + print $enum "};\n"; + $enum->close(); + + print $decl "static int hf_x11_glx_render_op_name = -1;\n\n"; + + print $impl "static const value_string glx_render_op_name"."[] = {\n"; + foreach my $req (sort {$a <=> $b} keys %request) { + print $impl " { $req, \"gl$request{$req}\" },\n"; + } + print $impl " { 0, NULL }\n"; + print $impl "};\n"; + print $impl "static value_string_ext mesa_enum_ext = VALUE_STRING_EXT_INIT(mesa_enum);\n"; + + print $reg "{ &hf_x11_glx_render_op_name, { \"render op\", \"x11.glx.render.op\", FT_UINT16, BASE_DEC, VALS(glx_render_op_name), 0, NULL, HFILL }},\n\n"; + +# Uses ett_x11_list_of_rectangle, since I am unable to see how the subtree type matters. + print $impl <<eot + +static void dispatch_glx_render(tvbuff_t *tvb, packet_info *pinfo, int *offsetp, proto_tree *t, guint byte_order, int length) +{ + while (length >= 4) { + guint32 op, len; + int next; + proto_item *ti; + proto_tree *tt; + + len = tvb_get_guint16(tvb, *offsetp, byte_order); + + op = tvb_get_guint16(tvb, *offsetp + 2, byte_order); + ti = proto_tree_add_uint(t, hf_x11_glx_render_op_name, tvb, *offsetp, len, op); + + tt = proto_item_add_subtree(ti, ett_x11_list_of_rectangle); + + ti = proto_tree_add_item(tt, hf_x11_request_length, tvb, *offsetp, 2, byte_order); + *offsetp += 2; + proto_tree_add_item(tt, hf_x11_glx_render_op_name, tvb, *offsetp, 2, byte_order); + *offsetp += 2; + + if (len < 4) { + expert_add_info(pinfo, ti, &ei_x11_request_length); + /* Eat the rest of the packet, mark it undecoded */ + len = length; + op = -1; + } + len -= 4; + + next = *offsetp + len; + + switch (op) { +eot + ; + foreach my $req (sort {$a <=> $b} keys %request) { + print $impl " case $req:\n"; + print $impl " mesa_$request{$req}(tvb, offsetp, tt, byte_order, len);\n"; + print $impl " break;\n"; + } + print $impl " default:\n"; + print $impl " proto_tree_add_item(tt, hf_x11_undecoded, tvb, *offsetp, len, ENC_NA);\n"; + print $impl " *offsetp += len;\n"; + + print $impl " }\n"; + print $impl " if (*offsetp < next) {\n"; + print $impl " proto_tree_add_item(tt, hf_x11_unused, tvb, *offsetp, next - *offsetp, ENC_NA);\n"; + print $impl " *offsetp = next;\n"; + print $impl " }\n"; + print $impl " length -= (len + 4);\n"; + print $impl " }\n}\n"; +} + +$enum = new IO::File "> $srcdir/x11-enum.h" + or die ("Cannot open $srcdir/x11-enum.h for writing\n"); +add_generated_header($enum, $srcdir . '/xcbproto'); +print $impl '#include "x11-enum.h"'."\n\n"; + +# XCB +foreach my $ext (@reslist) { + my $xml = XML::Twig->new( + start_tag_handlers => { + 'xcb' => \&xcb_start, + }, + twig_roots => { + 'xcb' => \&xcb, + 'import' => \&include, + 'request' => \&request, + 'struct' => \&struct, + 'union' => \&union, + 'xidtype' => \&xidtype, + 'xidunion' => \&xidtype, + 'typedef' => \&typedef, + 'error' => \&error, + 'errorcopy' => \&error, + 'event' => \&event, + 'enum' => \&enum, + }); + $xml->parsefile($ext) or die ("Cannot open $ext\n"); +} + +print $impl "static void register_x11_extensions(void)\n{\n"; +foreach my $reg (@register) { + print $impl " register_$reg();\n"; +} +print $impl "}\n"; + +# +# Editor modelines +# +# Local Variables: +# c-basic-offset: 4 +# tab-width: 8 +# indent-tabs-mode: nil +# End: +# +# ex: set shiftwidth=4 tabstop=8 expandtab: +# :indentSize=4:tabSize=8:noTabs=true: +# |