os/persistentdata/featuremgmt/featuremgr/tools/datfilehelpers/FMCreate.pm
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
sl@0
     1
#!perl -w
sl@0
     2
# Copyright (c) 2007-2009 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     3
# All rights reserved.
sl@0
     4
# This component and the accompanying materials are made available
sl@0
     5
# under the terms of "Eclipse Public License v1.0"
sl@0
     6
# which accompanies this distribution, and is available
sl@0
     7
# at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     8
#
sl@0
     9
# Initial Contributors:
sl@0
    10
# Nokia Corporation - initial contribution.
sl@0
    11
#
sl@0
    12
# Contributors:
sl@0
    13
#
sl@0
    14
# Description:
sl@0
    15
#
sl@0
    16
sl@0
    17
use strict;
sl@0
    18
sl@0
    19
#
sl@0
    20
# Perl module to create and maintain feature manager data files.
sl@0
    21
# You can either set up the information programmatically or else load up
sl@0
    22
# information from a pre-existing feature data file and then modify it. You
sl@0
    23
# can also save the information to a file (in feature manager dataset format).
sl@0
    24
#
sl@0
    25
# This class maintains header information plus two arrays, one containing
sl@0
    26
# feature flag information and the other containing default supported range
sl@0
    27
# information. Those are themselves objects and have their own accessor
sl@0
    28
# methods.
sl@0
    29
#
sl@0
    30
sl@0
    31
package FMCreate;
sl@0
    32
sl@0
    33
use featureflag;
sl@0
    34
use featuredsr;
sl@0
    35
sl@0
    36
#
sl@0
    37
# n e w
sl@0
    38
#
sl@0
    39
# Create a new FMCreate object. For example 'my $fmc = FMCreate->new();
sl@0
    40
#
sl@0
    41
sub new
sl@0
    42
{
sl@0
    43
	my $arg = shift;
sl@0
    44
	my $class = ref($arg) || $arg;
sl@0
    45
	my $self = {
sl@0
    46
			        typefield => "feat",  # 4 bytes wide.
sl@0
    47
					fileversion => 1,     # 2 bytes.
sl@0
    48
					fileflags => 0,       # 2 bytes.
sl@0
    49
					numfeatures => 0,     # 4 bytes. (this is 'x')
sl@0
    50
					numdefuid => 0,       # 4 bytes. (this is 'y')
sl@0
    51
					endian => "LE",
sl@0
    52
					packprefix => "V",    # Changed with endian-ness.
sl@0
    53
					                      # Used to create binary strings.
sl@0
    54
sl@0
    55
					featureflags => [],   # There are x of these.
sl@0
    56
					dsrs => [],           # There are y of these.
sl@0
    57
	           };
sl@0
    58
	bless $self, $class;
sl@0
    59
	return $self;
sl@0
    60
}
sl@0
    61
sl@0
    62
# Print to STDOUT the header information, feature flags information and
sl@0
    63
# default supported range information.
sl@0
    64
sub ShowALL
sl@0
    65
{
sl@0
    66
	my $self = shift;
sl@0
    67
	return undef unless(ref($self));
sl@0
    68
	$self->ShowHeader();
sl@0
    69
	$self->ShowFeatureFlags();
sl@0
    70
	$self->ShowDSRs();
sl@0
    71
	return 1;
sl@0
    72
}
sl@0
    73
sl@0
    74
# Print to STDOUT the header information we have.
sl@0
    75
sub ShowHeader
sl@0
    76
{
sl@0
    77
	my $self = shift;
sl@0
    78
	return undef unless(ref($self));
sl@0
    79
sl@0
    80
	# Get header information..
sl@0
    81
	my $typefield = $self->TypeField();
sl@0
    82
	my $fileversion = $self->FileVersion();
sl@0
    83
	my $fileflags = $self->FileFlags();
sl@0
    84
	my $numfeatures = $self->NumFeatures();
sl@0
    85
	my $numdefuid = $self->NumDefUid();
sl@0
    86
sl@0
    87
	# Display it in English.
sl@0
    88
	print "TYPEFIELD: '$typefield'\n";
sl@0
    89
	print "FILEVERSION: '$fileversion'\n";
sl@0
    90
	print "FILEFLAGS: '$fileflags'\n";
sl@0
    91
	print "NUMFEATURES: '$numfeatures'\n";
sl@0
    92
	print "NUMDSRS: '$numdefuid'\n";
sl@0
    93
sl@0
    94
	return(1);
sl@0
    95
}
sl@0
    96
