Skip to main content

PWR073: Transform common block into a module for better data encapsulation

Issue

Common blocks create globally accessible memory, complicating data flow analysis, and present error-prone characteristics that can lead to dificult-to-diagnose bugs.

Actions

To enhance code maintainability and safety, encapsulate data within modules that offer controlled access interfaces, rather than using common blocks.

Relevance

The common block construct declares a memory area that is accessible throughout the entire program. Such global accessibility can complicate the interpretation and maintenance of the program, as modifications in one part of the code can unexpectedly affect other areas. This is particularly risky when multiple components of the program depend on this shared data.

Furthermore, each procedure that accesses a common block must re-declare it, manually ensuring that all variables are defined in the same order. This requirement introduces significant risks for the programmer due to:

  • Lack of name consistency: In each re-declaration, different names can be assigned to the same memory location, complicating logical reasoning about the program.

  • Lack of type safety: It is also possible to associate different data types to the same memory location, potentially leading to memory corruption and other subtle errors.

Code examples

Refactoring common blocks into modules can be complex and impact multiple source files. Therefore, an incremental approach is recommended:

  1. Migrate all variables from the common block to a new module, including any block data initializations. Replace the common block with the new module in all affected locations.

  2. Gradually narrow the scope of the module by importing only the necessary variables where needed.

  3. Consider implementing procedures to manage access to the module's variables, thus further enforcing data encapsulation.

Let's start with an old-style program that relies on common blocks:

! example.f90
program test_common_block
use iso_fortran_env, only: real32
implicit none
real(kind=real32) :: var1
integer :: var2
common /my_common/ var1, var2

var1 = 3.14
var2 = 20

call printVar1
call printVar2

contains

subroutine printVar1
implicit none
real(kind=real32) :: var1
common /my_common/ var1

print *, "Var1: ", var1
end subroutine printVar1

subroutine printVar2
implicit none
integer :: var2
common /my_common/ var2

print *, "Var2: ", var2

! Did you spot the bug?
!
! In the common block re-definition, `var1` is missing. As a result, `var2`
! unintentionally references the memory location of `var1`.
!
! This lack of naming consistency, as well as type safety (note that `var1`
! also contains real data), make this error go easily unnoticed, as the code
! still compiles.
end subroutine printVar2
end program test_common_block

The inherent unsafety of common blocks leads to this unexpected output:

$ gfortran --version
GNU Fortran (Debian 12.2.0-14) 12.2.0
$ gfortran example.f90
$ ./a.out
Var1: 3.14000010
Var2: 1078523331

In contrast, modules prevent such issues by design. Let's refactor the program to encapsulate the variables inside a module (Step 1). Due to the simplicity of this example, the only keyword (Step 2) is already leveraged:

! solution_without_private.f90
module my_module
use iso_fortran_env, only: real32
implicit none
public
real(kind=real32) :: var1
integer :: var2
end module my_module

program test_module
use my_module, only: var1, var2
implicit none

var1 = 3.14
var2 = 20

call printVar1
call printVar2

contains

subroutine printVar1
use my_module, only: var1
implicit none

print *, "Var1: ", var1
end subroutine printVar1

subroutine printVar2
use my_module, only: var2
implicit none

print *, "Var2: ", var2
end subroutine printVar2
end program test_module

And now the program gives the correct results:

$ gfortran solution_without_private.f90
$ ./a.out
Var1: 3.14000010
Var2: 20

Depending on your preferences, you might also set the module variables as private and create procedures to manage access to the contained data (e.g., getters and setters):

! solution.f90
module my_module
use iso_fortran_env, only: real32
implicit none
private
real(kind=real32) :: var1
integer :: var2
public :: getVar1, setVar1, getVar2, setVar2

contains

real(kind=real32) function getVar1()
getVar1 = var1
end function getVar1

subroutine setVar1(value)
real(kind=real32), intent(in) :: value
var1 = value
end subroutine setVar1

integer function getVar2()
getVar2 = var2
end function getVar2

subroutine setVar2(value)
integer, intent(in) :: value
var2 = value
end subroutine setVar2
end module my_module

program test_module
use my_module, only: setVar1, setVar2
implicit none

call setVar1(3.14)
call setVar2(20)

call printVar1
call printVar2

contains

subroutine printVar1
use my_module, only: getVar1
implicit none

print *, "Var1: ", getVar1()
end subroutine printVar1

subroutine printVar2
use my_module, only: getVar2
implicit none

print *, "Var2: ", getVar2()
end subroutine printVar2
end program test_module

References