package resize_fat::directory;

use diagnostics;
use strict;

use common qw(:system);
use resize_fat::dir_entry;
use resize_fat::io;


my $format = "a8 a3 C C C S7 I";
my @fields = (
    'name',
    'extension',
    'attributes',
    'is_upper_case_name',
    'creation_time_low',	# milliseconds
    'creation_time_high',
    'creation_date',
    'access_date',
    'first_cluster_high',	# for FAT32
    'time',
    'date',
    'first_cluster',
    'length',
);

1;

sub entry_size { psizeof($format) }

# call `f' for each entry of the directory
# if f return true, then modification in the entry are taken back
sub traverse($$$) {
    my ($fs, $directory, $f) = @_;

    for (my $i = 0;; $i++) {
	my $raw = \substr($directory, $i * psizeof($format), psizeof($format));

	# empty entry means end of directory
	$$raw =~ /^\0*$/ and return $directory;

	my $entry; @{$entry}{@fields} = unpack $format, $$raw;

	&$f($entry) 
	    and	$$raw = pack $format, @{$entry}{@fields};
    }
    $directory;
}

sub traverse_all($$) {
    my ($fs, $f) = @_;

    my $traverse_all; $traverse_all = sub {
	my ($entry) = @_;

	&$f($entry);

        resize_fat::dir_entry::is_directory($entry) 
	    and traverse($fs, resize_fat::io::read_file($fs, resize_fat::dir_entry::get_cluster($entry)), $traverse_all);

	undef; # no need to write back (cf traverse)
    };

    my $directory = $resize_fat::isFAT32 ?
	resize_fat::io::read_file($fs, $fs->{fat32_root_dir_cluster}) :
	resize_fat::io::read($fs, $fs->{root_dir_offset}, $fs->{root_dir_size});
    traverse($fs, $directory, $traverse_all);
}


# function used by construct_dir_tree to translate the `cluster' fields in each
# directory entry
sub remap {
    my ($fs, $directory) = @_;

    traverse($fs->{fat_remap}, $directory, sub { resize_fat::dir_entry::remap($fs->{fat_remap}, $_[0]) });
}