sl@0
    97
# Call the 'Show' method in each of the feature flag objects we
sl@0
    98
# have - this will print their content to STDOUT.
sl@0
    99
sub ShowFeatureFlags
sl@0
   100
{
sl@0
   101
	my $self = shift;
sl@0
   102
	return undef unless(ref($self));
sl@0
   103
	my $ffs = $self->FeatureFlags;
sl@0
   104
	return 1 unless(@$ffs);
sl@0
   105
sl@0
   106
	print "\nFeature Flags\n";
sl@0
   107
	print "=============\n";
sl@0
   108
	for my $ff (@$ffs)
sl@0
   109
	{
sl@0
   110
		$ff->Show();
sl@0
   111
	}
sl@0
   112
	return 1;
sl@0
   113
}
sl@0
   114
sl@0
   115
# Call the 'Show' method in each of the default supported range objects we
sl@0
   116
# have - this will print their content to STDOUT.
sl@0
   117
sub ShowDSRs
sl@0
   118
{
sl@0
   119
	my $self = shift;
sl@0
   120
	return undef unless(ref($self));
sl@0
   121
	my $fds = $self->FeatureDSRs;
sl@0
   122
	return 1 unless(@$fds);
sl@0
   123
	print "\nFeature DSRs\n";
sl@0
   124
	print "============\n";
sl@0
   125
	for my $fd (@$fds)
sl@0
   126
	{
sl@0
   127
		$fd->Show();
sl@0
   128
	}
sl@0
   129
	return 1;
sl@0
   130
}
sl@0
   131
sl@0
   132
# Get/Set the endian-ness we want. Changes the 'packprefix' member which is
sl@0
   133
# used in the creation of binary data.
sl@0
   134
sub Endian
sl@0
   135
{
sl@0
   136
	my $self = shift;
sl@0
   137
	return undef unless(ref($self));
sl@0
   138
	my $arg = shift;
sl@0
   139
	return $self->{endian} unless(defined($arg));
sl@0
   140
	if($arg =~ m/(LE|BE)/i)
sl@0
   141
	{
sl@0
   142
		my $endian = uc($1);
sl@0
   143
		$self->{endian} = $endian;
sl@0
   144
		# Used by 'pack' to generate binary strings.
sl@0
   145
		$self->{packprefix} = "V" if($endian eq "LE");
sl@0
   146
		$self->{packprefix} = "N" if($endian eq "BE");
sl@0
   147
	}
sl@0
   148
	return $self->{endian};
sl@0
   149
}
sl@0
   150
sl@0
   151
# This is 'feat'.
sl@0
   152
sub TypeField
sl@0
   153
{
sl@0
   154
	my $self = shift;
sl@0
   155
	return undef unless(ref($self));
sl@0
   156
	my $arg = shift;
sl@0
   157
	$self->{typefield} = $arg if(defined($arg));
sl@0
   158
	return $self->{typefield};
sl@0
   159
}
sl@0
   160
sl@0
   161
sub FileVersion
sl@0
   162
{
sl@0
   163
	my $self = shift;
sl@0
   164
	return undef unless(ref($self));
sl@0
   165
	my $arg = shift;
sl@0
   166
	# Should we be testing for a numeric value?
sl@0
   167
	$self->{fileversion} = $arg if(defined($arg));
sl@0
   168
	return $self->{fileversion};
sl@0
   169
}
sl@0
   170
sl@0
   171
sub FileFlags
sl@0
   172
{
sl@0
   173
	my $self = shift;
sl@0
   174
	return undef unless(ref($self));
sl@0
   175
	my $arg = shift;
sl@0
   176
	$self->{fileflags} = $arg if(defined($arg));
sl@0
   177
	return $self->{fileflags};
sl@0
   178
}
sl@0
   179
sl@0
   180
# How many feature flag objects have we got?
sl@0
   181
sub NumFeatures
sl@0
   182
{
sl@0
   183
	my $self = shift;
sl@0
   184
	return undef unless(ref($self));
sl@0
   185
	my $arg = shift;
sl@0
   186
	$self->{numfeatures} = $arg if(defined($arg));
sl@0
   187
	return $self->{numfeatures};
sl@0
   188
}
sl@0
   189
sl@0
   190
# How many default supported range objects have we got?
sl@0
   191
