2016-11-07 5 views
2

Я имеющая ошибку сегм при выполнении следующего кода:вина Сегментации с массивом предполагаемого размера

program test 

call sub ('dave') 

end program test 


subroutine sub (arr) 

character (*) :: arr 
character (20) :: upperc 

arr = upperc (arr) 

return 

end subroutine sub 


character (*) function upperc (str) 

integer i, l 
character (*) :: str 

upperc = str 
l = len (str) 
do i = 1, l 
    icode = ichar (str (i:i)) 
    if ((icode >= ichar ('a')).and.(icode <= ichar ('z'))) then 
    upperc (i:i) = char (ichar ('A') + icode - ichar('a')) 
    end if 
enddo 

return 

end function upperc 

Неисправности сегментной происходит на следующую строке:

arr = upperc (arr) 

Вот компиляция и GDB выход выполнения:

[[email protected] ~]$ gfortran -g -Wall test.f90 
[[email protected] ~]$ gdb 
GNU gdb (GDB) Red Hat Enterprise Linux 7.6.1-80.el7 
Copyright (C) 2013 Free Software Foundation, Inc. 
License GPLv3+: GNU GPL version 3 or later <http://gnu.org/licenses/gpl.html> 
This is free software: you are free to change and redistribute it. 
There is NO WARRANTY, to the extent permitted by law. Type "show copying" 
and "show warranty" for details. 
This GDB was configured as "x86_64-redhat-linux-gnu". 
For bug reporting instructions, please see: 
<http://www.gnu.org/software/gdb/bugs/>. 
(gdb) file ./a.out 
Reading symbols from /home/dave/a.out...done. 
(gdb) run 
Starting program: /home/dave/./a.out 

Program received signal SIGSEGV, Segmentation fault. 
0x00007ffff70f1fa3 in __memmove_ssse3_back() from /lib64/libc.so.6 
Missing separate debuginfos, use: debuginfo-install glibc- 2.17-106.el7_2.8.x86_64 libgcc-4.8.5-4.el7.x86_64 libgfortran-4.8.5-4.el7.x86_64 libquadmath-4.8.5-4.el7.x86_64 
(gdb) backtrace 
#0 0x00007ffff70f1fa3 in __memmove_ssse3_back() from /lib64/libc.so.6 
#1 0x000000000040078c in sub (arr='dave', _arr=4) at test.f90:13 
#2 0x00000000004008e9 in test() at test.f90:3 
#3 0x000000000040091f in main (argc=1, argv=0x7fffffffe7b3) at test.f90:5 
#4 0x00007ffff6fc3b15 in __libc_start_main() from /lib64/libc.so.6 
#5 0x0000000000400669 in _start() 
(gdb) q 
A debugging session is active. 

    Inferior 1 [process 20941] will be killed. 
[[email protected] ~]$ 

Я получаю странное ощущение, что что-то делать с тем фактом, что я передаю буквальный, но, возможно, не.

В попытке исправить это, я попытался следующий вместо этого, и она отлично работает:

program test2 

character(20) text 

text = 'dave' 

call sub (text) 

end program test2 


subroutine sub (arr) 

character (*) :: arr 
character (20) :: upperc 

arr = upperc (arr) 

return 

end subroutine sub 


character (*) function upperc (str) 

integer i, l 
character (*) :: str 

upperc = str 
l = len (str) 
do i = 1, l 
    icode = ichar (str (i:i)) 
    if ((icode >= ichar ('a')).and.(icode <= ichar ('z'))) then 
    upperc (i:i) = char (ichar ('A') + icode - ichar('a')) 
    end if 
enddo 

return 

end function upperc 

Может кто-нибудь сказать мне, что происходит?

+0

Да, это делать с прохождением буквального. Вы пытаетесь изменить значение константы. Попробуйте поместить 'намерение (inout)' на фиктивный аргумент, чтобы видеть. – francescalus

+1

В ответах на [этот связанный вопрос] (http://stackoverflow.com/q/16457423) есть много деталей (хотя и в несколько иной настройке, но многое еще применимо). – francescalus

ответ

2

Проблема заключается в том, что подпрограмма sub записывает в переменную text из программы test2. Если вы не указали переменную как agrument на sub, но только 'dave', как в первой программе test, Fortran намеревается перезаписать область памяти, которая не назначается переменной. Это приводит к ошибке сегментации.

Ниже вы видите ваш рабочий пример. Обратите внимание на два изменения:

  1. длина символа text был установлен в 6 вместо 20. Это было сделано, чтобы продемонстрировать, что вещь еще совмещаются, как подпрограмма sub принимает аргумент произвольной длины.

  2. Пишет 2 раз значения переменной text к консоли, и это было изменено call sub(text)

program test3 

character(6) text 

text = 'dave' 

write(*,*) text 

call sub (text) 

write(*,*) text 

end program test3 


subroutine sub (arr) 

character (*) :: arr 
character (20) :: upperc 

arr = upperc (arr) 

return 

end subroutine sub 


character (*) function upperc (str) 

integer i, l 
character (*) :: str 

upperc = str 
l = len (str) 
do i = 1, l 
    icode = ichar (str (i:i)) 
    if ((icode >= ichar ('a')).and.(icode <= ichar ('z'))) then 
    upperc (i:i) = char (ichar ('A') + icode - ichar('a')) 
    end if 
enddo 

return 

end function upperc