gpt4 book ai didi

c - 如何将数据从 C 传递到汇编

转载 作者:行者123 更新时间:2023-11-30 21:12:37 25 4
gpt4 key购买 nike

我制作了一个分为两部分的程序。第一部分是用 C 编写的,它从用户那里获取数据。第二个是用汇编及其制作操作编写的。我在将数据从动态分配的数组传递到汇编模块时遇到问题。程序无法正常工作。我做错了什么?

#include <stdio.h>
#include <stdlib.h>
#include <time.h>

extern "C" near void licz_sr( int* );
extern "C" near void licz_znaki( char* );

int *tablica_liczb_dynamiczna;
char tablica_znakow[28]; //[0] = ilosc, [1] = kod szukany, [27] = ilosc wystapien
int wielkosc_tablicy = 0;
int kod;

int menu(), losuj_liczby(), losuj_znaki(), licz_srednia(), sprawdz_ilosc_znakow();
void wczytaj_liczby(), wczytaj_znaki();

int main() {
srand(time(NULL));

//tablica_liczb[0] = 5;
tablica_znakow[0] = 25;
menu();

return 0;
}

int menu(){

printf("\n\n\nCo mam zrobic?\nW = Wyliczyc srednia z losowej tablicy\nP = Policzyc ilosc znakow w losowej tablicy\nX = Wyjsc z programu\n");

kod = getchar();
getchar();

if (kod == 87 || kod == 119) {
//losuj_liczby();
wczytaj_liczby();
//licz_srednia();
}
else if (kod == 80 || kod == 112){
//losuj_znaki();
wczytaj_znaki();
sprawdz_ilosc_znakow();
}
else if (kod == 88 || kod == 120){
return 0;
}
menu();
}

void wczytaj_liczby()
{
printf("Ile znakow wprowadzic? ");
scanf("%d",&wielkosc_tablicy);
printf("wielkosc tab: "); printf("%d",wielkosc_tablicy);

tablica_liczb_dynamiczna = (int*)malloc(wielkosc_tablicy * sizeof(int));

for(int i = 0; i < wielkosc_tablicy; i++){
tablica_liczb_dynamiczna[i] = NULL;
}

printf("\n\nWczytaj liczby:");

for(int j = 0; j < wielkosc_tablicy; j++)

{
scanf("%d", &tablica_liczb_dynamiczna[j]);

printf("\n");
printf("%d", j);printf("element tab "); printf("%d",tablica_liczb_dynamiczna[j]);
printf("\n");
tablica_liczb_dynamiczna[j] = tablica_liczb_dynamiczna[j]-48;
}
licz_srednia();
}

void wczytaj_znaki()

{

for (int i =2; i < tablica_znakow[0]+2; i++)

{

tablica_znakow[i] = NULL;

}

printf("\n\nWczytaj znaki:");

char polecenie[25];

fgets( polecenie, 25, stdin );

for (int j =2; j < tablica_znakow[0]+2; j++)

{

tablica_znakow[j] = polecenie[j-2];

}

}



int licz_srednia()

{
licz_sr(tablica_liczb_dynamiczna);



//printf("\n\nSrednia jest rowna: "); printf("%c", tablica_liczb[6]+48); printf(", a reszta jest rowna: "); printf("%c", tablica_liczb[7]+48);

printf("\n\nSrednia jest rowna: "); printf("%c", tablica_liczb_dynamiczna[6]+48); printf(", a reszta jest rowna: "); printf("%c", tablica_liczb_dynamiczna[7]+48);

//cout << "Srednia jest rowna: " << tablica_liczb_dynamiczna[6]+48 << ", a reszta: " << tablica_liczb_dynamiczna[7]+48 << endl;

return 0;

}


int sprawdz_ilosc_znakow()

{

printf("\n\nWystapienia jakiego znaku mam liczyc? ");

tablica_znakow[1] = getchar();

getchar();

//printf("\n\n");

licz_znaki(tablica_znakow);



printf("\nZnak \""); printf("%c", tablica_znakow[1]); printf("\" wystapil "); printf("%c", tablica_znakow[27]+48);

if (tablica_znakow[27] != 1) printf(" razy.");

else printf(" raz.");

}

