gpt4 book ai didi

c - Unix 机器上 C 程序中的总线错误

转载 作者:太空宇宙 更新时间:2023-11-04 04:06:53 26 4
gpt4 key购买 nike

我对 C 还很陌生,遇到了一个我无法理解其原因的“总线错误”。我从未听说过 gdb,但在这个论坛上遇到了它,并尝试在我的问题程序中使用它并得到以下输出:

% gdb Proc1 GNU gdb 5.0

...

This GDB was configured as "sparc-sun-solaris2.8"...

(no debugging symbols found)...

(gdb) run

Starting program: /home/0/vlcek/CSE660/Lab3/Proc1

(no debugging symbols found)...

(no debugging symbols found)...

(no debugging symbols found)...

Program received signal SIGSEGV, Segmentation fault. 0x10a64 in main ()

我不知道这是什么意思,是说我的代码第 10 行有错误吗?如果是这样,我代码中的第 10 行只是“int main()”,所以我不确定那里的问题......当我尝试运行该程序时,它只说“总线错误”,所以我不确定该去哪里从这里走。我什至尝试在 main 之后放置一个 printf,但它不打印字符串,只给我一个总线错误。

下面是我的代码:

// Compilation Command: gcc -o Proc1 Proc1.c ssem.o sshm.o

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include "ssem.h"
#include "sshm.h"

// Code of Proc1
int main()
{int i, internal_reg;
int key1 = 111111, key2 = 222222, key3 = 333333, key4 = 444444;
/* here create and initialize all semaphores */
int sem1 = sem_create(key1, 1);
if (sem1 < 0) {
perror("sem failed");
}
int sem2 = sem_create(key2, 1);
if (sem2 < 0) {
perror("sem failed");
}
int sem3 = sem_create(key3, 1);
if (sem3 < 0) {
perror("sem failed");
}
int sem4 = sem_create(key4, 1);
if (sem4 < 0) {
perror("sem failed");
}
/* here created: shared memory array Account of size 3 */
int *Account;
int shmid = shm_get(123456, (void**) &Account, 3*sizeof(int));
if (shmid < 0) {
perror("shm failed");
}
Account[0]=10000;
Account[1]=10000;
Account[2]=10000;
/* synchronize with Proc2, Proc3 and Proc4 (4 process 4 way synchronization)*/

for (i = 0; i < 1000; i++)
{
sem_signal(sem1);
sem_signal(sem1);
sem_signal(sem1);

internal_reg = Account[0];
internal_reg = internal_reg - 200;
Account[0] = internal_reg;

/* same thing, except we're adding $100 to Account1 now... */
internal_reg = Account[1];
internal_reg = internal_reg + 200;
Account[1] = internal_reg;

if (i % 100 == 0 && i != 0) {
printf("Account 0: $%i\n", Account[0]);
printf("Account 1: $%i\n", Account[1]);
}

if (i == 300 || i == 600) {
sleep(1);
}

sem_wait(sem2);
sem_wait(sem3);
sem_wait(sem4);
}
/* Here add a code that prints contents of each account
and their sum after 100th, 200th, 300th, ...., and 1000th iterations*/

}

/*in the code above include some wait and signal operations on semaphores. Do no
t over-synchronize. */

这是 ssem 和 sshm 的文档:

/*
* ssem.c
*
* Version 1.0.0
* Date : 10 Jan 2002
*
*/


#include <sys/ipc.h>
#include <sys/sem.h>
#include <sys/types.h>

#include "ssem.h"

#define PERMS 0600

static struct sembuf op_lock[1] = {
0, -1, 0
};

static struct sembuf op_unlock[1] = {
0, 1, IPC_NOWAIT

};



int sem_create(int key,int initval)
{
int semid,i;
semid = semget((key_t)key, 1, IPC_CREAT | PERMS);

for(i=0;i<initval;i++)
semop(semid,&op_unlock[0],1);


return semid;

}

int sem_open(int key)
{
int semid;
semid = semget(key,0,0);
return semid;
}


int sem_wait(int semid)
{
return semop(semid,&op_lock[0],1);
}


int sem_signal(int semid)
{
return semop(semid,&op_unlock[0],1);
}


int sem_rm(int semid)
{
return semctl(semid, 0, IPC_RMID, 0);
}



/*
* sshm.c
*
* Routines for Simpler shared memory operations
* Version : 1.0.0.
* Date : 10 Jan 2002
*
*/

#include <sys/shm.h>
#include <sys/ipc.h>
#include <sys/types.h>

#include "sshm.h"

#define PERMS 0600

int shm_get(int key, void **start_ptr, int size)
{
int shmid;
shmid = shmget((key_t) key, size, PERMS | IPC_CREAT);
(*start_ptr) = (void *) shmat(shmid, (char *) 0, 0);
return shmid;

}


int shm_rm(int shmid)
{
return shmctl(shmid, IPC_RMID, (struct shmid_ds *) 0);

}

在使用 -ggdb 标志编译 Proc1.c 并运行 gdb 后,我得到以下信息:

Program received signal SIGSEGV, Segmentation fault. 0x10a64 in main () at Proc1.c:36

36 Account[0]=10000

为什么这会导致段错误?

将Account的声明改为

int *Account = 0;

并添加

printf("Account == %p\n", Account);

在账户[0] = 10000之前;

我在运行 Proc1 时得到以下信息:

Account == ffffffff
Bus error

最佳答案

为了从 gdb 获得更合理的结果,您应该使用 -ggdb 选项编译您的程序。然后,这会将调试信息(如行号)包含到您的程序中。

您当前看到的是程序计数器的内存地址(0x10a64)。除非您可以将在那里找到的汇编指令与您自己的 C 程序的一部分相关联,否则这对您没有太大帮助。

看起来您正在正确使用 shm_get。我认为库设计者在将函数命名为与 shmget 如此相似时犯了一个可怕的错误。

跟我想的一样。 Account 指针以其中的无效值(又名 0xffffffff(又名 (void *)(-1)))结尾。值 (void *)(-1) 通常表示某种错误,它在 shmat 的联机帮助页中明确提及。这表明库内的 shmat 调用失败。您可以通过以下方式判断它是否失败:

 if (Account == (void *)(-1)) {
perror("shmat failed");
}
Account[0] = 10000;
// ...

现在,它失败的原因是一个有趣的谜。显然 shmget 调用成功了。

我个人认为System V IPC此时基本上已弃用,您应该尽可能避免使用它。

关于c - Unix 机器上 C 程序中的总线错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5035958/

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