kerneltest/e32utils/hcrscripts/hcrdat.pm
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Tue, 31 Aug 2010 16:34:26 +0300
branchRCL_3
changeset 43 c1f20ce4abcf
parent 0 a41df078684a
permissions -rw-r--r--
Revision: 201035 Kit: 201035

#!perl -w
#
# Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies).
# All rights reserved.
# This component and the accompanying materials are made available
# under the terms of the License "Eclipse Public License v1.0"
# which accompanies this distribution, and is available
# at the URL "http://www.eclipse.org/legal/epl-v10.html".
#
# Initial Contributors:
# Nokia Corporation - initial contribution.
#
# Contributors:
#
# Description:
#
use strict;

#use Math::BigInt;

#
# Perl module to create and maintain feature manager data files.
# You can either set up the information programmatically or else load up
# information from a pre-existing feature data file and then modify it. You
# can also save the information to a file (in feature manager dataset format).
#
# This class maintains header information plus two arrays, one containing
# feature flag information and the other containing default supported range
# information. Those are themselves objects and have their own accessor
# methods.
#

package HCRdat;

use HCRrec;


#
# n e w
#
# Create a new HCRdat object. For example 'my $hd = HCRdat->new("filea");
#
sub new
{
	my $arg = shift;
	my $fn = shift;
	my $class = ref($arg) || $arg;
	my $self = {
			        fingerprint => "HCRf",  # 4 bytes wide.
					fileversion => 1,       # 2 bytes.
					fileflags => 0x0001,  # 2 bytes.
					numrecords => 0,      # 4 bytes. 
					lsdoffset => 0,       # 4 bytes. 
					lsdsize => 0,         # 4 bytes.
				    packprefix => "V",    # Changed with endian-ness.
					                      # Used to create binary strings.

					settingrecords => [],  # Array of objects
					lsd => [],             # Array of bytes
	           };
    bless $self, $class;
	return $self;
}


# Print to STDOUT the header information we have.
sub ShowHeader
{
	my $self = shift;
	return undef unless(ref($self));

	# Get header information..
	my $typefield = $self->TypeField();
	my $fileversion = $self->FileVersion();
	my $fileflags = $self->FileFlags();
	my $numrecords = $self->NumRecords();
	my $lsdoffset = $self->LsdOffset();
	my $lsdsize = $self->LsdSize();
	
	# Display it in English.
	print "  FINGERPRINTF: '$typefield'\n";
	print "  FILEVERSION: '$fileversion'\n";
	print "  FILEFLAGS: '$fileflags'\n";
	print "  NUMRECORDS: '$numrecords'\n";
	print "  LSDOFFSET: '$lsdoffset'\n";
    print "  LSDSIZE: '$lsdsize'\n";

	return(0);
}

# Get/Set the endian-ness we want. Changes the 'packprefix' member which is
# used in the creation of binary data.
sub Endian
{
	my $self = shift;
	return undef unless(ref($self));
	my $arg = shift;
	return $self->{endian} unless(defined($arg));
	if($arg =~ m/(LE|BE)/i)
	{
		my $endian = uc($1);
		$self->{endian} = $endian;
		# Used by 'pack' to generate binary strings.
		$self->{packprefix} = "V" if($endian eq "LE");
		$self->{packprefix} = "N" if($endian eq "BE");
	}
	return $self->{endian};
}

# This is the fingerprint.
sub TypeField
{
	my $self = shift;
	return undef unless(ref($self));
	my $arg = shift;
	$self->{fingerprint} = $arg if(defined($arg));
	return $self->{fingerprint};
}

sub FileVersion
{
	my $self = shift;
	return undef unless(ref($self));
	my $arg = shift;
	# Should we be testing for a numeric value?
	$self->{fileversion} = $arg if(defined($arg));
	return $self->{fileversion};
}

sub FileFlags
{
	my $self = shift;
	return undef unless(ref($self));
	my $arg = shift;
	$self->{fileflags} = $arg if(defined($arg));
	return $self->{fileflags};
}

