Ошибка шины в программе C на Unix-машине
Я довольно неопытен с C и сталкиваюсь с "ошибкой шины", которую я не могу понять причину. Я никогда не слышал о GDB, но наткнулся на него на этом форуме и попытался использовать его в моей проблемной программе и получил следующий вывод:
% gdb Proc1 GNU gdb 5.0
...
Этот GDB был настроен как "sparc-sun-solaris2.8"...
(символы отладки не найдены)...
(GDB) запустить
Запуск программы: / home / 0 / vlcek / CSE660 / Lab3 / Proc1
(символы отладки не найдены)...
(символы отладки не найдены)...
(символы отладки не найдены)...
Программа получила сигнал SIGSEGV, Ошибка сегментации. 0x10a64 в основном ()
Я понятия не имею, что это значит, это говорит о том, что в моем коде в строке 10 есть ошибка? Если это так, строка 10 в моем коде просто "int main ()", поэтому я не уверен, что проблема там... Когда я пытаюсь запустить программу, все, что она говорит, это "Ошибка шины", поэтому я не уверен, где перейти отсюда. Я даже попытался поместить printf сразу после main, и он не печатает строку, а только выдает ошибку Bus.
Ниже мой код:
// 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);
}
После компиляции Proc1.c с флагом -ggdb и запуска gdb я получил следующее:
Программа получила сигнал SIGSEGV, Ошибка сегментации. 0x10a64 в main () в Proc1.c: 36
36 Аккаунт [0]=10000
Почему это может вызвать ошибку сегментации?
После изменения декларации Аккаунта на
int *Account = 0;
и добавление
printf("Account == %p\n", Account);
до счета [0] = 10000;
Я получаю следующее при запуске Proc1:
Account == ffffffff
Bus error
2 ответа
Чтобы получить более разумные результаты от GDB, вы должны скомпилировать вашу программу с -ggdb
вариант. Это будет включать отладочную информацию (например, номера строк) в вашу программу.
То, что вы сейчас видите, это адрес памяти (0x10a64
) счетчика программы. Это не очень поможет вам, если вы не сможете соотнести инструкции по сборке, которые вы там найдете, с частью вашей программы на Си.
Похоже, вы используете 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 в настоящее время в основном устарела, и вам следует избегать его использования, если можете.
В зависимости от вашего компилятора и параметров компилятора вы можете столкнуться с проблемой псевдонимов, поскольку вы преобразуете адрес своего Account
указатель. Эти старые интерфейсы не соответствуют современным правилам сглаживания, а это означает, что оптимизатор предполагает, что значение Account
не изменится
Также вы должны получить аргумент для shm_get
как можно ближе к ожидаемому типу. Попробуйте возможно что-то вроде следующего.
void volatile* shmRet;
int shmid = shm_get(123456, (void**) &shmRet, 3*sizeof(int));
int *Account = shmRet;
У меня не та же архитектура, поэтому я не знаю точный прототип вашего shm_get
но обычно это также плохая идея использовать фиксированные клавиши для этого типа функций. Должна быть какая-то функция, которая возвращает вам ключ для использования в вашем приложении.