sub NumDefUid
sl@0
   192
{
sl@0
   193
	my $self = shift;
sl@0
   194
	return undef unless(ref($self));
sl@0
   195
	my $arg = shift;
sl@0
   196
	$self->{numdefuid} = $arg if(defined($arg));
sl@0
   197
	return $self->{numdefuid};
sl@0
   198
}
sl@0
   199
sl@0
   200
# Create a binary string containing the header information for the
sl@0
   201
# feature manager data file based on the various fields in this object.
sl@0
   202
sub CreateBinaryHeader
sl@0
   203
{
sl@0
   204
	my $self = shift;
sl@0
   205
	return undef unless(ref($self));
sl@0
   206
	my $hdrstring;
sl@0
   207
sl@0
   208
	# Get the letter for packing information with 'pack' into a binary form.
sl@0
   209
	my $pack16 = lc($self->{packprefix});
sl@0
   210
	my $pack32 = uc($self->{packprefix});
sl@0
   211
	
sl@0
   212
	# Get header information..
sl@0
   213
	my $typefield = $self->TypeField();
sl@0
   214
	my $fileversion = $self->FileVersion();
sl@0
   215
	my $fileflags = $self->FileFlags();
sl@0
   216
	my $numfeatures = $self->NumFeatures();
sl@0
   217
	my $numdefuid = $self->NumDefUid();
sl@0
   218
sl@0
   219
	# Write the 'type' field out. This is 'feat'. Would this be different on
sl@0
   220
	# big-endian systems?
sl@0
   221
	$hdrstring = $typefield;
sl@0
   222
sl@0
   223
	# Now the file version number. A 16-bit value.. Will this cause trouble
sl@0
   224
	# if the shifted value is signed?
sl@0
   225
	$hdrstring .= pack($pack16 . "1", $fileversion);
sl@0
   226
sl@0
   227
	# Now the file flags. Another 16-bit value..
sl@0
   228
	$hdrstring .= pack($pack16 . "1", $fileflags);
sl@0
   229
sl@0
   230
	# Now the number of listed features - a 32-bit value.
sl@0
   231
	$hdrstring .= pack($pack32 . "1", $numfeatures);
sl@0
   232
sl@0
   233
	# Now the number of listed features - a 32-bit value.
sl@0
   234
	$hdrstring .= pack($pack32 . "1", $numdefuid);
sl@0
   235
sl@0
   236
	return $hdrstring;
sl@0
   237
}
sl@0
   238
sl@0
   239
# Writes the binary file specified as an argument with the content of this
sl@0
   240
# and contained feature flag and dsr objects.
sl@0
   241
sub WriteToFile
sl@0
   242
{
sl@0
   243
	my $self = shift;
sl@0
   244
	return undef unless(ref($self));
sl@0
   245
	my $file = shift;
sl@0
   246
	return undef unless(defined($file));
sl@0
   247
    open FILE, "> $file" or die "Couldn't open file '$file' for writing.\n";
sl@0
   248
	binmode FILE;
sl@0
   249
	print FILE $self->BinaryContent();
sl@0
   250
	close FILE;
sl@0
   251
	return 1;
sl@0
   252
}
sl@0
   253
sl@0
   254
sl@0
   255
# Create the binary equivalent of the internal data and return it as a
sl@0
   256
# string.
sl@0
   257
sub BinaryContent
sl@0
   258
{
sl@0
   259
	my $self = shift;
sl@0
   260
	return undef unless(ref($self));
sl@0
   261
sl@0
   262
	# First get the header information for the registry manager data file.
sl@0
   263
	my $ret = $self->CreateBinaryHeader();
sl@0
   264
sl@0
   265
    # Get the feature flag entries.. This is an array reference.
sl@0
   266
	# For each one append the binary representation of the information
sl@0
   267
	# contained.
sl@0
   268
	my $ffs = $self->FeatureFlags;
sl@0
   269
	for my $ff (@$ffs)
sl@0
   270
	{
sl@0
   271
		$ret .= $ff->BinaryContent();
sl@0
   272
	}
sl@0
   273
sl@0
   274
    # Get the feature default supported range entries.. This is an array
sl@0
   275
	# reference too. For each one append the binary representation of 
sl@0
   276
	# uid range contained.
sl@0
   277
	my $fdsrs = $self->FeatureDSRs;
sl@0
   278
	for my $ff (@$fdsrs)
sl@0
   279
	{
sl@0
   280
		$ret .= $ff->BinaryContent();
sl@0
   281
	}
sl@0
   282
	return $ret;
sl@0
   283
}
sl@0
   284
sl@0
   285
# Return a reference to the 'feature flags' array.
sl@0
   286