# How many feature flag objects have we got?
sub NumRecords
{
	my $self = shift;
	return undef unless(ref($self));
	my $arg = shift;
	$self->{numrecords} += $arg if(defined($arg));
	return $self->{numrecords};
}


sub LsdOffset
{
	my $self = shift;
	return undef unless(ref($self));
	my $arg = shift;
	$self->{lsdoffset} = $arg if(defined($arg));
	return $self->{lsdoffset};
}

sub LsdSize
{
	my $self = shift;
	return undef unless(ref($self));
	my $arg = shift;
	$self->{lsdsize} = $arg if(defined($arg));
	return $self->{lsdsize};
}

# Create a binary string containing the header information for the
# feature manager data file based on the various fields in this object.
sub CreateBinaryHeader
{
	my $self = shift;
	return undef unless(ref($self));
	my $hdrstring;

	# Get the letter for packing information with 'pack' into a binary form.
	my $pack16 = lc($self->{packprefix});
	my $pack32 = uc($self->{packprefix});
	
	# Get header information..
	my $typefield = $self->TypeField();
	my $fileversion = $self->FileVersion();
	my $fileflags = $self->FileFlags();
	my $numrecords = $self->NumRecords();
	my $lsdoffset = $self->LsdOffset();
	my $lsdsize = $self->LsdSize();

	# Write the 'type' field out. This is 'feat'. Would this be different on
	# big-endian systems?
	$hdrstring = $typefield;

	# Now the file version number. A 16-bit value.. Will this cause trouble
	# if the shifted value is signed?
	$hdrstring .= pack($pack16 . "1", $fileversion);

	# Now the file flags. Another 16-bit value..
	$hdrstring .= pack($pack16 . "1", $fileflags);

	# Now the number of listed features - a 32-bit value.
	$hdrstring .= pack($pack32 . "1", $numrecords);

	# Now the number of listed features - a 32-bit value.
	$hdrstring .= pack($pack32 . "1", $lsdoffset);

	# Now the number of listed features - a 32-bit value.
	$hdrstring .= pack($pack32 . "1", $lsdsize);

	# Now the 3 reserved words
	$hdrstring .= pack($pack32 . "3", (0, 0, 0));

	return $hdrstring;
}

sub CreateImageHdr
{
	my $self = shift;
	return undef unless(ref($self));
	#my $partid = shift;
	#return -1 unless(defined($partid));

	# Add fingerprint, 1st reserved word and format version
	my $imghdr = pack "V4", (0x5F524348, 0x54524150, 0x00000000, 0x00000001);
	# Add space for image size, timestamp, 2nd reserved word
	$imghdr .= pack "V3", (0x00000000, time, 0x00000000);
	# Add space for payload checksum, HCR Payload constants: UID and 0x0 flags
    $imghdr .= pack "V3", (0x00000000, 0x10286AB8, 0x00000000);
    #Reserved space
    $imghdr .= pack "x216", (0x00000000);
 
    return $imghdr;
}

