我制作了一个分为两部分的程序。 第一部分用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.");
}
和装配部件。
.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]
元素,并将其余部分返回到表的printf
元素。然后C模块使用 #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.");
}*/
函数输出表的[7]和[6]元素。
.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
翻译的装配模块:
//
在setdiff
中并不是计划的重要组成部分。但我认为仍然需要理解这个想法。
我在Dos shell中使用Broland C(BC)。它工作正常。该程序应该从用户获取数据(用户确定数字的数量)。下载的数据保存在动态数组中。然后在汇编模块程序中计算数组中数据的平均值并返回该值。它很容易使用普通数组,但动态有点棘手......我不知道如何开始。
答案 0 :(得分:-2)
没有一般性答案,因为它是特定于实现的。阅读C11标准(草案n1570),您可能不会看到任何提及x86的内容。
您应该阅读与{em>特定系统,处理器和C编译器相关的ABI和calling conventions的文档。某些编译器可能遵循不同的调用约定(例如,使用每个函数注释或编译器配置或#pragma
- s)。对于x86,首先阅读(各种)x86 calling conventions上的wikipage(例如cdecl
,{16}模式中的fastcall
等等。)
如果使用GCC编译mycsource.c
C 代码,则可以使用gcc -fverbose-asm -Wall -O1 -S mycsource.c
进行编译并研究生成的汇编程序文件mycsource.s
以了解调用汇编程序函数。您也可以使用Inline Assembly Language in C Code。
如果您不使用GCC,请尝试让您的C编译器发出并保留汇编代码并对其进行研究(并理解,猜测或检查与您的问题相关的调用约定)。如果无法做到这一点,请使用一些反汇编程序(或调试程序的反汇编工具)。