2

我的任务是编写一个汇编例程,它使用 c 的 malloc 并从指向 asm 中的函数的指针插入值,数组打印它应该打印的内容,并且在程序打印的数字之后,一切似乎都正确,它添加了“NULL POINTER ASSIGNMENT "

c文件:

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

extern  int initarr (int** arr, int n, int (*initfunc)());

int getNum() 
{return (rand()%1000);}

void main()
{
int * arr, i, n, success;
printf("\nPlease enter the array size\n");
scanf("%d",&n);
success = initarr(&arr, n, getNum);   
if(!success) 
{
    printf("Memory Allocation Failed\n");
    return;
} 
printf("\nThe Numbers in the allocated array are:\n");
for(i=0; i<n; i++)   
    printf("%d  ", arr[i]);
} 

asm 文件:

;lab10.asm
.MODEL SMALL
.CODE
.386
;extern  int initarr (int** arr, int n, int (*initfunc)())
;arr=BP+4   n=BP+6  function=BP+8
EXTRN _malloc : NEAR
PUBLIC _initarr
_initarr PROC NEAR 
PUSH BP;preserve BP
MOV BP,SP;let me see whats in the stack
PUSH SI;preserve SI
MOV SI,[BP+4];SI controls array
MOV CX,[BP+6];CX=n from declaration
MOV eAX,8;size of one byte=8
MUL CX;how many bytes to allocate
SUB eSP,eAX;size in bytes to allocate
PUSH eAX;preserve for later add to sp
PUSH eCX;for ussing malloc
CALL _malloc;return addres is in eAX
TEST eAX,eAX;checks if allocation succeded
POP eCX;pop CX for the loop
POP eAX;pop AX to add to SP
ADD eSP,eAX;for stack won't change
JZ BYE;if allocation went wrong and eAX is already 0
MOV [SI],eAX;sets SI to point at the array
MOV SI,[SI];set SI to the array
LOOPER:
PUSH CX;for ussing rand
CALL [BP+8];call function
POP CX;pop CX for the loop
MOV [SI],eAX
ADD SI,2
LOOP LOOPER
;end of loop    
MOV eAX,1;for the return value
BYE:
POP SI;restore SI
POP BP;restore BP
RET
_initarr ENDP
END 

在此处输入图像描述

__________________编辑___________________

对于每个大于 5 的尺寸,它的工作都很棒 在此处输入图像描述

4

1 回答 1

2

一些编译器在 MSDOS 平台上清理代码时显示“空指针分配”消息。清理代码检查 NULL ptr 周围的内存区域的内容是否已更改(因为在 MSDOS 中没有访问冲突之类的东西)。因此,您需要检查您的代码在哪里可以取消引用 NULL 或其他低偏移量。在下面的源代码中,我可以发现一个错误:

CALL _malloc;return addres is in eAX
TEST eAX,eAX;checks if allocation succeded
POP eCX;pop CX for the loop
POP eAX;pop AX to add to SP
ADD eSP,eAX;for stack won't change

_malloc 返回 eAX 中的 ptr,稍后会被 POP eAX(包含 8)覆盖 2 个命令。稍后您取消引用它MOV [SI],eAX是错误消息的根本原因。

于 2012-12-27T23:27:13.400 回答