The London Perl and Raku Workshop takes place on 26th Oct 2024. If your company depends on Perl, please consider sponsoring and/or attending.
use strict;
use warnings;

package ELF::Extract::Sections;
BEGIN {
  $ELF::Extract::Sections::AUTHORITY = 'cpan:KENTNL';
}
{
  $ELF::Extract::Sections::VERSION = '0.03000101';
}

# ABSTRACT: Extract Raw Chunks of data from identifiable ELF Sections

use MooseX::Declare;

class ELF::Extract::Sections with MooseX::Log::Log4perl {


    use MooseX::Has::Sugar 0.0300;
    use MooseX::Types::Moose                ( ':all', );
    use MooseX::Types::Path::Tiny           ( 'File', );
    use ELF::Extract::Sections::Meta::Types ( ':all', );
    use Class::Load                         ( 'try_load_class', );

    require ELF::Extract::Sections::Section;



    has 'file' => ( isa => File, ro, required, coerce, );


    has 'sections' => ( isa => HashRef [ElfSection], ro, lazy_build, );


    has 'scanner' => ( isa => Str, ro, default => 'Objdump', );



    method BUILD ( $args ) {
        if ( not $self->file->stat ) {
            $self->log->logconfess(q{File Specifed could not be found.});
        }
    }


    method sorted_sections (  FilterField :$field!, Bool :$descending? ) {
        my $m = 1;
        $m = 0 - 1 if ($descending);
        return [ sort { $m * ( $a->compare( other => $b, field => $field ) ) }
              values %{ $self->sections } ];
    }



    method _build_sections {
        $self->log->debug('Building Section List');
        if ( $self->_scanner_instance->can_compute_size ) {
            return $self->_scan_with_size;
        }
        else {
            return $self->_scan_guess_size;
        }
    }



    has '_scanner_package' => ( isa => ClassName, ro, lazy_build, );


    has '_scanner_instance' => ( isa => Object, ro, lazy_build, );


    method _error_scanner_missing ( Str $scanner!, Str $package!, Str $error! ) {
        my $message = sprintf qq[The Scanner %s could not be found as %s\n.],
          $scanner, $package;
        $message .= '>' . $error;
        $self->log->logconfess($message);
    }


    method _build__scanner_package {
        my $pkg = 'ELF::Extract::Sections::Scanner::' . $self->scanner;
        my ( $success, $error ) = try_load_class($pkg);
        if ( not $success ) {
            $self->_error_scanner_missing( $self->scanner, $pkg, $error );
        }
        return $pkg;
    }


    method _build__scanner_instance {
        my $instance = $self->_scanner_package->new();
        return $instance;
    }



    method _warn_stash_collision ( Str $stashname!, Str $header!, Str $offset! ) {
        my $message = q[Warning, duplicate file offset reported by scanner.];
        $message .= sprintf q[<%s> and <%s> collide at <%s>.], $stashname,
          $header, $offset;
        $message .= sprintf q[Assuming <%s> is empty and replacing it.],
          $stashname;
        $self->log->warn($message);
    }


    method _stash_record ( HashRef $stash! , Str $header!, Str $offset! ) {
        if ( exists $stash->{$offset} ) {
            $self->_warn_stash_collision( $stash->{$offset}, $header, $offset );
        }
        $stash->{$offset} = $header;
    }


    method _build_section_section ( Str $stashName, Int $start, Int $stop , File $file ) {
        $self->log->info(" Section ${stashName} , ${start} -> ${stop} ");
        return ELF::Extract::Sections::Section->new(
            offset => $start,
            size   => $stop - $start,
            name   => $stashName,
            source => $file,
        );
    }


    method _build_section_table ( HashRef $ob! ) {
        my %datastash = ();
        my @k         = sort { $a <=> $b } keys %{$ob};
        my $i         = 0;
        while ( $i < $#k ) {
            $datastash{ $ob->{ $k[$i] } } = $self->_build_section_section(
                $ob->{ $k[$i] },
                $k[$i], $k[ $i + 1 ],
                $self->file
            );
            $i++;
        }
        return \%datastash;
    }


    method _scan_guess_size {
                              # HACK: Temporary hack around rt#67210
        scalar $self->_scanner_instance->open_file( file => $self->file );
        my %offsets = ();
        while ( $self->_scanner_instance->next_section() ) {
            my $name   = $self->_scanner_instance->section_name;
            my $offset = $self->_scanner_instance->section_offset;
            $self->_stash_record( \%offsets, $name, $offset );
        }
        return $self->_build_section_table( \%offsets );
    }


