c - 是什么导致了短 C 代码中的段错误?

标签 c debugging assembly compilation

大家。我编写了以下代码,编译并运行了它,但出现了段错误。

#include <stdio.h>
typedef long long ll_t;

void store_prod(ll_t * dest, int x, ll_t y) {
    * dest = x * y;
}

int main(void)
{
    int a = 2049;
    ll_t b = 2147483645;
    ll_t * c;

    store_prod(c, a, b);
    printf("%d * %lld = %lld\n", a, b, *c);

    return 0;
}

Dev-C++ 中的调试器转储以下内容:

0x00401500 <+0>:    push   %ebp
0x00401501 <+1>:    mov    %esp, %ebp
0x00401503 <+3>:    push   %ebx
0x00401504 <+4>:    sub    $0xc, %esp
0x00401507 <+7>:    mov    0x10(%ebp), %eax
0x0040150a <+10>:   mov    %eax, -0x10(%ebp)
0x0040150d <+13>:   mov    0x14(%ebp), %eax
0x00401510 <+16>:   mov    %eax, -0xc(%ebp)
0x00401513 <+19>:   mov    0xc(%ebp), %eax
0x00401516 <+22>:   mov    %eax, %edx
0x00401518 <+24>:   sar    $0x1f, %edx
0x0040151b <+27>:   mov    -0x10(%ebp), %ecx
0x0040151e <+30>:   mov    %ecx, %ebx
0x00401520 <+32>:   imul   %edx, %ebx
0x00401523 <+35>:   mov    -0xc(%ebp), %ecx
0x00401526 <+38>:   imul   %eax, %ecx
0x00401529 <+41>:   add    %ebx, %ecx
0x0040152b <+43>:   mull   -0x10(%ebp)
0x0040152e <+46>:   add    %edx, %ecx
0x00401530 <+48>:   mov    %ecx, %edx
0x00401532 <+50>:   mov    0x8(%ebp), %ecx
0x00401535 <+53>:   mov    %eax, (%ecx) ; The problem lies with this line
0x00401537 <+55>:   mov    %edx, 0x4(%ecx)
0x0040153a <+58>:   add    $0xc, %esp
0x0040153d <+61>:   pop    %ebx
0x0040153e <+62>:   pop    %ebp
0x0040153f <+63>:   ret

注册信息:

EAX 0x7fffe7fd  2147477501
ECX 0x40234e    4203342
EDX 0x400   1024
EBX 0x0 0
ESP 0x63fe58    0x63fe58
EBP 0x63fe68    0x63fe68
ESI 0x3d    61
EDI 0xb70d48    11996488
EIP 0x401535    0x401535 <store_prod+53>
EFLAGS  0x10206 [ PF IF RF ]
CS  0x23    35
SS  0x2b    43
DS  0x2b    43
ES  0x2b    43
FS  0x53    83
GS  0x2b    43

地址 [%ecx](即 0x40234e)处的代码似乎被覆盖了,但为什么会出现这样的访问冲突? C代码有什么问题吗? 非常感谢您的帮助!

最佳答案

您尝试写入不属于您的内存 → 未定义的行为(段错误可选)

ll_t * c; 行中声明了 c,但您没有使用 malloc 或将其指向有效地址(例如: ll_t d; ll_t *c = &d;)

ll_t * c; 更改为 ll_t * c = malloc(sizeof(ll_t)); 并检查 malloc 是否失败,或者更改到 ll_t d; ll_t *c = &d;

编辑:正如@Ajay Brahmakshatriya 指出的那样,还有另一种(按时间顺序排列的)未定义行为 - 将未初始化的变量 (c) 发送到 store_prod( ...)

关于c - 是什么导致了短 C 代码中的段错误?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/45479577/

相关文章:

php - 组织编程竞赛 - 如何编译和执行

python - Visual Studio Code (Python) - 'NoneType' 对象没有属性 'write'

c# - 抛出异常,除非在断点后运行

node.js - 类似于 Node 中的 bash -x

c - 使用 realloc 释放 2d 字符数组

c - 发送到:无效参数

c - 为什么编译器坚持在这里使用被调用者保存的寄存器?

assembly - 位字节字和双字 - 何时在汇编中使用什么?

assembly - 如何将 Vtune 分析限制为特定函数

c - C 中的指针操作