Update contrib.
2 # Copyright (c) 2007-2009 Nokia Corporation and/or its subsidiary(-ies).
4 # This component and the accompanying materials are made available
5 # under the terms of "Eclipse Public License v1.0"
6 # which accompanies this distribution, and is available
7 # at the URL "http://www.eclipse.org/legal/epl-v10.html".
9 # Initial Contributors:
10 # Nokia Corporation - initial contribution.
20 # Perl module to create and maintain feature manager data files.
21 # You can either set up the information programmatically or else load up
22 # information from a pre-existing feature data file and then modify it. You
23 # can also save the information to a file (in feature manager dataset format).
25 # This class maintains header information plus two arrays, one containing
26 # feature flag information and the other containing default supported range
27 # information. Those are themselves objects and have their own accessor
39 # Create a new FMCreate object. For example 'my $fmc = FMCreate->new();
44 my $class = ref($arg) || $arg;
46 typefield => "feat", # 4 bytes wide.
47 fileversion => 1, # 2 bytes.
48 fileflags => 0, # 2 bytes.
49 numfeatures => 0, # 4 bytes. (this is 'x')
50 numdefuid => 0, # 4 bytes. (this is 'y')
52 packprefix => "V", # Changed with endian-ness.
53 # Used to create binary strings.
55 featureflags => [], # There are x of these.
56 dsrs => [], # There are y of these.
62 # Print to STDOUT the header information, feature flags information and
63 # default supported range information.
67 return undef unless(ref($self));
69 $self->ShowFeatureFlags();
74 # Print to STDOUT the header information we have.
78 return undef unless(ref($self));
80 # Get header information..
81 my $typefield = $self->TypeField();
82 my $fileversion = $self->FileVersion();
83 my $fileflags = $self->FileFlags();
84 my $numfeatures = $self->NumFeatures();
85 my $numdefuid = $self->NumDefUid();
87 # Display it in English.
88 print "TYPEFIELD: '$typefield'\n";
89 print "FILEVERSION: '$fileversion'\n";
90 print "FILEFLAGS: '$fileflags'\n";
91 print "NUMFEATURES: '$numfeatures'\n";
92 print "NUMDSRS: '$numdefuid'\n";
97 # Call the 'Show' method in each of the feature flag objects we
98 # have - this will print their content to STDOUT.
102 return undef unless(ref($self));
103 my $ffs = $self->FeatureFlags;
104 return 1 unless(@$ffs);
106 print "\nFeature Flags\n";
107 print "=============\n";
115 # Call the 'Show' method in each of the default supported range objects we
116 # have - this will print their content to STDOUT.
120 return undef unless(ref($self));
121 my $fds = $self->FeatureDSRs;
122 return 1 unless(@$fds);
123 print "\nFeature DSRs\n";
124 print "============\n";
132 # Get/Set the endian-ness we want. Changes the 'packprefix' member which is
133 # used in the creation of binary data.
137 return undef unless(ref($self));
139 return $self->{endian} unless(defined($arg));
140 if($arg =~ m/(LE|BE)/i)
143 $self->{endian} = $endian;
144 # Used by 'pack' to generate binary strings.
145 $self->{packprefix} = "V" if($endian eq "LE");
146 $self->{packprefix} = "N" if($endian eq "BE");
148 return $self->{endian};
155 return undef unless(ref($self));
157 $self->{typefield} = $arg if(defined($arg));
158 return $self->{typefield};
164 return undef unless(ref($self));
166 # Should we be testing for a numeric value?
167 $self->{fileversion} = $arg if(defined($arg));
168 return $self->{fileversion};
174 return undef unless(ref($self));
176 $self->{fileflags} = $arg if(defined($arg));
177 return $self->{fileflags};
180 # How many feature flag objects have we got?
184 return undef unless(ref($self));
186 $self->{numfeatures} = $arg if(defined($arg));
187 return $self->{numfeatures};
190 # How many default supported range objects have we got?
194 return undef unless(ref($self));
196 $self->{numdefuid} = $arg if(defined($arg));
197 return $self->{numdefuid};
200 # Create a binary string containing the header information for the
201 # feature manager data file based on the various fields in this object.
202 sub CreateBinaryHeader
205 return undef unless(ref($self));
208 # Get the letter for packing information with 'pack' into a binary form.
209 my $pack16 = lc($self->{packprefix});
210 my $pack32 = uc($self->{packprefix});
212 # Get header information..
213 my $typefield = $self->TypeField();
214 my $fileversion = $self->FileVersion();
215 my $fileflags = $self->FileFlags();
216 my $numfeatures = $self->NumFeatures();
217 my $numdefuid = $self->NumDefUid();
219 # Write the 'type' field out. This is 'feat'. Would this be different on
220 # big-endian systems?
221 $hdrstring = $typefield;
223 # Now the file version number. A 16-bit value.. Will this cause trouble
224 # if the shifted value is signed?
225 $hdrstring .= pack($pack16 . "1", $fileversion);
227 # Now the file flags. Another 16-bit value..
228 $hdrstring .= pack($pack16 . "1", $fileflags);
230 # Now the number of listed features - a 32-bit value.
231 $hdrstring .= pack($pack32 . "1", $numfeatures);
233 # Now the number of listed features - a 32-bit value.
234 $hdrstring .= pack($pack32 . "1", $numdefuid);
239 # Writes the binary file specified as an argument with the content of this
240 # and contained feature flag and dsr objects.
244 return undef unless(ref($self));
246 return undef unless(defined($file));
247 open FILE, "> $file" or die "Couldn't open file '$file' for writing.\n";
249 print FILE $self->BinaryContent();
255 # Create the binary equivalent of the internal data and return it as a
260 return undef unless(ref($self));
262 # First get the header information for the registry manager data file.
263 my $ret = $self->CreateBinaryHeader();
265 # Get the feature flag entries.. This is an array reference.
266 # For each one append the binary representation of the information
268 my $ffs = $self->FeatureFlags;
271 $ret .= $ff->BinaryContent();
274 # Get the feature default supported range entries.. This is an array
275 # reference too. For each one append the binary representation of
276 # uid range contained.
277 my $fdsrs = $self->FeatureDSRs;
280 $ret .= $ff->BinaryContent();
285 # Return a reference to the 'feature flags' array.
289 return undef unless(ref($self));
290 return $self->{featureflags};
293 # Add a Feature Flag object. Perhaps there should be code to check if we
294 # already know about this feature flag. (i.e check the uid against the ones
299 return undef unless(ref($self));
301 die "Method 'AddFeatureFlag' requires a 'FeatureFlag' object as argument.\n"
302 unless(ref($arg) eq "FeatureFlag");
303 push @{$self->FeatureFlags()}, $arg;
304 $self->NumFeatures($self->NumFeatures() + 1);
308 # Return a reference to the 'feature dsrs' array.
312 return undef unless(ref($self));
313 return $self->{dsrs};
316 # Add a Feature 'Default Support Range' object.
320 return undef unless(ref($self));
322 die "Method 'AddFeatureDSR' requires a 'FeatureDSR' object as argument.\n"
323 unless(ref($arg) eq "FeatureDSR");
324 push @{$self->FeatureDSRs()}, $arg;
325 $self->NumDefUid($self->NumDefUid() + 1);
329 # This method loads up it's information from an existing feature manager
330 # data file. This will die if it thinks there is something wrong with the file.
334 return undef unless(ref($self) eq "FMCreate");
336 my $packprefix16 = lc($self->{packprefix});
337 my $packprefix32 = uc($self->{packprefix});
340 return undef unless(defined($file) and -f $file);
341 open FILE, $file or die "Couldn't open '$file'\n";
345 # First get the file size.
346 my $fsz = sysseek(FILE, 0, 2);
349 # Read the 'feat' marker from the top of the file. Check it.
350 die "Unable to read first 4 bytes from '$file'"
351 unless(4 == sysread(FILE, $feat, 4) );
352 die "First four bytes of '$file' do not contain 'feat'"
353 unless($feat eq "feat");
354 $self->TypeField($feat); # Pointless. It's set to that anyway.
356 # Read the file version number.
357 die "Unable to read two bytes from index 4 from '$file'"
358 unless(2 == sysread(FILE, $tmp, 2) );
359 my $filever = unpack( $packprefix16, $tmp );
360 $self->FileVersion($filever);
362 # Read the file flags.
363 die "Unable to read two bytes from index 6 from '$file'"
364 unless(2 == sysread(FILE, $tmp, 2) );
365 my $fileflags = unpack( $packprefix16, $tmp );
366 $self->FileFlags($fileflags);
368 # Read the number of features. Don't do anything with this yet..
369 die "Unable to read four bytes from index 8 from '$file'"
370 unless(4 == sysread(FILE, $tmp, 4) );
371 my $nfeat = unpack( $packprefix32, $tmp );
373 # Read the number of DSRs. Don't do anything with this yet..
374 die "Unable to read four bytes from index 12 from '$file'"
375 unless(4 == sysread(FILE, $tmp, 4) );
376 my $ndsr = unpack( $packprefix32, $tmp );
378 # Forget it if the filesize is clearly wrong.
379 my $expsz = 16 + 12*$nfeat + 8*$ndsr;
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);
382 # Now read in the feature flags.
384 for(my $ff=0 ; $ff<$nfeat ; $ff++)
387 die "Unable to read four bytes (uid) from index $offset from '$file'"
388 unless(4 == sysread(FILE, $tmp, 4) );
390 my $uid = unpack( $packprefix32, $tmp );
392 # Get the status word.
393 die "Unable to read four bytes (sw) from index $offset from '$file'"
394 unless(4 == sysread(FILE, $tmp, 4) );
396 my $sw = unpack( $packprefix32, $tmp );
398 # Get the user data word.
399 die "Unable to read four bytes (udw) from index $offset from '$file'"
400 unless(4 == sysread(FILE, $tmp, 4) );
402 my $ud = unpack( $packprefix32, $tmp );
404 my $featflag = FeatureFlag->new($uid, $sw, $ud);
405 die "Couldn't create a feature flag object!\n" unless(ref($featflag));
406 $self->AddFeatureFlag($featflag);
409 # Now read in the DSRs.
410 for( my $dsr=0 ; $dsr<$ndsr ; $dsr++ )
413 die "Unable to read four bytes (lowuid) from index $offset from '$file'"
414 unless(4 == sysread(FILE, $tmp, 4) );
416 my $lowuid = unpack( $packprefix32, $tmp );
419 die "Can't read four bytes (high uid) from index $offset from '$file'"
420 unless(4 == sysread(FILE, $tmp, 4) );
422 my $highuid = unpack( $packprefix32, $tmp );
424 my $fd = FeatureDSR->new($lowuid, $highuid);
425 die "Couldn't create 'FeatureDSR' object!\n" unless(ref($fd));
426 $self->AddFeatureDSR($fd);
429 # Check if our calculated file offset matches the end of the file.
430 # This is pointless actually, because we've already checked the file
432 my $fileoffset = sysseek(FILE, 0, 2);
433 die "End of file offset ($fileoffset) does not match end of DSRs!\n"
434 unless($fileoffset == $offset);
440 # Remove the feature flag object specified by UID held in this object (in
441 # the internal 'featureflags' array). Returns 1 on success, undef otherwise.
442 sub RemoveFeatureFlagByUID
445 return undef unless(ref($self));
447 return undef unless(defined($arg));
448 my $ffs = $self->FeatureFlags;
453 if($ff->UID() == $arg)
455 splice(@$ffs, $idx, 1);
456 $self->NumFeatures($self->NumFeatures() - 1);
464 # Return a reference to the 'FeatureFlag' object held in this object (in
465 # the internal 'featureflags' array) with the uid specified as an
466 # argument. This returns a reference so it's still in this object on return,
467 # you can modify it and then write out (for example) the data file.
468 sub GetFeatureFlagByUID
471 return undef unless(ref($self));
473 return undef unless(defined($arg));
474 my $ffs = $self->FeatureFlags;
477 return $ff if($ff->UID() == $arg);
483 # Remove the feature DSR object specified by UIDs held in this object (in
484 # the internal 'dsrs' array). Returns 1 on success, undef otherwise.
488 return undef unless(ref($self));
489 my($lowuid, $highuid) = @_;
490 return undef unless(defined($lowuid) and defined($highuid));
491 my $fdsrs = $self->FeatureDSRs;
494 for my $fdsr (@$fdsrs)
496 if( ($fdsrs->LowUID() == $lowuid) and
497 ($fdsrs->HighUID() == $highuid) )
499 splice(@$fdsrs, $idx, 1);
500 $self->NumDefUid($self->NumDefUid() - 1);
508 # Return a reference to the 'FeatureDSR' object held in this object (in
509 # the internal 'dsrs' array) with the low and high uids specified in the
510 # arguments. This returns a reference so it's still in this object on return,
511 # you can modify it (by changing the uids) and then write out (for example)
516 return undef unless(ref($self));
517 my($lowuid, $highuid) = @_;
518 return undef unless(defined($lowuid) and defined($highuid));
519 my $fdsrs = $self->FeatureDSRs;
520 for my $fdsr (@$fdsrs)
522 return $fdsr if( ($fdsr->LowUID() == $lowuid) and
523 ($fdsr->HighUID() == $highuid));