sub FeatureFlags
sl@0
   287
{
sl@0
   288
	my $self = shift;
sl@0
   289
	return undef unless(ref($self));
sl@0
   290
	return $self->{featureflags};
sl@0
   291
}
sl@0
   292
sl@0
   293
# Add a Feature Flag object. Perhaps there should be code to check if we
sl@0
   294
# already know about this feature flag. (i.e check the uid against the ones
sl@0
   295
# we have).
sl@0
   296
sub AddFeatureFlag
sl@0
   297
{
sl@0
   298
	my $self = shift;
sl@0
   299
	return undef unless(ref($self));
sl@0
   300
	my $arg = shift;
sl@0
   301
	die "Method 'AddFeatureFlag' requires a 'FeatureFlag' object as argument.\n"
sl@0
   302
   	    unless(ref($arg) eq  "FeatureFlag");
sl@0
   303
	push @{$self->FeatureFlags()}, $arg;
sl@0
   304
	$self->NumFeatures($self->NumFeatures() + 1);
sl@0
   305
    return 1;
sl@0
   306
}
sl@0
   307
sl@0
   308
# Return a reference to the 'feature dsrs' array.
sl@0
   309
sub FeatureDSRs
sl@0
   310
{
sl@0
   311
	my $self = shift;
sl@0
   312
	return undef unless(ref($self));
sl@0
   313
	return $self->{dsrs};
sl@0
   314
}
sl@0
   315
sl@0
   316
# Add a Feature 'Default Support Range' object.
sl@0
   317
sub AddFeatureDSR
sl@0
   318
{
sl@0
   319
	my $self = shift;
sl@0
   320
	return undef unless(ref($self));
sl@0
   321
	my $arg = shift;
sl@0
   322
	die "Method 'AddFeatureDSR' requires a 'FeatureDSR' object as argument.\n"
sl@0
   323
   	    unless(ref($arg) eq  "FeatureDSR");
sl@0
   324
	push @{$self->FeatureDSRs()}, $arg;
sl@0
   325
	$self->NumDefUid($self->NumDefUid() + 1);
sl@0
   326
    return 1;
sl@0
   327
}
sl@0
   328
sl@0
   329
# This method loads up it's information from an existing feature manager
sl@0
   330
# data file. This will die if it thinks there is something wrong with the file.
sl@0
   331
