package partition_table_bsd; # $Id$
use diagnostics;
use strict;
use vars qw(@ISA);
@ISA = qw(partition_table_raw);
use common qw(:common :system :file :functional);
use partition_table_raw;
use partition_table;
use c;
#- very bad and rough handling :(
my %typeToDos = (
8 => 0x83,
1 => 0x82,
);
my %typeFromDos = reverse %typeToDos;
my ($main_format, $main_fields) = list2kv(
I => 'magic',
S => 'type',
S => 'subtype',
a16 => 'typename',
a16 => 'packname',
I => 'secsize',
I => 'nsectors',
I => 'ntracks',
I => 'ncylinders',
I => 'secpercyl',
I => 'secprtunit',
S => 'sparespertrack',
S => 'sparespercyl',
I => 'acylinders',
S => 'rpm',
S => 'interleave',
S => 'trackskew',
S => 'cylskew',
I => 'headswitch',
I => 'trkseek',
I => 'flags',
a20 => 'drivedata',
a20 => 'spare',
I => 'magic2',
S => 'checksum',
S => 'npartitions',
I => 'bbsize',
I => 'sbsize',
a128=> 'partitions',
a236=> 'blank',
);
$main_format = join '', @$main_format;
my @fields = qw(size start fsize type frag cpg);
my $format = "I I I C C S";
my $magic = 0x82564557;
my $nb_primary = 8;
my $offset = 0x40;
sub read($$) {
my ($hd, $sector) = @_;
my $tmp;
local *F; partition_table_raw::openit($hd, *F) or die "failed to open device";
c::lseek_sector(fileno(F), $sector, $offset) or die "reading of partition in sector $sector failed";
sysread F, $tmp, psizeof($main_format) or die "error while reading partition table in sector $sector";
my %info; @info{@$main_fields} = unpack $main_format, $tmp;
#- TODO verify checksum
my $size = psizeof($format);
my @pt = map {
my %h; @h{@fields} = unpack $format, $_;
$h{type} = $typeToDos{$h{type}} || $h{type};
\%h;
} $info{partitions} =~ /(.{$size})/g;
#- check magic number
$info{magic} == $magic or die "bad magic number";
$info{magic2} == $magic or die "bad magic number";
[ @pt ], \%info;
}
# write the partition table (and extended ones)
# for each entry, it uses fields: start, size, type, active
sub write($$$;$) {
my ($hd, $sector, $pt, $info) = @_;
#- handle testing for writing partition table on file only!
local *F;
if ($::testing) {
my $file = "/tmp/partition_table_$hd->{device}";
open F, ">$file" or die "error opening test file $file";
} else {
partition_table_raw::openit($hd, *F, 2) or die "error opening device $hd->{device} for writing";
c::lseek_sector(fileno(F), $sector, $offset) or return 0;
}
#- TODO compute checksum
$info->{npartitions} = $nb_primary; #- is it ok?
@$pt == $nb_primary or die "partition table does not have $nb_primary entries";
$info->{partitions} = join '', map {
local $_->{type} = $typeFromDos{$_->{type}} || $_->{type};
pack $format, @$_{@fields};
} @$pt;
syswrite F, pack($main_format, @$info{@$main_fields}), psizeof($main_format) or return 0;
1;
}
sub info {
my ($hd) = @_;
my $dtype_scsi = 4; #- taken from fdisk, removed unused one,
my $dtype_ST506 = 6; #- see fdisk for more
{
magic => $magic,
magic2 => $magic,
dtype => $hd->{device} =~ /^sd/ ? $dtype_scsi : $dtype_ST506,
secsize => $common::SECTORSIZE,
ncylinders => $hd->{geom}{cylinders},
secpercyl => $hd->cylinder_size(),
secprtunit => $hd->{geom}{totalsectors},
rpm => 3600,
interleave => 1,
trackskew => 0,
cylskew => 0,
headswitch => 0,
trkseek => 0,
bbsize => 8192, #- size of boot area, with label
sbsize => 8192, #- max size of fs superblock
};
}
sub clear_raw {
my ($hd) = @_;
{ raw => [ ({}) x $nb_primary ], info => info($hd) };
}
1;
r/animtim/designWork'>user/animtim/designWork
Mageia Installer and base platform for many utilities