From ac8c6b6c6fcf42f01df80c2209ed93a7004e556d Mon Sep 17 00:00:00 2001 From: Mathieu Faverge <mathieu.faverge@inria.fr> Date: Mon, 14 May 2018 22:42:55 +0200 Subject: [PATCH] Add f90 example --- wrappers/fortran90/CMakeLists.txt | 33 +++-- wrappers/fortran90/examples/spm_driver.f90 | 70 +++++++++ wrappers/fortran90/examples/spm_user.f90 | 156 +++++++++++++++++++++ 3 files changed, 242 insertions(+), 17 deletions(-) create mode 100644 wrappers/fortran90/examples/spm_driver.f90 create mode 100644 wrappers/fortran90/examples/spm_user.f90 diff --git a/wrappers/fortran90/CMakeLists.txt b/wrappers/fortran90/CMakeLists.txt index a1a40c22..439a7beb 100644 --- a/wrappers/fortran90/CMakeLists.txt +++ b/wrappers/fortran90/CMakeLists.txt @@ -37,21 +37,20 @@ install(TARGETS spmf # # Add examples # -#set (EXAMPLES -# fsimple.f90 -# flaplacian.f90 -# fstep-by-step.f90 -# ) -# -#foreach (_file ${EXAMPLES}) -# get_filename_component(_name_we ${_file} NAME_WE) -# add_executable(${_name_we} examples/${_file}) -# target_link_libraries(${_name_we} pastixf) -# -# install(TARGETS ${_name_we} RUNTIME DESTINATION examples ) -# install(FILES examples/${_file} DESTINATION examples ) -# -# add_test(fortran_${_name_we} ./${_name_we}) -# -#endforeach() +set (EXAMPLES + spm_driver.f90 + spm_user.f90 + ) + +foreach (_file ${EXAMPLES}) + get_filename_component(_name_we ${_file} NAME_WE) + add_executable(${_name_we} examples/${_file}) + target_link_libraries(${_name_we} spmf) + + install(TARGETS ${_name_we} RUNTIME DESTINATION examples ) + install(FILES examples/${_file} DESTINATION examples ) + + add_test(fortran_${_name_we} ./${_name_we}) + +endforeach() diff --git a/wrappers/fortran90/examples/spm_driver.f90 b/wrappers/fortran90/examples/spm_driver.f90 new file mode 100644 index 00000000..fc7f05e4 --- /dev/null +++ b/wrappers/fortran90/examples/spm_driver.f90 @@ -0,0 +1,70 @@ +! +! @file spm_driver.f90 +! +! Fortran 90 example using a matrix read with the spm driver. +! +! @copyright 2017-2017 Bordeaux INP, CNRS (LaBRI UMR 5800), Inria, +! Univ. Bordeaux. All rights reserved. +! +! @version 6.0.0 +! @author Mathieu Faverge +! @date 2017-01-01 +! +program spm_driver + use iso_c_binding + use spmf + ! use mpi_f08 + implicit none + + type(spmatrix_t), target :: spm + type(spmatrix_t), pointer :: spm2 + real(kind=c_double) :: normA + real(kind=c_double) :: eps = 1.e-15 + integer(c_int) :: info + integer(kind=spm_int_t) :: nrhs + real(kind=c_double), dimension(:,:), allocatable, target :: x0, x, b + type(c_ptr) :: x0_ptr, x_ptr, b_ptr + + ! + ! Initialize the problem + ! 1- The matrix + call spmReadDriver( SpmDriverLaplacian, "d:10:10:10:4.", spm, info ) + + call spmCheckAndCorrect( spm, spm2 ) + if (.not. c_associated(c_loc(spm), c_loc(spm2))) then + call spmExit( spm ) + spm = spm2 + end if + + call spmPrintInfo( spm ) + + ! Scale A for better stability with low-rank computations + call spmNorm( SpmFrobeniusNorm, spm, normA ) + call spmScalMatrix( 1. / normA, spm ) + + ! 2- The right hand side + nrhs = 10 + allocate(x0(spm%n, nrhs)) + allocate(x( spm%n, nrhs)) + allocate(b( spm%n, nrhs)) + x0_ptr = c_loc(x0) + x_ptr = c_loc(x) + b_ptr = c_loc(b) + + ! Compute b = A * x, with x random + call spmGenRHS( SpmRhsRndX, nrhs, spm, x0_ptr, spm%n, b_ptr, spm%n, info ) + + ! Copy x0 into x + x = x0 + + ! + ! Check the solution + ! + call spmCheckAxb( eps, nrhs, spm, x0_ptr, spm%n, b_ptr, spm%n, x_ptr, spm%n, info ) + + call spmExit( spm ) + deallocate(x0) + deallocate(x) + deallocate(b) + +end program spm_driver diff --git a/wrappers/fortran90/examples/spm_user.f90 b/wrappers/fortran90/examples/spm_user.f90 new file mode 100644 index 00000000..cdf8df9b --- /dev/null +++ b/wrappers/fortran90/examples/spm_user.f90 @@ -0,0 +1,156 @@ +! +! @file spm_user.f90 +! +! Fortran 90 example using a laplacian matrix. +! +! @copyright 2015-2017 Bordeaux INP, CNRS (LaBRI UMR 5800), Inria, +! Univ. Bordeaux. All rights reserved. +! +! @version 6.0.0 +! @author Mathieu Faverge +! @date 2017-01-01 +! +program spm_user + use iso_c_binding + use spmf + implicit none + + integer(kind=spm_int_t), dimension(:), allocatable, target :: rowptr + integer(kind=spm_int_t), dimension(:), allocatable, target :: colptr + real(kind=c_double), dimension(:), allocatable, target :: values + real(kind=c_double), dimension(:,:), allocatable, target :: x0, x, b + type(c_ptr) :: x0_ptr, x_ptr, b_ptr + real(kind=c_double) :: eps = 1.e-15 + type(spmatrix_t), target :: spm + type(spmatrix_t), pointer :: spm2 + integer(kind=spm_int_t) :: dim1, dim2, dim3, n, nnz + integer(kind=spm_int_t) :: i, j, k, l, nrhs + integer(c_int) :: info + + ! + ! Generate a 10x10x10 complex Laplacian in IJV format + ! + dim1 = 10 + dim2 = 10 + dim3 = 10 + n = dim1 * dim2 * dim3 + nnz = (2*(dim1)-1) * dim2 * dim3 + (dim2-1)*dim1*dim3 + dim2*dim1*(dim3-1) + + allocate(rowptr(nnz)) + allocate(colptr(nnz)) + allocate(values(nnz)) + + l = 1 + do i=1,dim1 + do j=1,dim2 + do k=1,dim3 + rowptr(l) = (i-1) + dim1 * (j-1) + dim1 * dim2 * (k-1) + 1 + colptr(l) = (i-1) + dim1 * (j-1) + dim1 * dim2 * (k-1) + 1 + values(l) = 6. + + if (i == 1) then + values(l) = values(l) - 1. + end if + if (i == dim1) then + values(l) = values(l) - 1. + end if + if (j == 1) then + values(l) = values(l) - 1. + end if + if (j == dim2) then + values(l) = values(l) - 1. + end if + if (k == 1) then + values(l) = values(l) - 1. + end if + if (k == dim3) then + values(l) = values(l) - 1. + end if + + values(l) = values(l) * 8. + l = l + 1 + + if (i < dim1) then + rowptr(l) = i + dim1 * (j-1) + dim1 * dim2 * (k-1) + 1 + colptr(l) = (i-1) + dim1 * (j-1) + dim1 * dim2 * (k-1) + 1 + values(l) = - 1. - 1. * I + l = l + 1 + end if + if (j < dim2) then + rowptr(l) = (i-1) + dim1 * j + dim1 * dim2 * (k-1) + 1 + colptr(l) = (i-1) + dim1 * (j-1) + dim1 * dim2 * (k-1) + 1 + values(l) = - 1. - 1. * I + l = l + 1 + end if + if (k < dim3) then + rowptr(l) = (i-1) + dim1 * (j-1) + dim1 * dim2 * k + 1 + colptr(l) = (i-1) + dim1 * (j-1) + dim1 * dim2 * (k-1) + 1 + values(l) = -1. - 1. * I + l = l + 1 + end if + end do + end do + end do + + if ( l .ne. nnz+1 ) then + write(6,*) 'l ', l, " nnz ", nnz + end if + + ! + ! Create the spm out of the internal data + ! + call spmInit( spm ) + spm%mtxtype = SpmSymmetric + spm%flttype = SpmDouble + spm%fmttype = SpmIJV + spm%n = n + spm%nnz = nnz + spm%dof = 1 + spm%rowptr = c_loc(rowptr) + spm%colptr = c_loc(colptr) + spm%values = c_loc(values) + + call spmUpdateComputedFields( spm ) + + call spmCheckAndCorrect( spm, spm2 ) + if (.not. c_associated(c_loc(spm), c_loc(spm2))) then + deallocate(rowptr) + deallocate(colptr) + deallocate(values) + + spm%rowptr = c_null_ptr + spm%colptr = c_null_ptr + spm%values = c_null_ptr + + call spmExit( spm ) + spm = spm2 + end if + + call spmPrintInfo( spm ) + + ! 2- The right hand side + nrhs = 10 + allocate(x0(spm%n, nrhs)) + allocate(x( spm%n, nrhs)) + allocate(b( spm%n, nrhs)) + x0_ptr = c_loc(x0) + x_ptr = c_loc(x) + b_ptr = c_loc(b) + + ! Compute b = A * x, with x random + call spmGenRHS( SpmRhsRndX, nrhs, spm, x0_ptr, spm%n, b_ptr, spm%n, info ) + + ! Copy x0 into x + x = x0 + + ! + ! Check the solution + ! + call spmCheckAxb( eps, nrhs, spm, x0_ptr, spm%n, b_ptr, spm%n, x_ptr, spm%n, info ) + + call spmExit( spm ) + deallocate(x0) + deallocate(x) + deallocate(b) + +end program spm_user -- GitLab