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

标签 c assembly

我制作了一个分为两部分的程序。 第一部分是用 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/

相关文章:

c - 在 asm 中,printf() 如何知道传入了多少个变量?

string - 移动到缓冲区内字符串的末尾 - 汇编语言

汇编(MIPS)正确使用: Registers vs.堆栈

C:帮助理解指针

c - SciPy中16384.0*floor(v/16384.0)的使用说明

assembly - 汇编中main函数开头的栈内存操作

linux - 为什么 'syscall' 指令在 Linux 下不起作用?

c++ - 在 Windows 中,如何拥有作为重定向管道的非阻塞标准输入?

c - 为什么 C 的编译 stderr 包含我正在使用的静态库的开发人员的密码?

c - 如何在管道上放置指针?