sub LoadUp
sl@0
   332
{
sl@0
   333
	my $self = shift;
sl@0
   334
	return undef unless(ref($self) eq "FMCreate");
sl@0
   335
sl@0
   336
	my $packprefix16 = lc($self->{packprefix});
sl@0
   337
	my $packprefix32 = uc($self->{packprefix});
sl@0
   338
sl@0
   339
	my $file = shift;
sl@0
   340
	return undef unless(defined($file) and -f $file);
sl@0
   341
	open FILE, $file or die "Couldn't open '$file'\n";
sl@0
   342
	binmode FILE;
sl@0
   343
	my ($tmp, $feat);
sl@0
   344
sl@0
   345
	# First get the file size.
sl@0
   346
	my $fsz = sysseek(FILE, 0, 2);
sl@0
   347
	sysseek(FILE, 0, 0);
sl@0
   348
sl@0
   349
	# Read the 'feat' marker from the top of the file. Check it.
sl@0
   350
	die "Unable to read first 4 bytes from '$file'"
sl@0
   351
	    unless(4 == sysread(FILE, $feat, 4) );
sl@0
   352
	die "First four bytes of '$file' do not contain 'feat'"
sl@0
   353
	    unless($feat eq "feat");
sl@0
   354
	$self->TypeField($feat); # Pointless. It's set to that anyway.
sl@0
   355
sl@0
   356
	# Read the file version number.
sl@0
   357
	die "Unable to read two bytes from index 4 from '$file'"
sl@0
   358
	    unless(2 == sysread(FILE, $tmp, 2) );
sl@0
   359
	my $filever = unpack( $packprefix16, $tmp );
sl@0
   360
	$self->FileVersion($filever);
sl@0
   361
sl@0
   362
	# Read the file flags.
sl@0
   363
	die "Unable to read two bytes from index 6 from '$file'"
sl@0
   364
	    unless(2 == sysread(FILE, $tmp, 2) );
sl@0
   365
	my $fileflags = unpack( $packprefix16, $tmp );
sl@0
   366
	$self->FileFlags($fileflags);
sl@0
   367
sl@0
   368
	# Read the number of features. Don't do anything with this yet..
sl@0
   369
	die "Unable to read four bytes from index 8 from '$file'"
sl@0
   370
	    unless(4 == sysread(FILE, $tmp, 4) );
sl@0
   371
	my $nfeat = unpack( $packprefix32, $tmp );
sl@0
   372
sl@0
   373
	# Read the number of DSRs. Don't do anything with this yet..
sl@0
   374
	die "Unable to read four bytes from index 12 from '$file'"
sl@0
   375
	    unless(4 == sysread(FILE, $tmp, 4) );
sl@0
   376
	my $ndsr = unpack( $packprefix32, $tmp );
sl@0
   377
sl@0
   378
	# Forget it if the filesize is clearly wrong.
sl@0
   379
	my $expsz = 16 + 12*$nfeat + 8*$ndsr;
sl@0
   380
	die "The file '$file' is $fsz bytes long, but the content suggests it should be $expsz bytes long. NFeatures: $nfeat   NDSRs: $ndsr\n" unless($expsz == $fsz);
sl@0
   381
sl@0
   382
	# Now read in the feature flags.
sl@0
   383
	my $offset = 16;
sl@0
   384
	for(my $ff=0 ; $ff<$nfeat ; $ff++)
sl@0
   385
	{
sl@0
   386
		# Get the UID.
sl@0
   387
	    die "Unable to read four bytes (uid) from index $offset from '$file'"
sl@0
   388
	        unless(4 == sysread(FILE, $tmp, 4) );
sl@0
   389
		$offset += 4;
sl@0
   390
		my $uid = unpack( $packprefix32, $tmp );
sl@0
   391
sl@0
   392
		# Get the status word.
sl@0
   393
	    die "Unable to read four bytes (sw) from index $offset from '$file'"
sl@0
   394
	        unless(4 == sysread(FILE, $tmp, 4) );
sl@0
   395
		$offset += 4;
sl@0
   396
		my $sw = unpack( $packprefix32, $tmp );
sl@0
   397
sl@0
   398
		# Get the user data word.
sl@0
   399
	    die "Unable to read four bytes (udw) from index $offset from '$file'"
sl@0
   400
	        unless(4 == sysread(FILE, $tmp, 4) );
sl@0
   401
		$offset += 4;
sl@0
   402
		my $ud = unpack( $packprefix32, $tmp );
sl@0
   403
sl@0
   404
		my $featflag = FeatureFlag->new($uid, $sw, $ud);
sl@0
   405
		die "Couldn't create a feature flag object!\n" unless(ref($featflag));
sl@0
   406
		$self->AddFeatureFlag($featflag);
sl@0
   407
	}
sl@0
   408
sl@0
   409
	# Now read in the DSRs.
sl@0
   410
	for( my $dsr=0 ; $dsr<$ndsr ; $dsr++ )
sl@0
   411
	{
sl@0
   412
		# Get the low UID.
sl@0
   413
	    die "Unable to read four bytes (lowuid) from index $offset from '$file'"
sl@0
   414
	        unless(4 == sysread(FILE, $tmp, 4) );
sl@0
   415
		$offset += 4;
sl@0
   416
		my $lowuid = unpack( $packprefix32, $tmp );
sl@0
   417
sl@0
   418
		# Get the high UID.
sl@0
   419
	    die "Can't read four bytes (high uid) from index $offset from '$file'"
sl@0
   420
	        unless(4 == sysread(FILE, $tmp, 4) );
sl@0
   421
		$offset += 4;
sl@0
   422
		my $highuid = unpack( $packprefix32, $tmp );
sl@0
   423
sl@0
   424
		my $fd = FeatureDSR->new($lowuid, $highuid);
sl@0
   425
		die "Couldn't create 'FeatureDSR' object!\n" unless(ref($fd));
sl@0
   426
		$self->AddFeatureDSR($fd);
sl@0
   427
	}
sl@0
   428
sl@0
   429
	# Check if our calculated file offset matches the end of the file.
sl@0
   430
	# This is pointless actually, because we've already checked the file
sl@0
   431
	# size..
sl@0
   432
	my $fileoffset = sysseek(FILE, 0, 2);
sl@0
   433
	die "End of file offset ($fileoffset) does not match end of DSRs!\n"
sl@0
   434
	    unless($fileoffset == $offset);
sl@0
   435
sl@0
   436
	close FILE;
sl@0
   437
	return 1;
sl@0
   438
}
sl@0
   439
sl@0
   440
