ошибка сегментации C и fortran

------ main.c ---------

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

int main()
{   
    char* lib_name = "./a.out";
    int array[5] = {1,2,3,4,5};
    int size_a = sizeof(array)/sizeof(int);            
    void* handle = dlopen(lib_name, RTLD_NOW);
    if (handle) {
        printf("[%s] dlopen(\"%s\", RTLD_NOW): incarcare finalizata\n", 
           __FILE__, lib_name);
    }
    else {
        printf("[%s] nu poate fi deschis: %s\n", __FILE__, dlerror());
        exit(EXIT_FAILURE);
    }
    void (*subrutine_fortran)(int*, int*) = dlsym(handle, "putere");
    if (subrutine_fortran) {
        printf("[%s] dlsym(handle, \"_set_name\"): simbol gasit\n", __FILE__);
    }
    else {
        printf("[%s] simbol negasit: %s\n", __FILE__, dlerror());
        exit(EXIT_FAILURE);
    }



    subrutine_fortran(&array,&size_a);
    //dlclose(handle);
    for(int i=1;i<4;i++) {
    array[i]=array[i]+1;
    }
}

------ привет.f90 --------

subroutine putere(a,h) bind(c)
    use ISO_C_BINDING
    implicit none
    integer(c_int) :: h
    integer(c_int), dimension(h) :: a
    integer i
    do concurrent (i=0:5)
        a(i)=a(i)*10
    end do
    !write (*,*) a
end subroutine

Когда я выполняю цикл по элементам массива:

for(int i=1;i<4;i++) {
  array[i]=array[i]+1;
}

Я получаю ошибку сегментации.

Этого не происходит, когда я пишу:

array[3]=array[3]+1

person tracius01    schedule 15.08.2011    source источник
comment
Если вы запустите свой код в отладчике, в какой строке возникает ошибка сегментации?   -  person Oliver Charlesworth    schedule 15.08.2011
comment
Массивы Fortran основаны на 1 или 0? (Вопрос не риторический.)   -  person zwol    schedule 15.08.2011
comment
Программа получила сигнал EXC_BAD_ACCESS, не удалось получить доступ к памяти. Причина: KERN_INVALID_ADDRESS по адресу: 0x0000000a00000df3 0x0000000a00000df3 в ?? () (gdb) где # 0 0x0000000a00000df3 в ?? () Невозможно получить доступ к памяти по адресу 0xa00000df3 # 1 0x0000000100000d0c в start ()   -  person tracius01    schedule 15.08.2011
comment
@Zack, я считаю, что Fortran использует массивы на основе 1. folk.uio.no/steikr/doc/f77/tutorial/arrays. html   -  person EMiller    schedule 15.08.2011


Ответы (1)


Ваш C-код таков:

int array[5] = {1,2,3,4,5};
int size_a = sizeof(array)/sizeof(int);            

subrutine_fortran(&array,&size_a);

и ваш код Fortran таков:

subroutine putere(a,h) bind(c)
    use ISO_C_BINDING
    implicit none
    integer(c_int) :: h
    integer(c_int), dimension(h) :: a
    integer i
    do concurrent (i=0:5)
        a(i)=a(i)*10
    end do
    !write (*,*) a
end subroutine

Это неверно в нескольких отношениях - как указывает Зак, массивы Fortran индексируются 1 (даже если они поступают откуда-то еще, например, C). Таким образом, он должен начинаться с 1. Кроме того, если бы 0 был правильным, размер был бы неправильным. Вы хотите что-то вроде

    do concurrent (i=1:h)

С этим изменением у меня все работает.

person Jonathan Dursi    schedule 15.08.2011
comment
Я не хочу претендовать на то, что знал, что массивы Fortran основаны на 1. Смутное воспоминание о том, что где-то это читал, да. - person zwol; 15.08.2011