See also the "detailed" documentation
These routines provide some of the features of the posix or Windows memory mapped files under a simple and unique Fortran interface.
2 main usages:
- allocating arrays that are potentially bigger than the RAM+swap size, and which are backed by a temporary file
- opening existing files or creating new files (still potentially bigger than the RAM+swap size), and mapping them to an array
Private mapping is possible, with optional write-back of the modifications to the file. Such mapping can be useful when one doesn't want to modify the file on disk, or when one wants to work only in memory and decide when to update (or not) the file.
Anonmymous mapping is also possible, i.e. allocating virtual memory without a physical backing file. This is actually what the C malloc() generally does (and therefore also what the Fortran allocate() does) when the allocated size is above some threshold. This option is provided to easily switch between on-disk and in-memory only modes by just changing an argument in the calls.
The interface manipulates either bytes or array elements and returns a C pointer, which the user has to convert to a Fortran pointer.
The programmer can either manage themselves the conversion between elements and bytes (the module provides utility functions for the conversions), or provide the mapping routine an optional mold=
argument with a variable of the same type+kind as the array that will receive the mapping. e.g. these 2 codes are equivalent:
type(fmmap_t) :: x
real(real64), pointer :: a(:)
integer(c_size_t) :: n = 10**9, nbytes
if (approach_1) then
nbytes = fmmap_e2b( n, storage_size(a) )
call x%create( FMMAP_SCRATCH, "", nbytes )
else if (approach_2) then
call x%create( FMMAP_SCRATCH, "", n, mold=1.0_real64 )
end if
call c_f_pointer( x%cptr(), [n] )
use iso_C_binding, cst => c_size_t
use fmmap
type sometype
integer :: i
double precision :: a
character(len=7) :: str
end type
type(sometype), pointer :: pt(:)
type(fmmap_t) :: x
integer(cst) :: n, length
...
...
n = 10_cst ** 9 !! can be larger than RAM+swap space
!> converts n elements to a number of bytes
length = fmmap_e2b(n, storage_size(pt))
!> creates a mapping to a temporary file
call fmmap_create(x, FMMAP_SCRATCH, "", length)
!> conversion to a Fortran pointer
call c_f_pointer(x%cptr(), pt, [n])
!> work on pt(:) as if it was a classical array
! ...
! ...
!> closes the mapping and delete the file
call fmmap_destroy(x)
use iso_C_binding, cst => c_size_t
use fmmap
integer, pointer, contiguous :: pi(:,:), tmpi(:,:)
type(fmmap_t) :: x
integer(cst) :: n, length
...
n = 1000_cst !! can be larger than RAM+swap space
!> converts n*n elements to a number of bytes
length = fmmap_elem2byte(n*n, storage_size(pi))
!> Mapping to a new named file
call x%create( FMMAP_NEW, "./foo1.bin", length)
!> conversion to a Fortran pointer, in 2 stages because we need a lower bound /= 1
call c_f_pointer(x%cptr(), tmpi, [n,n])
pi(0:n-1,1:n) => tmpi
!> work on pi(:,:) as if it was a classical array
! ...
! ...
!> closes the mapping (the file is NOT deleted)
call x%destroy()
use iso_C_binding, cst => c_size_t
use fmmap
integer, pointer :: pi(:)
type(fmmap_t) :: x
integer(cst) :: n, length
...
!> Mapping to a existing named file
call x%create( FMMAP_OLD, "./foo1.bin", n, mold=0, private=.true.)
!> Conversion to a Fortran pointer
call c_f_pointer( x%cptr(), pi, [n] )
!> work on pi(:) as if it was a classical array
!> All the changes are resident only in memory, the file is unmodified
! ...
! ...
if (...) then
!> Closes the mapping
!> All the changes are lost and the original file is kept
call x%destroy()
else
!> Alternatively...
!> All the changes are written back to the file before unmapping
call x%destroy( writeback=.true. )
end if
The repository has an fpm (Fortran Package Manager) structure:
fpm test
On Windows, the presence of the _WIN32
macro is assumed
macOS 15 / gcc-gfortran 14
Windows 11 MSYS2 / gcc-gfortran 13
Linux Debian 11 / Intel icc-ifort 2021
v0.11.3: Lubuntu 22.04 / gcc-gfortran 11
Under Windows MSYS2 the _WIN32
macro is not defined in gfortran (while it is in gcc). I don't know how it is under environments others than MSYS2. The fix is to pass it explicitly: fpm test --flag "-D_WIN32"
- It is assumed that the Fortran file storage unit is a byte. This is checked in the routines.
- The whole file is mapped, it's not possible to map only a portion of it
- The access is always read+write, there's no option to restrict it
- The file is always opened with non-blocking access (i.e. it can be opened again by the same process or another process), which corresponds to the usual behavior on posix systems but not Windows.
- Mapping to an array of a derived type containing allocatable or pointer components is not allowed (well, it's technically possible, but the memory allocated by these components won't be part of the mapping).
None...