sub WriteToImage
{
	my $self = shift;
	return undef unless(ref($self));
	my $imgfile = shift;
	return -1 unless(defined($imgfile));
	my $datfile = shift;
	return -1 unless(defined($datfile));
	#my $partid = shift;
	#return -1 unless(defined($partid));
	my $rc = 0;
	
    open IMGFILE, "> $imgfile" or die "Couldn't open file '$imgfile' for writing.\n";
	binmode IMGFILE;
   
 	syswrite(IMGFILE, $self->CreateImageHdr(), 256);	
 	
    open DATFILE, "$datfile" or die "Couldn't open file '$datfile' for reading.\n";
	binmode DATFILE;
	# print FILE $self->BinaryContent();
	
    #my $wordsum = 0x1200000000;
    #my $wordsum = Math::BigInt->new("0x0220100123");
    #printf("test: %x\n", $wordsum->external();

	my $imgsize = 256;
	my $word;
	printf("-reading image:\n")  if ($mhd::otrace);
	while (sysread (DATFILE, $word, 4)) {
	    #printf ("%08x ",$word)  if ($mhd::otrace);
        my $iword = unpack("V" , $word);
	    printf ("%08x ",$iword)  if ($mhd::otrace);
        $rc = syswrite (IMGFILE, $word, 4);
        die "error: ($rc) failed to write datfile word into imgfile.\n" if ($rc != 4);
	    #$wordsum->badd($iword);
        $imgsize += 4;
	    print "\n" if (($mhd::otrace) && ($imgsize%16==0));
        }
    print "\n" if ($mhd::otrace);
    # ordsum: 0x". $wordsum ."\n" if ($mhd::otrace);
	my $checksum = 0x12345678;
	close DATFILE;
	
	printf("-image size: %d, checksum: 0x%08x", $imgsize, $checksum) if ($mhd::otrace);
	
	$rc = sysseek(IMGFILE, 16, 0);
	die "error: ($rc) failed to seek in image to write header.\n" if ($rc != 16);

	# Write out the image size	
	my $imginfo1 = pack "V1", ($imgsize);
	$rc = syswrite(IMGFILE, $imginfo1, 4);
	die "error: ($rc) failed to write image size/checksum to image header.\n" if ($rc != 4);

	$rc = sysseek(IMGFILE, 28, 0);
	die "error: ($rc) failed to seek in image to write header.\n" if ($rc != 28);
	
	# Write out the image checksum 
	my $imginfo2 = pack "V1", ($checksum);
	$rc = syswrite(IMGFILE, $imginfo2, 4);
	die "error: ($rc) failed to write image size/checksum to image header.\n" if ($rc != 4);
  
	close IMGFILE; 	
	
    return 0;
}

# Writes the binary file specified as an argument with the content of this
# and contained feature flag and dsr objects.
sub WriteToFile
{
	my $self = shift;
	return undef unless(ref($self));
	my $file = shift;
	return undef unless(defined($file));
    open FILE, "> $file" or die "Couldn't open file '$file' for writing.\n";
	binmode FILE;
	print FILE $self->BinaryContent();
	
	close FILE;
	return 0;
}


# Create the binary equivalent of the internal data and return it as a
# string.
sub BinaryContent
{
	my $self = shift;
	return undef unless(ref($self));

    # Get the feature flag entries.. This is an array reference.
	# For each one append the binary representation of the information
	# contained.
	my $records = "";
    my $lsd = "";
   	my $ffs_ref = $self->SettingRecords();
	my $ff;

    my $count = 0;
	foreach $ff (@$ffs_ref)
	{
	    $count++;
	    printf("-encoding record: %04d (0x%08x:%04d)\n", $count, $ff->CUID(), $ff->EID());
		$records .= $ff->GetRecHdrBinary(length ($lsd));
	    my $stype = $ff->Type();
    	if (($stype & 0xffff0000) && ($ff->Length() > 0)) {
		    $lsd .= $ff->GetRecLsdBinary();
            }
	}

    $self->LsdOffset(32+length ($records));     # header size 32
    $self->LsdSize(length ($lsd));
    
	my $header = $self->CreateBinaryHeader();

	return $header . $records . $lsd;
}

# Return a reference to the 'feature flags' array.
sub SettingRecords
{
	my $self = shift;
	return undef unless(ref($self));
	return $self->{settingrecords};
}

# Add a Feature Flag object. Perhaps there should be code to check if we
# already know about this feature flag. (i.e check the uid against the ones
# we have).
sub AddSettingRecord
{
	my $self = shift;
	return undef unless(ref($self));
	my $arg = shift;
	die "panic: method 'AddSettingRecord' requires a 'HCRrec' object as argument.\n"
   	    unless(ref($arg) eq  "HCRrec");
   	
	push @{$self->SettingRecords()}, $arg;
	$self->NumRecords(1);
	
    return 0;
}


1;