Skip to content

Commit a939815

Browse files
committed
add ABI check, improve C++ code and output check
1 parent 081ff60 commit a939815

12 files changed

+73
-71
lines changed

CMakeLists.txt

+4-9
Original file line numberDiff line numberDiff line change
@@ -11,17 +11,12 @@
1111
cmake_minimum_required(VERSION 3.14)
1212

1313
project(Fortran_C_CXX_interface
14-
LANGUAGES C Fortran CXX)
14+
LANGUAGES C Fortran CXX
15+
VERSION 1.1.0)
1516

1617
include(CTest)
1718

18-
if(CMAKE_Fortran_COMPILER_ID STREQUAL GNU)
19-
# this helps show the options are/aren't conflicting between C and Fortran
20-
# at build time
21-
string(APPEND CMAKE_Fortran_FLAGS -fimplicit-none)
22-
endif()
23-
24-
add_compile_options(-Wall)
19+
include (cmake/compilers.cmake)
2520

2621
add_library(c_lib call_c.c)
2722
target_compile_features(c_lib PRIVATE c_std_99)
@@ -40,7 +35,7 @@ add_test(NAME CXXCallFortran COMMAND cxx_call_f)
4035
add_library(cxx_lib OBJECT call_cxx.cxx)
4136
target_compile_features(cxx_lib PRIVATE cxx_std_11)
4237

43-
add_executable(f_call_cxx f_call_cxx.f90 $<TARGET_OBJECTS:cxx_lib>)
38+
add_executable(f_call_cxx f_call_c.f90 $<TARGET_OBJECTS:cxx_lib>)
4439
# LINKER_LANGUAGE option is necessary for ifort at least
4540
set_target_properties(f_call_cxx PROPERTIES LINKER_LANGUAGE Fortran)
4641
add_test(NAME FortranCallC++ COMMAND f_call_cxx)

c_call_f.f90

+5-5
Original file line numberDiff line numberDiff line change
@@ -1,17 +1,17 @@
11
module mytest
22

3-
use, intrinsic:: iso_c_binding, only: dp=>c_double, c_int
3+
use, intrinsic:: iso_c_binding, only: c_int, C_SIZE_T
44

55
implicit none (type, external)
66

77
contains
88

9-
pure subroutine timestwo(z,z2,N) bind(c)
9+
pure subroutine timestwo(z, z2, N) bind(c)
1010
! elemental is not allowed with BIND(C)
1111

12-
integer(c_int), intent(in) :: N
13-
real(dp),intent(in) :: z(N)
14-
real(dp),intent(out) :: z2(N)
12+
integer(C_SIZE_T), intent(in) :: N
13+
integer(c_int),intent(in) :: z(N)
14+
integer(c_int),intent(out) :: z2(N)
1515

1616
z2 = 2*z
1717

