1

我使用了一些从 C++ 调用 Fortran 例程的科学计算代码,它突然开始在 gcc 6 下发出警告。这是准系统问题:

mult考虑定义在 中的 Fortran 子例程mult.f90

subroutine mult(c)
  complex*16 c
  c = c * c
  return
end

我从 C++ 文件中调用它test.cpp

#include <complex>
#include <iostream>

extern "C" void mult_(std::complex<double> *);

int main() {
  std::complex<double> z (1,0);
  mult_(&z);

  std::cout << z << "\n";
  return 0;
}

当我使用 g++-6 编译文件时,我收到以下警告:

$ g++-6 -O3 -W -Wall test.cpp mult.f90 -flto -o test2

test.cpp:4:17: warning: type of ‘mult_’ does not match original declaration [-Wlto-type-mismatch]
 extern "C" void mult_(std::complex<double> *);
                 ^
mult.f90:1:1: note: ‘mult’ was previously declared here
 subroutine mult(c)
 ^
mult.f90:1:1: note: code may be misoptimized unless -fno-strict-aliasing is used

如果我执行以下任何操作,警告就会消失:

  • 将 g++-6(我的版本是 6.2.0)替换为 g++-5(版本 5.4.1)
  • -flto不带标志编译
  • 使用 double(而不是 std::complex)和 real*8(而不是 complex*16)

我应该担心,还是我可以忽略这个警告?在前一种情况下,我该如何解决这个问题?

4

1 回答 1

1

我能找到的最接近的相关问题是 https://gcc.gnu.org/bugzilla/show_bug.cgi?id=78562#c6

编译器警告似乎不相关。

我用 C++ 编译了以下代码,并将生成的汇编程序与 fortran 代码进行了比较

extern "C" 
{
  void mult_(std::complex<double> *z)
  {
      *z = *z * *z; 
  }
}

查看结果:实现略有不同(fortran 似乎甚至不使用单个 GP 寄存器,而 C++ 使用 RBX)但调用约定等完全相同,因此您不必担心

rep ~ $ g++ -S -O3 -Wall test2.cpp
rep ~ $ g++ -S -O3 mult.f90
rep ~ $ cat mult.s

-----------------------------Snip----------------------------

.LFB0:
        .cfi_startproc
        movsd   (%rdi), %xmm0
        movsd   8(%rdi), %xmm1
        movapd  %xmm0, %xmm2
        movapd  %xmm1, %xmm3
        mulsd   %xmm0, %xmm2
        mulsd   %xmm1, %xmm3
        mulsd   %xmm1, %xmm0
        subsd   %xmm3, %xmm2
        addsd   %xmm0, %xmm0
        movsd   %xmm2, (%rdi)
        movsd   %xmm0, 8(%rdi)
        ret
        .cfi_endproc

-----------------------------Snip----------------------------

rep ~ $ cat test2.s

-----------------------------Snip----------------------------

.LFB1991:
        .cfi_startproc
        movsd   8(%rdi), %xmm3
        pushq   %rbx
        .cfi_def_cfa_offset 16
        .cfi_offset 3, -16
        movsd   (%rdi), %xmm2
        movq    %rdi, %rbx
        movapd  %xmm3, %xmm1
        movapd  %xmm2, %xmm0
        call    __muldc3
        movsd   %xmm0, (%rbx)
        movsd   %xmm1, 8(%rbx)
        popq    %rbx
        .cfi_def_cfa_offset 8
        ret
        .cfi_endproc

-----------------------------Snip----------------------------
rep ~ $
于 2016-12-08T15:29:16.297 回答