On 8/4/09, Otto Moerbeek <o...@drijf.net> wrote:
> On Tue, Jul 28, 2009 at 03:26:08PM +1000, leon zadorin wrote:
>
>> That's all I am saying. Feel free to ignore or make "blah blah blah"
>> noises :-)
>>
>> So now we can, perhaps, get back (if at all) to the man pages and what
>> they are implying wrt original question.
>>
>> Leon.
>

Hello Otto,

Firstly I'd like to thank you for replying with actually interesting
and constructive comments.

Secondly, I hope you don't mind, I'd like to address your post in an
"out of order" fashion. So... starting from the source-code of the
fsck_ffs...

>
> You might want to check the source of fsck_ffs. It contains code to
> locate alternate superblocks, and that code will not work for the 'c'
> partition.
>

I'm starting by looking in setup.c (in sbin/fsck_ffs). Namely the
implementation of 'calcsb' call, and it appears to work fine even if
'c' partition is passed to it, e.g. "/dev/rsvnd3c"

the line
'lp = getdisklabel(dev, devfd)'
produces disklabel fine even if there is no disk-label installed (in
coherence with man 5 disklabel)...

the line
'pp = &lp->d_partitions[*cp - 'a'];'
evaluates to 2 ('c' - 'a') and indeed the default disklabel has no
less than value of 3 for 'd_npartitions' so accessing the 3rd array
element for (c) partition info appears to be ok...

the above observations have been tested with a minor program (see
attachment or at the end of message).

Perhaps, *indeed*, I am not looking in *all* of the right places and
so in the meantime (as I will be looking more into the rest of the
fsck_ffs code when I get more time), I thought I'd hack up a quick
empirical scenario: svnd an image, plonk newfs on c partition then
clobber the starting 16k and see if fsck firstly detects bad
superblock and then recovers ok...

... and the tests appear to indicate that it does actually work.

I have attached the relevant files as a .tar attachment as well as
pasting at the end of the email (in case of attachments being stripped
at your end -- but mail agent may wrap badly...).

> I'm just back from vacation, so I'm late to jump in. The answer krw@
> has given is right. We (and as a consequence the kernel) take the
> liberty to change fields in disklabel, especially the entry for the
> 'c' partition.

If the above paragraph appears to relate to future code behavior --
then I'm fine with this as I have no expectations (nor requirements)
for future-compatibility w.r.t. disklabel (at least w.r.t. the context
of my original post). In other words, I was only interested on knowing
what happened on that particular version/release of the code and how
it correlated with the assertions/explanations in man pages (e.g.
vnconfig, caveats section)

> As for the slighly off-topic part of the thread: if we write a
> guideline into our manual page, it has a good reason to be there.
> Consider the man page to be your much needed authorative source of
> information.

Sure, but if there are multiple sources of information (multiple man
pages, code itself) then all should be in coherence w.r.t. each other.

So if man vnconfig (caveats section) gives certain
explanations/examples (e.g. implying the need for disklabel being
present on disk) then such should be reproducible and in accord with
other man pages (e.g. man 5 disklabel).

Otherwise (e.g. in case whet/if a normative point of reference for
behavior is not supported by other docs) there is not only a case for
starting ambiguity but also for a misleading conclusion.

I guess, if one definitively decided to reserve the 'c' partition for
special purposes, then instead of vnconfig caveat's section going into
detailed explanations/assertions as to why a given use of 'c'
partition would *not* work point blank (w/o testing such assertions)
it would rather be more succinct to state something like: "The 'c'
partition is reserved for non file-system use only. Consequently the
results of applying a file-system directly onto 'c' partition are
undefined." ... the 'undefined' being the keyword here (as opposed to
stating that it will *not* work, when in some cases it appears to work
just fine) -- thus allowing for practical observations to vary w/o any
point of contention w.r.t. man pages assertions.

It'd be, possibly, also a good idea to reduce the duplication of such
info and move the whole section into a disklabel man page.

Of course, it'd be even a finer point to exemplify elsewhere (not man
pages even) the examples of why/when the 'c' partition is practically
(or theoretically) likely to change on-the-fly (i.e. compare how the
"entire *physical* disk" representation may change arbitrarily)... but
that is more of a wish-list.