call_c.c

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
void timestwo(double x[], double x2[], int N){
1+
void timestwo(int x[], int x2[], int N){
22
for (int i=0; i<N; i++)
33
x2[i] = x[i] * 2;
44
}

call_cxx.cxx

+1-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@
33
// nm libcxx.a
44
// or whatever you name this library file.
55

6-
extern "C" void timestwo(double x[], double x2[], int N){
6+
extern "C" void timestwo(int x[], int x2[], int N){
77
for (auto i=0; i<N; i++)
88
x2[i] = x[i] * 2;
99
}

cmake/abi_check/CMakeLists.txt

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
cmake_minimum_required(VERSION 3.14)
2+
project(abi_check LANGUAGES C Fortran)
3+
4+
add_library(addone OBJECT addone.c)
5+
6+
add_executable(main main.f90 $<TARGET_OBJECTS:addone>)

cmake/abi_check/addone.c

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
int addone(int N){
2+
return N + 1;
3+
}

cmake/abi_check/main.f90

+17
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
program adder
2+
3+
implicit none (type, external)
4+
5+
interface
6+
integer function addone(N) bind(C)
7+
integer, intent(in), value :: N
8+
end function addone
9+
end interface
10+
11+
12+
if (addone(2) /= 3) error stop "unexpected addone result"
13+
14+
print *, "OK: 2+1=3"
15+
16+
17+
end program

cmake/compilers.cmake

+18
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
# check C and Fortran compiler ABI compatibility
2+
3+
try_compile(abi_ok ${CMAKE_CURRENT_BINARY_DIR}/abi_check ${CMAKE_CURRENT_LIST_DIR}/abi_check abi_check)
4+
if(abi_ok)
5+
message(STATUS "C and Fortran compiler detected to be ABI-compatible.")
6+
else()
7+
message(FATAL ERROR "C compiler {CMAKE_C_COMPILER_ID} {CMAKE_C_COMPILER_VERSION} and Fortran compiler ${CMAKE_Fortran_COMPILER_ID} ${CMAKE_Fortran_COMPILER_VERSION} are ABI-incompatible.")
8+
endif()
9+
10+
11+
if(CMAKE_Fortran_COMPILER_ID STREQUAL GNU)
12+
# this helps show the options are/aren't conflicting between C and Fortran
13+
# at build time
14+
string(APPEND CMAKE_Fortran_FLAGS -fimplicit-none)
15+
add_compile_options(-Wextra)
16+
endif()
17+
18+
add_compile_options(-Wall)

cxx_call_f.cxx

+10-8
Original file line numberDiff line numberDiff line change
@@ -1,22 +1,24 @@
11
#include <iostream>
22
#include <vector>
33

4-
extern "C" void timestwo(double [],double [],int*);
4+
extern "C" void timestwo(int [], int [], size_t*);
55

66

77
int main()
88
{
9-
auto N = 3;
10-
std::vector<double> x(N);
11-
std::vector<double> x2(N);
129

13-
for (auto i = 0; i < x.size(); i++)
14-
x[i] = i;
10+
std::vector<int> x = {0, 1, 2};
11+
auto N = x.size();
12+
std::vector<int> x2(N);
1513

1614
timestwo(&x.front(), &x2.front(), &N);
1715

18-
for (auto i: x2)
19-
std::cout << i << ' ';
16+
for (auto i=0u; i < x2.size(); i++){
17+
if (x2[i] != 2*x[i]){
18+
fprintf(stderr, "value %d != %d", x2[i], x[i]);
19+
return EXIT_FAILURE;
20+
}
21+
}
2022

2123
return EXIT_SUCCESS;
2224
}

f_call_c.f90

+7-9
Original file line numberDiff line numberDiff line change
@@ -1,25 +1,23 @@
11
program call_c
2-
!! Demonstrate Fortran calling C.
2+
!! Demonstrate Fortran calling C or C++.
33
!!
44
!! Normally BIND(C) should be used after the function name in the interface block,
55
!! rather than postpending underscore(s).
66

7-
use, intrinsic :: iso_c_binding, only: dp=>c_double, c_int
7+
use, intrinsic :: iso_c_binding, only: c_int
88
implicit none (type, external)
99

1010

1111
interface
12-
13-
subroutine timestwo(x, x2, N) bind (c)
14-
import c_int, dp
15-
integer(c_int), value :: N
16-
real(dp) :: x(N), x2(N)
12+
subroutine timestwo(a, a2, L) bind (c)
13+
import c_int
14+
integer(c_int), value :: L
15+
integer(c_int) :: a(L), a2(L)
1716
end subroutine timestwo
18-
1917
end interface
2018

2119
integer(c_int) :: N, i
22-
real(dp), allocatable :: x(:), x2(:)
20+
integer(c_int), allocatable :: x(:), x2(:)
2321

2422
N = 3
2523

f_call_cxx.f90

-37
This file was deleted.

meson.build

+1-1
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,7 @@ f_call_c = executable('f_call_c', 'f_call_c.f90',
1515
test('Fortran calling C', f_call_c, timeout: 10)
1616

1717
cxx_lib = library('call_cxx', 'call_cxx.cxx')
18-
f_call_cxx = executable('f_call_cxx', 'f_call_cxx.f90',
18+
f_call_cxx = executable('f_call_cxx', 'f_call_c.f90',
1919
link_with: cxx_lib,
2020
link_language: 'fortran')
2121
test('Fortran calling C++', f_call_cxx, timeout: 10)

0 commit comments

Comments
 (0)