yop, J'ai déjà discuté du sujet avec Lionel, mais ce que je voudrais semble assez balaise à réaliser de manière fiable. Je m'explique. Le but est d'écrire du C sans relogement, et comprenant pourtant : - des romcalls - des ramcalls - des appels à ram_throw (utilisation de ce header : http://www.yaronet.com/posts.php?sl=&s=123645&p=1&h=8#8 ) Pour les romcalls, tigcc sait faire ça. Reste le problème des ramcalls. En assembleur, j'appelle les ramcalls de cette manière : pea arg1 move.w arg2,-(sp) move.b #3,d0 dc.w $F012 addq.l #6,sp Le truc, c'est que je n'ai pas la moindre idée de comment écrire ça en C. Le dc.w passe encore, mais forcer le push d'un argument, on fait comment ? Et le pop ? Faut tout se fader à chaque fois en assembleur inline ? Mon but est justement de passer au C pour gagner en facilité/rapidité, et donc de ne plus me taper d'asm à la main... Comment faire ? Question ultime : y a-t-il un/des switches garantissant aucun relogement dans le binaire ? Ou faut-il s'attendre à devoir patcher les fichiers .s après compilation ? Merci d'avance. si je savais faire ça, il y a longtemps que je serais passé au C pour mes projets. "MSVC, le soft qui arrive à générer des problèmes à partir de solutions" © |
Peut-être -mno-explicit-relocs ? Pour le reste : une macro, à la rigueur ? C'est pas portable, mais de toute manière, tu ne porteras jamais ton programme, donc autant passer du temps sur le reste |
Suis-je bien sûr que ma macro construire à coup de asm() ne sera pas détruite par le compilo ou le linker ? Il interprête ça comment l'asm inline ? Typiquement pour lui, un dc.w $F0xy ne veut rien dire, ça ne lui posera pas de problème de l'insérer dans le code ? "MSVC, le soft qui arrive à générer des problèmes à partir de solutions" © |
Ce n'est pas évident de faire ce genre de choses en C, c'est bien pour ça qu'on a des tas de bidouilles pour les ROM_CALLs F-Line en C. |
Ah merde. C'est dur de faire une macro pour ça ? Le compilateur ou le linker risquent de me péter quelque chose si j'arrive à écrire les macros ? "MSVC, le soft qui arrive à générer des problèmes à partir de solutions" © |
Je ne pense pas non, normalement avec de l'asm inline il ne devrait rien toucher. Un truc du genre : #define RAM_CALL asm { dc.w #$cafe } (enfin je n'ai plus en tête la syntaxe pour l'asm inline, mais tu vois l'idée, quoi... |
Oué, pareil pour la syntaxe, mais ça ressemblerait plus à ça : #define truc asm(".word 0xabcd"); "MSVC, le soft qui arrive à générer des problèmes à partir de solutions" © |
Une possibilité que j'ai suggérée à Martial est l'utilisation des RAM_CALLs et RAM_throw F-Line comme des fonctions C tout à fait normales, puis le post-processing du source ASM produit par le compilo, pour remplacer le jbsr/jsr par un dc.w / .word. La compilation en plusieurs étapes, il a l'habitude, ça n'est pas ça qui va le déranger. Membre de la TI-Chess Team. Co-mainteneur de GCC4TI (documentation en ligne de GCC4TI), TIEmu et TILP. |
Ouep, c'est sûr. Je me demandais juste s'il y avait plus simple Mais ta solution reste parfaitement valable. En fait, j'aurais juste du C ou du C++ à écrire pour post-processer. Un langage de script serait mieux me dit-on, mais je sais pas faire "MSVC, le soft qui arrive à générer des problèmes à partir de solutions" © |
Et euh... y a-t-il un moyen pour copier du code dans la pile ? En fait, en asm, je rassemble les pointeurs de libcall/syscall dont j'ai besoin dans la pile, et je les fait précéder à tout hasard de $4EF9 Ca ressemble à ça : [box=pile améliorée] \RegTable: ; WARNING: will fail using functions > lib##antispam##00FF (should use word sized offsets) moveq.l #0,d0 \RegLoop: move.b (a1)+,d0 bne.s \Continue tst.b (a1) bne.s \Continue ; End of table: dc.b 0,0 rts \Continue: pea (a4) RAMC RAM_kernel::LibsPtr addq.l #4,sp moveq.l #0,d0 ; PreOS bug: LibsPtr destroys d0 move.b (a2)+,d0 move.w #$4EF9,0(a6,d0.w) move.l a0,2(a6,d0.w) bra.s \RegLoop[/box] avec les offsets qui vont bien pour obtenir les numéros des libcall/syscall et l'offset de leur pointeur dans la pile : [box=superbes tables] ButillibCallsTable:
dc.b BUTILLIB_InitCmdLine
dc.b BUTILLIB_GetCurrentArgPtr
dc.b BUTILLIB_GetNextArgPtr
dc.b BUTILLIB_IsArgSwitch
dc.b BUTILLIB_IsNextArg
dc.b BUTILLIB_GetFilePtr
dc.b BUTILLIB_CheckFileType
dc.b BUTILLIB_ExecSwitchRoutine
dc.b BUTILLIB_SkipDummyLines
dc.b BUTILLIB_SkipSpaces
dc.b BUTILLIB_SkipLine
dc.b BUTILLIB_GetFileHandle
dc.b BUTILLIB_GetPrevArgPtr
dc.b BUTILLIB_AdvanceToChar
dc.b 0,0
ButillibCallsOffsets:
dc.b INIT_CMDLINE
dc.b GET_CURRENT_ARG_PTR
dc.b GET_NEXT_ARG_PTR
dc.b IS_ARG_SWITCH
dc.b IS_NEXT_ARG
dc.b GET_FILE_PTR
dc.b CHECK_FILE_TYPE
dc.b EXEC_SWITCH_ROUTINE
dc.b SKIP_DUMMY_LINES
dc.b SKIP_SPACES
dc.b SKIP_LINE
dc.b GET_FILE_HANDLE
dc.b GET_PREV_ARG_PTR
dc.b ADVANCE_TO_CHAR
PedroMCallsTable:
dc.b PEDROM_stdin
dc.b PEDROM_printf
dc.b PEDROM_fprintf
dc.b 0,0
PedroMCallsOffsets:
dc.b STDERR
dc.b PRINTF
dc.b FPRINTF[/box] Comme ça avec des appels en jsr non-relogés (et a6 pointeur de stack frame permanent) : [box=juuuuummmmmmp] lea CMDLINE(a6),a0 ; Get CMDLINE* jsr GET_NEXT_ARG_PTR(a6) ; Skip current arg lea CMDLINE(a6),a0 ; Get CMDLINE* jsr IS_ARG_SWITCH(a6) ; Check if the arg is an switch[/box] Vala, ça me fait tout ce que je veux sans m'embêter. On saute dans la pile et ça rebondit vers la bonne fonction de dll ou de pedrom. Mais... comment on fait ça en C ? "MSVC, le soft qui arrive à générer des problèmes à partir de solutions" © |
on écrit la fonction trampoline() en asm inline? Nspire wiki CONDUCTEUR Va-et-vient Des QUATRE MANCHE AVEC DES DIODES |
Oué, c'est exactemetn ça. En fait, c'est le contenu de cette fonction, le fait d'exécuter la struct {unsigned short opcode = 0x4EF9; void (*);} trampoline; qui risque d'être un peu chaud "MSVC, le soft qui arrive à générer des problèmes à partir de solutions" © |
Non, je crois que le plus simple est la solution de Lionel, compiler sans faire d'objet ni linker, patcher les fichiers assembleur, et enfin assembler le résultat. Ca doit pas être dur, c'est juste du parsing de texte, en C c'est très faisable (merde, je voulais éditer) "MSVC, le soft qui arrive à générer des problèmes à partir de solutions" © |
Pourquoi ne pas proposer un patch pour GCC4TI qui intégrerait ça ? Tu pourrais même appeler le switch pour l'activer "-folco" « Tout homme porte sur l'épaule gauche un singe et, sur l'épaule droite, un perroquet. » — Jean Cocteau |
C'est d'ailleurs comme ça que fonctionnent les ROM_CALLs en F-Line, aussi… |
./12: un script Perl qui lit le fichier en entier en un seul coup et une forêt de s///g ressemblerait à: use strict;
use File::Path;
# Files not explicitely closed on exit, but Perl does it for us.
package main;
my $file;
my $file2;
my $filecontents;
$file = $ARGV[0];
$file2 = $ARGV[1];
# Sanitize input file names - optional.
# Slurp input file !
open(INFILE,"$file") or die "Can't open $file: $!";
read(INFILE, $filecontents, -s INFILE);
close INFILE;
$filecontents =~ s/jbsr\s+rename/.word 0xFnnn/g;
$filecontents =~ s/jbsr\s+unlink/.word 0xFmmm/g;
...
print $filecontents; # sur stdout
# ou bien
# open(OUTFILE, ">$file2") or die "Can't open $file2: $!";
# print OUTFILE, $filecontents;
# close OUTFILE; En C/C++, c'est plus long à développer Membre de la TI-Chess Team. Co-mainteneur de GCC4TI (documentation en ligne de GCC4TI), TIEmu et TILP. |
Zerosquare (./13) : Pas con, mais faudrait que je me penche sur la structure de la bête Zerosquare (./13) : Kevin Kofler (./14) : Lionel Debroux (./15) : Je ne connais pas Perl "MSVC, le soft qui arrive à générer des problèmes à partir de solutions" © |
Je sais que tu ne connais pas Perl, c'est pour ça que je t'ai montré le corps d'un script qui ferait le boulot Membre de la TI-Chess Team. Co-mainteneur de GCC4TI (documentation en ligne de GCC4TI), TIEmu et TILP. |
Lionel Debroux (./15) : Pas vraiment (juste un petit peu), je dirais plutôt que c'est assez simple, et on peut se baser sur http://tigcc-linux.cvs.sourceforge.net/viewvc/tigcc-linux/tigcc-linux/sources/patcher/src/ . C'est aussi le logiciel à modifier pour intégrer ça à TIGCC/*nix (ou un fork, il y a peu de chances que ce soit accepté dans la version officielle!), TIGCC/W32 utilise actuellement du code Delphi pour faire la même chose. |
Ok, merci à vous deux. "MSVC, le soft qui arrive à générer des problèmes à partir de solutions" © |
Bon, je reprends, déjà merci pour tout. Le parsing de source me semble le plus adapté dans mon cas. Ca résoud donc le problème des ramcalls. C'est possible d'écrire ça en C ? butillib##antispam##000d:
subq.w #8,ARGIT(a0)
butillib##antispam##0002:
addq.w #4,ARGIT(a0)
butillib##antispam##0009:
move.w ARGIT(a0),d0
movea.l ARGV(a0),a0
movea.l 0(a0,d0.w),a0
rts Ca revient à 3 exports dans la même fonction... A mon avis c'est mort, mais sait-on jamais "MSVC, le soft qui arrive à générer des problèmes à partir de solutions" © |
Ce n'est pas possible, du moins pas sans des hacks affreux qui ont de fortes chances de ne pas avoir l'effet voulu. |
Je m'en doutais Est-il alors vraiment impossible d'avoir du code C aussi optimal que de l'assembleur ? "MSVC, le soft qui arrive à générer des problèmes à partir de solutions" © |
Pour un processeur comme le 68000 et dans certaines situations, oui, sans aucun doute possible. Membre de la TI-Chess Team. Co-mainteneur de GCC4TI (documentation en ligne de GCC4TI), TIEmu et TILP. |
butillib##antispam##000d()
{
butillib##antispam##internal(d)
}
butillib##antispam##0002()
{
butillib##antispam##internal(2)
}
butillib##antispam##0009()
{
butillib##antispam##internal(9)
}
butillib##antispam##internal( enum e )
{
switch ( e ) {
case 000d:
subq.w #8,ARGIT(a0)
case 0002:
addq.w #4,ARGIT(a0)
case 0009:
move.w ARGIT(a0),d0
movea.l ARGV(a0),a0
movea.l 0(a0,d0.w),a0
default: erreur quelconque...
}
} C'est pas vraiment la même chose, mais c'est pas loin. |
Ouep... Mais il y a un passage d'argument à faire. Je ne pense pas pouvoir faire mieux en C de toute façon, c'est clair... "MSVC, le soft qui arrive à générer des problèmes à partir de solutions" © |
bah après ton assembleur, tu codes un compilo ou tu améliores gtc Nspire wiki CONDUCTEUR Va-et-vient Des QUATRE MANCHE AVEC DES DIODES |
"MSVC, le soft qui arrive à générer des problèmes à partir de solutions" © |
Euh... j'arrive pas à écrire "pointeur sur une liste de pointeurs de fonction" dans un proto ... Pointeur de fonction : void (*Ptr)() Liste de pointeurs : (void (*Ptr)())[] => marche pas, expected declaration specifier Pointeur sur pointeur : *(void (*Ptr)()) => marche pas, idem Et des paquets de variantes, genre : un pointeur de fonction : (*) un pointeur de pointeur de fonction (*)* Ca merdouille dans tous les sens Ca s'exprime comment ? "MSVC, le soft qui arrive à générer des problèmes à partir de solutions" © |
D'après la doc de TIGCC/GCC4TI, int (*x[5])(); An array of pointers to a function which returns an integer La version avec une taille indéfinie de tableau de cette écriture doit être int (**x)() Et un pointeur sur ce truc doit être int (***x)() (non testé) Membre de la TI-Chess Team. Co-mainteneur de GCC4TI (documentation en ligne de GCC4TI), TIEmu et TILP. |