Anyway, here are the quick test hacks/files:

the practical test (done on OpenBSD zion 4.5 GENERIC#1749 i386,
Intel(R) Core(TM)2 Duo CPU E6550 @ 2.33GHz)

the shell script:

#!/bin/sh

function bail
{
        echo "Colud not ${*}."
        exit 1
}

echo "Setting up the test scenario before corrupting"

dd if=/dev/zero of=./image bs=1m count=77 || bail "dd initial image"
vnconfig svnd3 ./image || bail "vnconfig svnd"
newfs svnd3c || bail "newfs"
fsck_ffs -f /dev/svnd3c || bail "initially fsck"
mount /dev/svnd3c /mnt || bail "initially mount"
dd if=/dev/arandom of=/mnt/test
cp /mnt/test ./image.test || bail "back up comparison file"
cmp /mnt/test ./image.test || bail "sanity check for files comparison"
umount /mnt || bail "initially unmount"
dd if=/dev/zero of=/dev/svnd3c bs=1 count=16000 || bail "clobber superblock"

echo "Executing after-effect..."
fsck_ffs -f /dev/svnd3c || bail "recovery fsck"
mount /dev/svnd3c /mnt || bail "recovery mount"
cmp /mnt/test ./image.test || bail "recovery check for files comparison"
umount /mnt || bail "recovery unmount"
vnconfig -u svnd3 || bail "recovery (un)vnconfig svnd"

echo "All done -- fsck_ffs does work on 'c' partition afterall..."

And the light ++c hack to see if disklabel is generated on the fly for
DIOCGDINFO call on disk with 'c' partition only:

#include <sys/param.h>
#include <sys/time.h>
#include <ufs/ufs/dinode.h>
#include <ufs/ffs/fs.h>
#include <sys/stat.h>
#include <sys/ioctl.h>
#include <sys/disklabel.h>

#include <errno.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
#include <unistd.h>

#include <iostream>
#include <stdexcept>


struct autofd {
        int const fd;
        autofd(char const * const dev)
        : fd(::open(dev, O_RDONLY)) {
                if (fd < 0)
                        throw ::std::runtime_error("can't open device");
        }
        ~autofd()
        {
                ::close(fd);
        }
        operator int ()
        {
                return fd;
        }
};

int
main()
{
        try {
                char const * const dev = "/dev/rsvnd3c";

                struct stat statb;
                if (stat(dev, &statb) < 0)
                        throw ::std::runtime_error("stat dev");
                
                if (!S_ISCHR(statb.st_mode))
                        throw ::std::runtime_error("not a char dev");

                autofd fd(dev);

                struct disklabel lab;
                if (::ioctl(fd, DIOCGDINFO, reinterpret_cast<char *>(&lab)) < 0)
                        throw ::std::runtime_error("get disklabel");

                if (lab.d_npartitions < 3)
                                throw ::std::runtime_error("no c partition 
placeholder");

                struct disklabel::partition & pp(lab.d_partitions[2]);
                if (pp.p_fstype != FS_BSDFFS)
                        throw ::std::runtime_error("not labeled as bsd fs");

                ::std::cout <<
                        "sector size: " << lab.d_secsize << "\n" <<
                        "sectors: " << lab.d_nsectors << "\n" <<
                        "tracks: " << lab.d_ntracks << "\n" <<
                        "sectors per cylinder: " << lab.d_secpercyl << "\n" <<
                        "partition size in sectors: " << pp.p_size << "\n" <<
                        "partition start sector: " << pp.p_offset << "\n";

                return 0;
        } catch (::std::exception const & e) {
                ::std::cerr << "Bailing -- got an error: " << e.what() << "\n";
        } catch (...) {
                ::std::cerr << "Bailing -- unknown exception caught\n";
        }
        return -1;
}

The code is built with:
gcc main.cc -lstdc++ -lm


Leon.

[demime 1.01d removed an attachment of type application/x-tar which had a name 
of leontest.tar]

Reply via email to