    method _scan_with_size {
        my %datastash = ();
        $self->_scanner_instance->open_file( file => $self->file );
        while ( $self->_scanner_instance->next_section() ) {
            my $name   = $self->_scanner_instance->section_name;
            my $offset = $self->_scanner_instance->section_offset;
            my $size   = $self->_scanner_instance->section_size;
            $datastash{$name} =
              $self->_build_section_section( $name, $offset, $offset + $size,
                $self->file );
        }
        return \%datastash;
    }

};

1;

__END__

=pod

=head1 NAME

ELF::Extract::Sections - Extract Raw Chunks of data from identifiable ELF Sections

=head1 VERSION

version 0.03000101

=head1 SYNOPSIS

    use ELF::Extract::Sections;

    # Create an extractor object for foo.so
    my $extractor = ELF::Extract::Sections->new( file => '/path/to/foo.so' );

    # Scan file for section data, returns a hash
    my %sections  = ${ $extractor->sections };

    # Retreive the section object for the comment section
    my $data      = $sections{.comment};

    # Print the stringified explanation of the section
    print "$data";

    # Get the raw bytes out of the section.
    print $data->contents  # returns bytes

=head1 CAVEATS

=over 4

=item 1. Beta Software

This code is relatively new. It exists only as a best attempt at present until further notice. It
has proved as practical for at least one application, and this is why the module exists. However, it can't be
guaranteed it will work for whatever you want it to in all cases. Please report any bugs you find.

=item 2. Feature Incomplete

This only presently has a very bare-bones functionality, which should however prove practical for most purposes.
If you have any suggestions, please tell me via "report bugs". If you never seek, you'll never find.

=item 3. Humans

This code is written by a human, and like all human code, it sucks. There will be bugs. Please report them.

=back

=head1 PUBLIC ATTRIBUTES

=head2 file

Returns the file the section data is being created for.

=head2 sections

Returns a HashRef of the available sections.

=head2 scanner

Returns the name of the default scanner plug-in

=head1 PUBLIC METHODS

=head2 new ( file => FILENAME )

Creates A new Section Extractor object with the default scanner

=head2 new ( file => FILENAME , scanner => 'Objdump' )

Creates A new Section Extractor object with the specified scanner

=head2 sorted_sections ( field => SORT_BY )

Returns an ArrayRef sorted by the SORT_BY field, in the default order.

=head2 sorted_sections ( field => SORT_BY, descending => DESCENDING )

Returns an ArrayRef sorted by the SORT_BY field. May be Ascending or Descending depending on requirements.

=head3 DESCENDING

Optional parameters. True for descending, False or absent for ascending.

=head3 SORT_BY

A String of the field to sort by. Valid options at present are

=head4 name

The Section Name

=head4 offset

The Sections offset relative to the start of the file.

=head4 size

The Size of the section.

=head1 PUBLIC ATTRIBUTE BUILDERS

These aren't really user serviceable, but they make your front end work.

=head2 _build_sections

See L</sections>

=head1 PRIVATE ATTRIBUTES

=head2 _scanner_package

    isa => ClassName, ro, lazy_build

=head2 _scanner_instance

    isa => Object, ro, lazy_build

=head1 PRIVATE ATTRIBUTE BUILDERS

=head2 _build__scanner_package

Builds L</_scanner_package>

=head2 _build__scanner_instance

Builds L</_scanner_instance>

=head1 PRIVATE_METHODS

=head2 _warn_stash_collision

    method _warn_stash_collision ( Str $stashname!, Str $header!, Str $offset! ) {

    }

=head2 _stash_record( HashRef, Str, Str )

    method _stash_record ( HashRef $stash! , Str $header!, Str $offset! ) {

    }

=head2 _build_section_section( Str, Int, Int, File )

    method _build_section_section ( Str $stashName, Int $start, Int $stop , File $file ) {

    }

=head2 _build_section_table( HashRef )

    method _build_section_table ( HashRef $ob! ) {
    }

=head2 _scan_guess_size

    method _scan_guess_size {

    }

=head2 _scan_with_size

    method _scan_with_size {
    }

=head1 DEBUGGING

This library uses L<Log::Log4perl>. To see more verbose processing notices, do this:

    use Log::Log4perl qw( :easy );
    Log::Log4perl->easy_init($DEBUG);

For convenience to make sure you don't happen to miss this fact, we never initialize Log4perl ourselves, so it will
spit the following message if you have not set it up:

    Log4perl: Seems like no initialization happened. Forgot to call init()?

To suppress this, just do

    use Log::Log4perl qw( :easy );

I request however you B<don't> do that for modules intended to be consumed by others without good cause.

=head1 AUTHOR

Kent Fredric <kentnl@cpan.org>

=head1 COPYRIGHT AND LICENSE

This software is copyright (c) 2013 by Kent Fredric.

This is free software; you can redistribute it and/or modify it under
the same terms as the Perl 5 programming language system itself.

=cut