# # Copyright 2002 Patrik Stridvall # # This library is free software; you can redistribute it and/or # modify it under the terms of the GNU Lesser General Public # License as published by the Free Software Foundation; either # version 2.1 of the License, or (at your option) any later version. # # This library is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU # Lesser General Public License for more details. # # You should have received a copy of the GNU Lesser General Public # License along with this library; if not, write to the Free Software # Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA # package c_type; use strict; use output qw($output); sub new { my $proto = shift; my $class = ref($proto) || $proto; my $self = {}; bless ($self, $class); return $self; } ######################################################################## # set_find_align_callback # sub set_find_align_callback { my $self = shift; my $find_align = \${$self->{FIND_ALIGN}}; $$find_align = shift; } ######################################################################## # set_find_kind_callback # sub set_find_kind_callback { my $self = shift; my $find_kind = \${$self->{FIND_KIND}}; $$find_kind = shift; } ######################################################################## # set_find_size_callback # sub set_find_size_callback { my $self = shift; my $find_size = \${$self->{FIND_SIZE}}; $$find_size = shift; } sub kind { my $self = shift; my $kind = \${$self->{KIND}}; my $dirty = \${$self->{DIRTY}}; local $_ = shift; if(defined($_)) { $$kind = $_; $$dirty = 1; } if (!defined($$kind)) { $self->_refresh(); } return $$kind; } sub _name { my $self = shift; my $_name = \${$self->{_NAME}}; my $dirty = \${$self->{DIRTY}}; local $_ = shift; if(defined($_)) { $$_name = $_; $$dirty = 1; } return $$_name; } sub name { my $self = shift; my $name = \${$self->{NAME}}; my $dirty = \${$self->{DIRTY}}; local $_ = shift; if(defined($_)) { $$name = $_; $$dirty = 1; } if($$name) { return $$name; } else { my $kind = \${$self->{KIND}}; my $_name = \${$self->{_NAME}}; return "$$kind $$_name"; } } sub pack { my $self = shift; my $pack = \${$self->{PACK}}; my $dirty = \${$self->{DIRTY}}; local $_ = shift; if(defined($_)) { $$pack = $_; $$dirty = 1; } return $$pack; } sub align { my $self = shift; my $align = \${$self->{ALIGN}}; $self->_refresh(); return $$align; } sub fields { my $self = shift; my $count = $self->field_count; my @fields = (); for (my $n = 0; $n < $count; $n++) { my $field = 'c_type_field'->new($self, $n); push @fields, $field; } return @fields; } sub field_base_sizes { my $self = shift; my $field_base_sizes = \${$self->{FIELD_BASE_SIZES}}; $self->_refresh(); return $$field_base_sizes; } sub field_aligns { my $self = shift; my $field_aligns = \${$self->{FIELD_ALIGNS}}; $self->_refresh(); return $$field_aligns; } sub field_count { my $self = shift; my $field_type_names = \${$self->{FIELD_TYPE_NAMES}}; my @field_type_names = @{$$field_type_names}; my $count = scalar(@field_type_names); return $count; } sub field_names { my $self = shift; my $field_names = \${$self->{FIELD_NAMES}}; my $dirty = \${$self->{DIRTY}}; local $_ = shift; if(defined($_)) { $$field_names = $_; $$dirty = 1; } return $$field_names; } sub field_offsets { my $self = shift; my $field_offsets = \${$self->{FIELD_OFFSETS}}; $self->_refresh(); return $$field_offsets; } sub field_sizes { my $self = shift; my $field_sizes = \${$self->{FIELD_SIZES}}; $self->_refresh(); return $$field_sizes; } sub field_type_names { my $self = shift; my $field_type_names = \${$self->{FIELD_TYPE_NAMES}}; my $dirty = \${$self->{DIRTY}}; local $_ = shift; if(defined($_)) { $$field_type_names = $_; $$dirty = 1; } return $$field_type_names; } sub size { my $self = shift; my $size = \${$self->{SIZE}}; $self->_refresh(); return $$size; } sub _refresh { my $self = shift; my $dirty = \${$self->{DIRTY}}; return if !$$dirty; my $find_align = \${$self->{FIND_ALIGN}}; my $find_kind = \${$self->{FIND_KIND}}; my $find_size = \${$self->{FIND_SIZE}}; my $align = \${$self->{ALIGN}}; my $kind = \${$self->{KIND}}; my $size = \${$self->{SIZE}}; my $field_aligns = \${$self->{FIELD_ALIGNS}}; my $field_base_sizes = \${$self->{FIELD_BASE_SIZES}}; my $field_offsets = \${$self->{FIELD_OFFSETS}}; my $field_sizes = \${$self->{FIELD_SIZES}}; my $pack = $self->pack; $pack = 4 if !defined($pack); my $max_field_align = 0; my $offset = 0; my $offset_bits = 0; my $n = 0; foreach my $field ($self->fields) { my $type_name = $field->type_name; my $type_size = &$$find_size($type_name); my $base_type_name = $type_name; if ($base_type_name =~ s/^(.*?)\s*(?:\[\s*(.*?)\s*\]|:(\d+))?$/$1/) { my $count = $2; my $bits = $3; } my $base_size = &$$find_size($base_type_name); my $align = &$$find_align($base_type_name); if (defined($align)) { $align = $pack if $align > $pack; $max_field_align = $align if $align > $max_field_align; if ($offset % $align != 0) { $offset = (int($offset / $align) + 1) * $align; } } if ($$kind !~ /^(?:struct|union)$/) { $$kind = &$$find_kind($type_name) || ""; } if (!defined($type_size)) { $$align = undef; $$size = undef; return; } elsif ($type_size >= 0) { if ($offset_bits) { $offset += $pack * int(($offset_bits + 8 * $pack - 1 ) / (8 * $pack)); $offset_bits = 0; } $$$field_aligns[$n] = $align; $$$field_base_sizes[$n] = $base_size; $$$field_offsets[$n] = $offset; $$$field_sizes[$n] = $type_size; $offset += $type_size; } else { $$$field_aligns[$n] = $align; $$$field_base_sizes[$n] = $base_size; $$$field_offsets[$n] = $offset; $$$field_sizes[$n] = $type_size; $offset_bits += -$type_size; } $n++; } $$align = $pack; $$align = $max_field_align if $max_field_align < $pack; if ($offset_bits) { $offset += $pack * int(($offset_bits + 8 * $pack - 1 ) / (8 * $pack)); $offset_bits = 0; } $$size = $offset; if ($$kind =~ /^(?:struct|union)$/) { if ($$size % $$align != 0) { $$size = (int($$size / $$align) + 1) * $$align; } } $$dirty = 0; } package c_type_field; sub new { my $proto = shift; my $class = ref($proto) || $proto; my $self = {}; bless ($self, $class); my $type = \${$self->{TYPE}}; my $number = \${$self->{NUMBER}}; $$type = shift; $$number = shift; return $self; } sub align { my $self = shift; my $type = \${$self->{TYPE}}; my $number = \${$self->{NUMBER}}; my $field_aligns = $$type->field_aligns; return $$field_aligns[$$number]; } sub base_size { my $self = shift; my $type = \${$self->{TYPE}}; my $number = \${$self->{NUMBER}}; my $field_base_sizes = $$type->field_base_sizes; return $$field_base_sizes[$$number]; } sub name { my $self = shift; my $type = \${$self->{TYPE}}; my $number = \${$self->{NUMBER}}; my $field_names = $$type->field_names; return $$field_names[$$number]; } sub offset { my $self = shift; my $type = \${$self->{TYPE}}; my $number = \${$self->{NUMBER}}; my $field_offsets = $$type->field_offsets; return $$field_offsets[$$number]; } sub size { my $self = shift; my $type = \${$self->{TYPE}}; my $number = \${$self->{NUMBER}}; my $field_sizes = $$type->field_sizes; return $$field_sizes[$$number]; } sub type_name { my $self = shift; my $type = \${$self->{TYPE}}; my $number = \${$self->{NUMBER}}; my $field_type_names = $$type->field_type_names; return $$field_type_names[$$number]; } 1;