以及 assembly 部分。

.model small, C


Dane SEGMENT

Tablica DB (?)
Ilosc DB (?)

Dane ENDS


PUBLIC licz_znaki

.code


licz_znaki PROC near


push bp
mov bp, sp
mov di, [bp+4]
; [bp+4] = adres
; [di] = liczba
; [bp+12] = tab[1]
; [bp+16] = tab[2]
; [bp+20] = tab[3]
; [bp+24] = tab[4]
xor cx, cx
mov cl, [di]
inc di
;inc di
xor ax, ax
xor bx, bx
xor dx, dx
mov ah, [di] ;w ah jest znak, którego szukamy
inc di
;inc di
Petla_z:
mov al, [di] ;w ah jest aktualny znak z tablicy
cmp al, ah
jz Dodaj_do_wyniku
Powrot_do_petli:
inc di
;inc di
loop Petla_z

Dodaj_do_wyniku:
jcxz Koncowka
inc bx ;bh przechowuje ilosc wystapien
jmp short Powrot_do_petli

Koncowka:
mov [di], bx

pop bp
ret

licz_znaki ENDP

PUBLIC licz_sr

.code


licz_sr PROC near


push bp
mov bp, sp
mov di, [bp+4]
mov bx, [di]
; [bp+4] = adres
; [di] = liczba
; [bp+12] = tab[1]
; [bp+16] = tab[2]
; [bp+20] = tab[3]
; [bp+24] = tab[4]
xor cx, cx
mov cl, [di]
inc di
inc di
xor ax, ax
xor dx, dx
Petla_l:
add ax, [di]
inc di
inc di
loop Petla_l

div bx

mov [di], ax
inc di
inc di
mov [di], dx ;wj

pop bp
ret

licz_sr ENDP


end

翻译后的 C 模块:说明:具有普通数组(非动态)的程序正在将数据下载到 [8] 元素数组。用户可以输入5个数字。汇编模块将平均值返回到表的 [6] 元素,并将除法的其余部分返回到表的 [7] 元素。然后C模块使用printf函数输出表的[7]和[6]元素。

 #include <stdio.h>
#include <stdlib.h>
#include <time.h>

extern "C" near void count_averageExtern( int* );
extern "C" near void count_characters( char* );

int *dynamic_arrau;
//char tablica_znakow[28]; //[0] = ilosc, [1] = kod szukany, [27] = ilosc wystapien
int size_of_array = 0;
int code;

int menu(), count_average(), sprawdz_ilosc_znakow();
void download_numbers(), wczytaj_znaki();

int main() {
srand(time(NULL));

//tablica_liczb[0] = 5;
tableOfCharacters[0] = 25;
menu();

return 0;
}

int menu(){

printf("\n\n\nWhat to do?\nW = average\nP = not important\nX = exit\n");

code = getchar();
getchar();

if (code == 87 || code == 119) {
//losuj_liczby();
download_numbers();
count_average();
}
/*else if (kod == 80 || kod == 112){
//losuj_znaki();
wczytaj_znaki();
sprawdz_ilosc_znakow();
}*/
else if (kod == 88 || kod == 120){
return 0;
}
menu();
}

void download_numbers()
{
printf("How many numbers input? ");
scanf("%d",&size_of_array);

dynamic_arrau = (int*)malloc(size_of_array * sizeof(int));

for(int i = 0; i < size_of_array; i++){
dynamic_arrau[i] = NULL;
}

printf("\n\Input numbers:");

for(int j = 0; j < size_of_array; j++)

{
scanf("%d", &dynamic_arrau[j]);

printf("\n");
tablica_liczb_dynamiczna[j] = tablica_liczb_dynamiczna[j]-48;
}
licz_srednia();
}

/*void wczytaj_znaki() {

for (int i =2; i < tablica_znakow[0]+2; i++)

{

tablica_znakow[i] = NULL;

}

printf("\n\nWczytaj znaki:");

char polecenie[25];

fgets( polecenie, 25, stdin );

for (int j =2; j < tablica_znakow[0]+2; j++)

{

tablica_znakow[j] = polecenie[j-2];

}

}
*/


int count_average()

