Hello 2009/5/9 Javier Martín <lordhab...@gmail.com>
> El sáb, 09-05-2009 a las 11:17 +0200, Vladimir 'phcoder' Serbinenko > escribió: > > +/* Realmode far ptr (2 * 16b) to the previous INT13h handler. */ > > +extern grub_uint32_t grub_drivemap_int13_oldhandler; > > I prefer it to be 2 variables because of the way they interract so > > nobody would do something like (char *) > > grub_drivemap_int13_oldhandler; > Two variables? I don't understand. You want the CS and IP parts of the > far pointer separated? Yes. Especially that in .S you declared it as two words. But it's not an absolute must > Why would anyone try to use it like you said? The > comment explicits that it is a realmode pointer, and as such it is not > usable as a linear pmode pointer. > > > +/* The type "void" is used for imported assembly labels, takes no > > storage and > > + serves just to take the address with &label. Do NOT put void* > > here. */ > > +/* Start of the handler bundle. */ > > +extern void grub_drivemap_int13_handler_base; > > The common practice is to use declarations like > > extern char grub_drivemap_int13_handler_base[]; > > it makes pointer arithmetics easy > That variable is used only twice in the code: one in a call to memcpy > and another one inside the macro INT13_OFFSET. It would still be so even > if it were changed to a char array, with the added risk that a char > array can be modified with almost nothing to gain (as casts would still > be needed). The casts are needed if you declare it as char foo[]; If someone modifies an array he does it on purpose > In fact, I'm declaring all the labels as "const void" so > no-one tries to overwrite the "master" data instead of the deployed > data. > > > +typedef struct drivemap_node > > +{ > > + grub_uint8_t newdrive; > > + grub_uint8_t redirto; > > + struct drivemap_node *next; > > +} drivemap_node_t; > > Here you could reuse Bean's list functions > After examining the API, I think such a change would be too invasive for > a "mature" patch right now. However, once the patch is in and drivemap > is working for everyone, I can work on modifying it to use > <grub/list.h>. The biggest hurdle I see is that there is no way to > automatically maintain "key" uniqueness like the current methods do, so > an iteration over the list would be required. > > As a side note/rant, <grub/list.h> desperately needs documentation > comments for other developers to be able to actually use it without > wondering if they're going into the darkness. In particular I need to > know what grub_list_insert does without delving into "list.c" (for > example: "if all tests fail, will the item be inserted last or not at > all?"). This rant also applies to other grub include files. > > > > It's not necessary to try to open the disk actually because for some > > reason biosdisk module may be not loaded (e.g. grub may use ata) and > > loading it may cause nasty effects (e.g. conflict between ata and > > biosdisk) > > Same applies for revparse_biosdisk. And if user wants to map to > > unexistant disk it's ok. So just use tryparse_diskstring > Hmm... this is a profound design issue: you are arguing that drivemap > should not care about actual BIOS disks, but just their drive numbers, > and let you map 0x86 to 0x80 even if you have no (hd6). I do not agree, > since the main use for GRUB is a bootloader, not a test platform (which > would be the only reason to perform that kind of mappings which are sure > to cause havoc). Thus, by default drivemap only lets you perform > mappings that will work - any stress tester is free to modify the code. > > Regarding opening the disk, the biosdisk interface does not assure in > any way (and again there is almost no API documentation) that hd0 will > be 0x80, though it's the sensible thing to expect. However, expecting > sensible things from an unstable API prone to redesigns (like the > command API, for example) Mapping something to non-existant drive makes the drive non-existant. I think it's fair It's almost guaranteed that numbering of disks won't change > usually leads to nasty surprises later on, so > unless 1) the biosdisk interface so specifies and 2) hdN will _always_ > refer to biosdisk and not another possible handler, I think the check is > a Good Thing (tm). It's not if it may hang when you use ata.mod > > > > + if ((str[0] == 'f' || str[0] == 'h') && str[1] == 'd') > > change this to something like > > if (! ((str[0] == 'f' || str[0] == 'h') && str[1] == 'd')) > > return ... > > It makes code nicer by avoiding unnecessary long else > I think the "else" you were talking about was one line long, but it was > not needed anyways and so I've removed it. > > > + grub_errno = GRUB_ERR_NONE; > > No need to set grub_errno explicitely unless you want to ignore an > > error > I may have got it wrong, but I think that functions like strtoul only > _set_ errno if there _has_ been an error, but leave it unchanged if > there hasn't. As I want to check if the conversion failed, I need to > have the variable to check in a known pre-state. > If grub_errno is set at the begining of the function then either scripting or your code has a bug > > > + unsigned long drivenum = grub_strtoul (str + 2, 0, 0); > > + if (grub_errno != GRUB_ERR_NONE || drivenum > 127) > > I think it's enough to just check the range > See above. > > > + { > > + /* N not a number or out of range. */ > > + goto fail; > > Since at fail you have just a return, just put a return here with > > informative error message > The function was rearranged so only the first goto is required now. > +fail: + return grub_error (GRUB_ERR_BAD_ARGUMENT, "device format \"%s\" " + "invalid: must be (f|h)dN, with 0 <= N < 128", str); I don't see a clear benefit of using goto here. Perhaps providing a separate informative messages for 2 fail cases is a better option > > > + } > > + else > > else is unnecessary, just continue with the code > I fear you don't give me enough context to find this one. If it was > inside tryparse_diskstring, it has been removed. > > > + if (cmd->state[OPTIDX_LIST].set || 0 == argc) > > argc == 0 sounds better. Also moving list function to a separate > > function is a good idea > Ok, changed. I'd just like to remark that if someone slipped and changed > the line to "argc = 0" it would pass unnoticed by the compiler, while "0 > = argc" would create a compile-time error. It's what warnings are for > Regarding slicing the listing > functionality, I don't see the advantages. > It makes code more readable and avoids long else clause > > + push %bp > > + mov %sp, %bp > > You don't need to change bp. Just use sp. In this case it makes code > > easier > That would make it more difficult to address passed parameters like the > flags at [BP+6] because their location would depend on the current > status of SP (which might change depending on code above the relevant > lines). I checked and can say that removing bp code makes the rest actually easier. If you don't see how tell me I'll modify it > I think that trying to shave off those 4 or so bytes would make > the code unnecessarily more complicated, which is usually a no-no when > assembly is involved. Besides, this prolog sequence and the > corresponding epilog before returning is practically a standard in x86. > > > + lcall *% > > cs:GRUB_DRIVEMAP_INT13H_OFFSET(grub_drivemap_int13_oldhandler) > > + pop %bp /* The pushed flags were removed by iret. */ > > Use rather: > > pushf > > lcall *% > > cs:GRUB_DRIVEMAP_INT13H_OFFSET(grub_drivemap_int13_oldhandler) > Why would we want to save our current flags? We want the old int13 > handler to receive the flags it would have received if the drivemap > int13 handler were not present, that is, the original caller flags which > were stored above CS:IP in the stack when the "int $0x13" call was > issued. The comment means that "iret" pops not just CS:IP but also the > flags, so we can proceed directly to popping %bp. > > > Also remember to restore original %dl after the call > I think there is no need to do so, because BIOS calls modify %dl anyway. > Then you can remove %bp and just make a long jump. This way int 13 handler recieves the original stack intact > > > > Could you also prepare a changelog entry for it? > Hmm... I don't really know how to write it, given that both files are > new and the patch does not modify any other GRUB function. Am I supposed > to also declare the modifications in the rmk files? (written in mailer so identation is wrong) <your contact line> <short description> * conf/i386-pc.rmk (pkglib_MODULES): add drivemap.mod (drivemap_mod_SOURCES): new variable (drivemap_mod_CFLAGS): likewise (drivemap_mod_LDFLAGS): likewise (drivemap_mod_ASFLAGS): likewise * commands/i386/pc/drivemap.c: new file * commands/i386/pc/drivemap_int13h.S: likewise > > -- > -- Lazy, Oblivious, Recurrent Disaster -- Habbit > > _______________________________________________ > Grub-devel mailing list > Grub-devel@gnu.org > http://lists.gnu.org/mailman/listinfo/grub-devel > > -- Regards Vladimir 'phcoder' Serbinenko
_______________________________________________ Grub-devel mailing list Grub-devel@gnu.org http://lists.gnu.org/mailman/listinfo/grub-devel