Note
Go to the end to download the full example code.
CPU & GPU backends
Details about the CPU & GPU backend of the package.
PyEPRI was designed in a modular fashion. The functions of this
package rely on standard datatypes (mostly arrays) and not on specific
classes. The only nonstandard object involved in this package is a
backend system whose role is to facilitate CPU & GPU compatible
scripting and development. More precisely, the PyEPR backend system
allows the use of the libraries {numpy
, cupy
, torch
} in a
unified framework. Deep understanding of this tutorial is not
mandatory but taking a quick look at this tutorial will facilitate
the understanding of all upcoming EPR imaging demo examples.
What is a PyEPRI backend?
Although most common functions of the numpy
libraries are usually
also available in the cupy
, and torch
libraries, those
functions rely on library-dependent types (e.g., numpy.ndarray
,
cupy.ndarray
,``torch.Tensor``, …) and library-dependent
datatypes (e.g., numpy.float32
, cupy.float32
,
torch.float32
). Besides, parameter naming conventions or default
values oftenly differ between those libraries. The role of the backend
is to remap those library-dependent types, datatypes and common
functions towards a standardized counterpart (examples are provided
below).
Technically, a PyEPRI backend is an instance of the
pyepri.backends.Backend
class and must be passed as input of
most functions of this package. It is also recommended to prefer the
usage of the backend commends (e.g., backend.abs
,
backend.meshgrid
, backend.rand
, …) to their
library-dependent counterparts in your own script. Doing so, moving
from a CPU based computation framework with numpy to a GPU based
computation framework with cupy or PyTorch is possible by simply
changing the backend instance (keeping the rest of the script
unchanged) as we systematically do in all provided EPR imaging demo
examples.
Remark: The available backend functions have been restricted to the functions needed in the internal submodules and in the tutorial and example scripts. The backend functionalities may be extended in future releases of the PyEPRI package, depending on the needs.
Numpy backend (CPU)
Since numpy
is systematically installed with pyepri
(as a
mandatory dependency of this package), you should be able to create
a numpy backend using the following commands.
# ---------------------- #
# Create a numpy backend #
# ---------------------- #
import pyepri.backends as backends
backend = backends.create_numpy_backend()
Once the backend
object is created, it can be passed to any
PyEPRI function requiring a backend input parameter. Doing so, all
operations within the functions will be performed on the CPU using
the Numpy library.
You can also access to a bunch of standard functions on arrays,
e.g. backend.abs
, backend.cos
, backend.sin
,
backend.linspace
, backend.meshgrid
, backend.rand
, and
many more. Those functions are lambda functions that simply remap
their inputs to the approriate library-dependent function (in this
case, to the functions numpy.abs
, numpy.cos
, numpy.sin
,
numpy.linspace
, numpy.meshgrid
, numpy.random.rand
, etc.)
For instance the next commands call the backend.rand
function to
generate a random array with shape (3, 4)
:
a = backend.rand(3, 4, dtype='float32')
print('type(a) is: %s' % type(a))
print('a.dtype is: %s' % a.dtype)
type(a) is: <class 'numpy.ndarray'>
a.dtype is: float32
The type and datatype of the generated array will depend on the backend instance as summarized below.
Backend library |
Type of a |
Datatype of a |
---|---|---|
numpy |
numpy.ndarray |
numpy.float32 |
cupy |
cupy.ndarray |
cupy.float32 |
torch |
torch.Tensor |
torch.float32 |
Here the backend instance is a numpy backend so the type and
datatype of the array a
are respectively numpy.ndarray
and
numpy.float32
.
In practice, the backend instance provides a way to avoid calling
the library dependent functions numpy.random.rand
,
cupy.random.rand
and torch.rand
as well as the library
dependent datatypes numpy.float32
, cupy.float32
and
torch.float32
when developing a script. When the backend
instance is changed, the function mapping is changed but those
change have no impact on the content of the script. Only the choice
of the backend instance will determine how the execution is
performed.
Note that a mini-documentation is provided for each standardized
function of a backend instance. The documentation can be displayed
using the help()
function, as illustrated below for the
backend.rand
function.
help(backend.rand)
Help on function <lambda> in module pyepri.backends:
<lambda> lambda *dims, dtype='float32'
return numpy.random.rand(*dims).astype(dtype)
More information about the standardized naming conventions and
remapping functionalities are available in the
pyepri.backends
submodule documentation.
Cupy backend (GPU)
If the Cupy library is installed on your system (see the installation instructions for the installation of this library as an optional dependency of the PyEPRI package), you should be able to instantiate a cupy backend by uncommenting the next command (an error will be raised if Cupy is not installed).
# --------------------------- #
# Create a cupy backend (GPU) #
# --------------------------- #
#backend = backends.create_cupy_backend() # uncomment here for a CUPY backend
Then, you can execute again the backend.rand
commands presented
above and check that the computed array has now the type
cupy.ndarray
and datatype cupy.float32
.
PyTorch backend (CPU or GPU)
If the PyTorch library is installed on your system (see the installation instructions for the installation of this library as an optional dependency of the PyEPRI package), you should be able to instantiate a torch backend using either a CPU or a GPU device by uncommenting the next commands (an error will be raised if PyTorch is not installed or if the requested device is not available).
# -------------------------------- #
# Create a torch-cpu backend (CPU) #
# -------------------------------- #
#backend = backends.create_torch_backend('cpu') # uncomment here for a PyTorch-CPU backend
# --------------------------------- #
# Create a torch-cuda backend (GPU) #
# --------------------------------- #
device = 'cuda' # select GPU device (if several GPU are available, you
# can explicitly select the device index by using
# device = 'cuda:0', device = 'cuda:1', etc.)
#backend = backends.create_torch_backend(device=device) # uncomment here for a PyTorch GPU backend
Now, by executing again the backend.rand
commands presented
above, you can check that the computed array has now the type
torch.Tensor
and datatype torch.float32
.
Total running time of the script: (0 minutes 0.453 seconds)
Estimated memory usage: 236 MB