# Remove the feature flag object specified by UID held in this object (in
sl@0
   441
# the internal 'featureflags' array). Returns 1 on success, undef otherwise.
sl@0
   442
sub RemoveFeatureFlagByUID
sl@0
   443
{
sl@0
   444
	my $self = shift;
sl@0
   445
	return undef unless(ref($self));
sl@0
   446
	my $arg = shift;
sl@0
   447
	return undef unless(defined($arg));
sl@0
   448
	my $ffs = $self->FeatureFlags;
sl@0
   449
sl@0
   450
	my $idx = 0;
sl@0
   451
	for my $ff (@$ffs)
sl@0
   452
	{
sl@0
   453
		if($ff->UID() == $arg)
sl@0
   454
		{
sl@0
   455
			splice(@$ffs, $idx, 1);
sl@0
   456
			$self->NumFeatures($self->NumFeatures() - 1);
sl@0
   457
			return 1;
sl@0
   458
		}
sl@0
   459
		$idx++;
sl@0
   460
	}
sl@0
   461
	return undef;
sl@0
   462
}
sl@0
   463
sl@0
   464
# Return a reference to the 'FeatureFlag' object held in this object (in
sl@0
   465
# the internal 'featureflags' array) with the uid specified as an
sl@0
   466
# argument. This returns a reference so it's still in this object on return,
sl@0
   467
# you can modify it and then write out (for example) the data file.
sl@0
   468
sub GetFeatureFlagByUID
sl@0
   469
{
sl@0
   470
	my $self = shift;
sl@0
   471
	return undef unless(ref($self));
sl@0
   472
	my $arg = shift;
sl@0
   473
	return undef unless(defined($arg));
sl@0
   474
	my $ffs = $self->FeatureFlags;
sl@0
   475
	for my $ff (@$ffs)
sl@0
   476
	{
sl@0
   477
		return $ff if($ff->UID() == $arg);
sl@0
   478
	}
sl@0
   479
	return undef;
sl@0
   480
}
sl@0
   481
sl@0
   482
sl@0
   483
# Remove the feature DSR object specified by UIDs held in this object (in
sl@0
   484
# the internal 'dsrs' array). Returns 1 on success, undef otherwise.
sl@0
   485
sub RemoveDSRByUIDs
sl@0
   486
{
sl@0
   487
	my $self = shift;
sl@0
   488
	return undef unless(ref($self));
sl@0
   489
	my($lowuid, $highuid) = @_;
sl@0
   490
	return undef unless(defined($lowuid) and defined($highuid));
sl@0
   491
	my $fdsrs = $self->FeatureDSRs;
sl@0
   492
sl@0
   493
	my $idx = 0;
sl@0
   494
	for my $fdsr (@$fdsrs)
sl@0
   495
	{
sl@0
   496
		if( ($fdsrs->LowUID() == $lowuid) and
sl@0
   497
            ($fdsrs->HighUID() == $highuid) )
sl@0
   498
		{
sl@0
   499
			splice(@$fdsrs, $idx, 1);
sl@0
   500
			$self->NumDefUid($self->NumDefUid() - 1);
sl@0
   501
			return 1;
sl@0
   502
		}
sl@0
   503
		$idx++;
sl@0
   504
	}
sl@0
   505
	return undef;
sl@0
   506
}
sl@0
   507
sl@0
   508
# Return a reference to the 'FeatureDSR' object held in this object (in
sl@0
   509
# the internal 'dsrs' array) with the low and high uids specified in the
sl@0
   510
# arguments. This returns a reference so it's still in this object on return,
sl@0
   511
# you can modify it (by changing the uids) and then write out (for example)
sl@0
   512
# the data file.
sl@0
   513
sub GetDSRByUIDs
sl@0
   514
{
sl@0
   515
	my $self = shift;
sl@0
   516
	return undef unless(ref($self));
sl@0
   517
	my($lowuid, $highuid) = @_;
sl@0
   518
	return undef unless(defined($lowuid) and defined($highuid));
sl@0
   519
	my $fdsrs = $self->FeatureDSRs;
sl@0
   520
	for my $fdsr (@$fdsrs)
sl@0
   521
	{
sl@0
   522
		return $fdsr if( ($fdsr->LowUID() == $lowuid) and
sl@0
   523
                         ($fdsr->HighUID() == $highuid));
sl@0
   524
	}
sl@0
   525
	return undef;
sl@0
   526
}
sl@0
   527
sl@0
   528
1;
sl@0
   529