{
count_averageExtern(dynamic_arrau);
//printf("\n\nSrednia jest rowna: "); printf("%c", tablica_liczb[6]+48); printf(", a reszta jest rowna: "); printf("%c", tablica_liczb[7]+48);
printf("\n\nAverage: "); printf("%c", dynamic_arrau[6]+48); printf(", and rest of division: "); printf("%c", dynamic_arrau[7]+48);
return 0;
}


/* int sprawdz_ilosc_znakow()

{

printf("\n\nWystapienia jakiego znaku mam liczyc? ");

tablica_znakow[1] = getchar();

getchar();

//printf("\n\n");

licz_znaki(tablica_znakow);



printf("\nZnak \""); printf("%c", tablica_znakow[1]); printf("\" wystapil "); printf("%c", tablica_znakow[27]+48);

if (tablica_znakow[27] != 1) printf(" razy.");

else printf(" raz.");

}*/

翻译后的组装模块:

    .model small, C


Data SEGMENT

Tablica DB (?)
Ilosc DB (?)

Data ENDS


/* PUBLIC licz_znaki

.code


licz_znaki PROC near


push bp
mov bp, sp
mov di, [bp+4]
; [bp+4] = adres
; [di] = liczba
; [bp+12] = tab[1]
; [bp+16] = tab[2]
; [bp+20] = tab[3]
; [bp+24] = tab[4]
xor cx, cx
mov cl, [di]
inc di
;inc di
xor ax, ax
xor bx, bx
xor dx, dx
mov ah, [di] ;w ah jest znak, którego szukamy
inc di
;inc di
Petla_z:
mov al, [di] ;w ah jest aktualny znak z tablicy
cmp al, ah
jz Dodaj_do_wyniku
Powrot_do_petli:
inc di
;inc di
loop Petla_z

Dodaj_do_wyniku:
jcxz Koncowka
inc bx ;bh przechowuje ilosc wystapien
jmp short Powrot_do_petli

Koncowka:
mov [di], bx

pop bp
ret

licz_znaki ENDP
*/
PUBLIC count_averageExtern

.code


count_averageExtern PROC near


push bp
mov bp, sp
mov di, [bp+4]
mov bx, [di]
; [bp+4] = addres
; [di] = number
; [bp+12] = tab[1]
; [bp+16] = tab[2]
; [bp+20] = tab[3]
; [bp+24] = tab[4]
xor cx, cx
mov cl, [di]
inc di
inc di
xor ax, ax
xor dx, dx
Loop_l:
add ax, [di]
inc di
inc di
loop Loop_1

div bx

mov [di], ax
inc di
inc di
mov [di], dx ;wj

pop bp
ret

count_averageExtern ENDP


end

//中并不是程序的重要部分。但我认为还是需要理解idea。

我在 Dos shell 中使用 Broland C(BC)。效果很好。该程序应该从用户处获取数据(用户确定数字的数量)。下载的数据保存在动态数组中。然后在汇编模块中程序计算数组中数据的平均值并返回值。对于普通数组来说很容易,但是对于动态数组来说有点棘手......我不知道如何开始。

最佳答案

没有通用答案,因为它是特定于实现的。阅读C11标准(草案 n1570 ),您可能不会看到任何提及 x86 的内容。

您应该阅读 ABI 的文档和 calling conventions与您的特定系统、处理器和 C 编译器相关。某些编译器可能遵循不同调用约定(例如,使用每个函数注释或编译器配置或#pragma-s)。对于 x86,请首先阅读维基页面(各种)x86 calling conventions (例如 16 位模式下的 cdeclfastcall 等...)

如果使用 GCC 编译您的 mycsource.c C 代码,您可以使用 gcc -fverbose-asm -Wall -O1 -S mycsource.c 进行编译,并研究生成的汇编程序文件 mycsource.s 以了解对汇编程序函数的调用。您还可以使用Inline Assembly Language in C Code .

如果您不使用 GCC,请尝试要求您的 C 编译器发出并保留汇编代码并研究它(并理解、猜测或检查与您的问题相关的调用约定)。如果不可能,请使用一些反汇编程序(或调试器的反汇编工具)。

关于c - 如何将数据从 C 传递到汇编,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44631632/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com