This GitHub repository offers a versatile and comprehensive collection of deep learning models that are specifically designed to restore and enhance various electron microscopy modalities. Developed by Ivan Lobato (Ivanlh20@gmail.com), these models provide an all-in-one solution for researchers, engineers, and electron microscopy enthusiasts who want to leverage the latest neural network technology to improve the quality of their electron microscopy images.
I.Lobato1,2, T. Friedrich1,2, S. Van Aert1,2
1EMAT, University of Antwerp, Department of Physics, Groenenborgerlaan 171, B-2020 Antwerp, Belgium
2NANOlab Center of Excellence, University of Antwerp, Department of Physics, Groenenborgerlaan 171, B-2020 Antwerp, Belgium
Paper: https://www.nature.com/articles/s41524-023-01188-0
State-of-the-art electron microscopes such as scanning electron microscopes (SEM), scanning transmission electron microscopes (STEM) and transmission electron microscopes (TEM) have become increasingly sophisticated. However, the quality of experimental images is often hampered by stochastic and deterministic distortions arising from the instrument or its environment. These distortions can arise during any stage of the imaging process, including image acquisition, transmission, or visualization. In this paper, we will discuss the main sources of distortion in TEM and S(T)EM images, develop models to describe them and propose a method to correct these distortions using a convolutional neural network. We demonstrate the effectiveness of our approach on a variety of experimental images and show that it can significantly improve the signal-to-noise ratio resulting in an increase in the amount of quantitative structural information that can be extracted from the image. Overall, our findings provide a powerful framework for improving the quality of electron microscopy images and advancing the field of structural analysis and quantification in materials science and biology. The source code and trained models for our approach are made available in the accompanying repository.
One of the main advantages of our image restoration method is that the training data is generated using realistic physical models of the noise found in various microscopy modalities, as well as for an appropriate range of values for the noise model parameters. This methodology allows for the direct application of our network to experimental data, without requiring additional training for a particular specimen or microscope settings
Figure 1. Experimental image restoration for various microscopy modalities. The top row illustrates the raw experimental images, while the bottom row displays the restored versions
Below are the quick versions of the installation commands. For detailed instructions, please refer to the section below.
-
GPU support
pip install --upgrade pip pip install tensorflow[and-cuda]==2.14.1 pip install tk_r_em
-
CPU-only support
pip install tensorflow-cpu==2.14.1.* tk_r_em
- GPU support
conda install -c conda-forge cudatoolkit=11.2.* cudnn=8.1.* mkdir -p $CONDA_PREFIX/etc/conda/activate.d echo 'export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:$CONDA_PREFIX/lib/' > $CONDA_PREFIX/etc/conda/activate.d/env_vars.sh python -m pip install tensorflow==2.10.* tk_r_em
- CPU-only support
python -m pip install tensorflow-cpu==2.10.* tk_r_em
Docker provides an isolated environment, known as a container, where applications can run consistently across different platforms. Further details can be found in this guide.
-
Run with CUDA (GPU Support):
docker run -it --rm -v path_to_local_folder:/data --gpus all dreamleadsz/tk_r_em:cuda
-
Run with CPU Only:
docker run -it --rm -v path_to_local_folder:/data dreamleadsz/tk_r_em:cpu
Below are the step-by-step instructions for installing the package with and without GPU support.
To utilize tk_r_em, you'll need to install TensorFlow. If you plan to use GPU acceleration, the installation of CUDA libraries is also necessary. The required version of TensorFlow for tk_r_em varies depending on your operating system. We recommend installing TensorFlow within a virtual environment to prevent conflicts with other packages.
miniconda is the recommended approach for installing TensorFlow with GPU support. It creates a separate environment to avoid changing any installed software in your system. This is also the easiest way to install the required software especially for the GPU setup.
Let us start by creating a new conda environment and activate it with the following command:
conda create -n py310_gpu python=3.10.*
conda activate py310_gpu
If you plan to run TensorFlow on a GPU, you'll need to install the NVIDIA GPU driver. You can use the following command to verify it is installed.
nvidia-smi
TensorFlow requires a recent version of pip, so upgrade your pip installation to be sure you're running the latest version.
pip install --upgrade pip
For TensorFlow version 2.10.* on Windows, which was the last TensorFlow release to support GPU on native Windows, you can install the NVIDIA GPU driver and then install the following specific version of CUDA and cuDNN using Conda:
conda install -c conda-forge cudatoolkit=11.2.* cudnn=8.1.*
To ensure that the system paths recognize CUDA when your environment is activated, you can run the following commands:
mkdir -p $CONDA_PREFIX/etc/conda/activate.d
echo 'export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:$CONDA_PREFIX/lib/' > $CONDA_PREFIX/etc/conda/activate.d/env_vars.sh
These commands create a shell script in the activate.d directory, which sets the LD_LIBRARY_PATH environment variable when your environment is activated. This allows TensorFlow to locate the CUDA libraries that it needs to run on the GPU.
After installing the CUDA libraries, you can install TensorFlow. The required version of TensorFlow varies depending on your operating system.
On Linux, install TensorFlow version 2.14.1.* using pip:
-
GPU support
pip install tensorflow[and-cuda]==2.14.1
-
CPU-only support
pip install tensorflow-cpu==2.14.1.*
Note that running on CPU may be slower than running on a GPU, but it should still be functional.
On Windows, the last version of TensorFlow that supported GPU on native Windows was 2.10.*. Starting with TensorFlow 2.11, you'll need to install TensorFlow in WSL2 or install tensorflow-cpu instead.
-
GPU support
pip install tensorflow==2.10.*
-
CPU-only support
pip install tensorflow-cpu==2.10.*
With these installations, you should now have TensorFlow set up with GPU support (if applicable).
After installing TensorFlow, you can install tk_r_em using pip:
pip install tk_r_em
This command will install the latest version of tk_r_em and its required dependencies.
This option is ideal if you want to edit the code. Clone the repository:
$ git clone https://github.com/Ivanlh20/r_em.git
Then, change into its directory and install it using pip:
pip install -e .
You are now ready to run tk_r_em.
Electron microscopy techniques, such as Scanning Electron Microscopy (SEM), Scanning Transmission Electron Microscopy (STEM), and Transmission Electron Microscopy (TEM), exhibit unique sources of noise and feature variations across both low and high resolutions. Consequently, we have trained our network architecture on six diverse datasets, encompassing low-resolution (LR) and high-resolution (HR) images for each microscopy modality. Our results demonstrate that optimal performance is attained by training separate networks for LR and HR features, particularly at low doses, where the network can capitalize on the specific feature distribution acquired during the training phase.
You may now incorporate tk_r_em into your Python code.
-
The first example demonstrates the application of restoration on simulated data:
import os import matplotlib # Check if running on remote SSH and use appropriate backend for matplotlib remote_ssh = "SSH_CONNECTION" in os.environ matplotlib.use('Agg' if remote_ssh else 'TkAgg') import matplotlib.pyplot as plt def fcn_set_gpu_id(gpu_visible_devices: str = "0") -> None: os.environ["CUDA_DEVICE_ORDER"] = "PCI_BUS_ID" os.environ['CUDA_VISIBLE_DEVICES'] = gpu_visible_devices fcn_set_gpu_id("0") from tk_r_em import load_network, load_sim_test_data def fcn_inference(): """ Perform inference on test data using a pre-trained model and visualize the results. """ # select one of the available networks from [sfr_hrsem, sfr_lrsem, sfr_hrstem, sfr_lrstem, sfr_hrtem, sfr_lrtem] net_name = 'sfr_hrstem' # load its corresponding data x, y = load_sim_test_data(net_name) # load its corresponding model r_em_nn = load_network(net_name) r_em_nn.summary() n_data = x.shape[0] batch_size = 8 # run inference y_p = r_em_nn.predict(x, batch_size) fig, axs = plt.subplots(3, n_data, figsize=(48, 6)) for ik in range(n_data): x_ik = x[ik, :, :, 0].squeeze() y_p_ik = y_p[ik, :, :, 0].squeeze() y_ik = y[ik, :, :, 0].squeeze() ir = 0 axs[ir][ik].imshow(x_ik, cmap='viridis') axs[ir][ik].set_xticks([]) axs[ir][ik].set_yticks([]) axs[ir][ik].grid(False) if ik == 0: axs[ir][ik].set_ylabel(f"Detected {net_name} image", fontsize=14, ) ir = 1 axs[ir][ik].imshow(y_p_ik, cmap='viridis') axs[ir][ik].set_xticks([]) axs[ir][ik].set_yticks([]) axs[ir][ik].grid(False) if ik == 0: axs[ir][ik].set_ylabel(f"Restored {net_name} image", fontsize=14) axs[ir][ik].set_title(f'{ik}', fontsize=8) ir = 2 axs[ir][ik].imshow(y_ik, cmap='viridis') axs[ir][ik].set_xticks([]) axs[ir][ik].set_yticks([]) axs[ir][ik].grid(False) if ik == 0: axs[ir][ik].set_ylabel(f"Ground truth {net_name} image", fontsize=14) fig.subplots_adjust(hspace=2, wspace=10) fig.tight_layout() if remote_ssh: plt.savefig(f"restored_{net_name}.png", format='png') else: fig.show() print(ik) if __name__ == '__main__': fcn_inference()
The above function in the code loads pre-trained neural networks corresponding to various electron microscopy techniques, including High-Resolution Scanning Electron Microscopy (HRSEM), Low-Resolution Scanning Electron Microscopy (LRSEM), High-Resolution Scanning Transmission Electron Microscopy (HRSTEM), Low-Resolution Scanning Transmission Electron Microscopy (LRSTEM), High-Resolution Transmission Electron Microscopy (HRTEM), and Low-Resolution Transmission Electron Microscopy (LRTEM). The function then loads test data for each of the networks and runs inference on the data using the corresponding neural network.
Figures 2 and 3 display the output images produced by the code using the pre-trained networks for HRSTEM and LRSTEM, respectively.
Figure 2 displays simulated images that closely resemble those obtained through High-Resolution Scanning Transmission Electron Microscopy (HRSTEM). The first row depicts the input images, the second row exhibits the restored images, and the third row row the ground truth images.
Figure 3 illustrates simulated images that closely resemble those obtained through Low-Resolution Scanning Transmission Electron Microscopy (LRSTEM). The first row depicts the input images, the second row exhibits the restored images, and the third row displays the ground truth images.
-
The second example demonstrates the application of restoration on experimental HRSTEM data:
import os import matplotlib # Check if running on remote SSH and use appropriate backend for matplotlib remote_ssh = "SSH_CONNECTION" in os.environ matplotlib.use('Agg' if remote_ssh else 'TkAgg') import matplotlib.pyplot as plt def fcn_set_gpu_id(gpu_visible_devices: str = "0") -> None: os.environ["CUDA_DEVICE_ORDER"] = "PCI_BUS_ID" os.environ['CUDA_VISIBLE_DEVICES'] = gpu_visible_devices fcn_set_gpu_id("0") from tk_r_em import load_network, load_hrstem_exp_test_data def fcn_inference(): """ Perform inference on test data using a pre-trained model and visualize the results. """ # select one of the available networks from [sfr_hrsem, sfr_lrsem, sfr_hrstem, sfr_lrstem, sfr_hrtem, sfr_lrtem] net_name = 'sfr_hrstem' # load experimental hrstem data x = load_hrstem_exp_test_data('exp_hrstem') # load its corresponding model r_em_nn = load_network(net_name) r_em_nn.summary() n_data = x.shape[0] batch_size = 8 # run inference y_p = r_em_nn.predict(x, batch_size) fig, axs = plt.subplots(2, n_data, figsize=(48, 6)) for ik in range(n_data): x_ik = x[ik, :, :, 0].squeeze() y_p_ik = y_p[ik, :, :, 0].squeeze() ir = 0 axs[ir][ik].imshow(x_ik, cmap='hot') axs[ir][ik].set_xticks([]) axs[ir][ik].set_yticks([]) axs[ir][ik].grid(False) if ik == 0: axs[ir][ik].set_ylabel(f"Experimental {net_name} image", fontsize=14, ) ir = 1 axs[ir][ik].imshow(y_p_ik, cmap='hot') axs[ir][ik].set_xticks([]) axs[ir][ik].set_yticks([]) axs[ir][ik].grid(False) if ik == 0: axs[ir][ik].set_ylabel(f"Restored {net_name} image", fontsize=14) fig.subplots_adjust(hspace=2, wspace=10) fig.tight_layout() if remote_ssh: plt.savefig(f"restored_{net_name}.png", format='png') else: fig.show() print('Done') if __name__ == '__main__': fcn_inference()
The above function in the code loads pre-trained neural networks and the test data corresponding High-Resolution Scanning Transmission Electron Microscopy (HRSTEM). The output images produced by the code are show in Figure 8.
Figure 4 showcases experimental images obtained through HRSTEM. The first row displays the input images, while the second row shows the restored images. The first and second columns correspond to experimental images taken by Zezhong Zang, the third and fourth columns correspond to experimental images published in a Nature paper by Sandra Van Aert. The last column displays a low-dose experimental drift-corrected HRSTEM image of the SrTiO3 sample.
-
As a final example, we demonstrated how our network can restore a single image on a GPU with limited memory
In case you encounter a memory error while running the r_em_nn.predict(x) function on an image, you can try using r_em_nn.predict_patch_based(x) instead to restore the image. This function uses a window-based patch method, which involves dividing the image into smaller, overlapping patches and processing each patch separately.
The window size and stride determine the size of the patches and the amount of overlap between them. By using overlapping patches, the neural network can better capture the spatial features within each patch, resulting in improved image restoration.
After the neural network has processed each patch, the restored image is then reconstructed by combining the individual patches. This approach allows for processing large images that may not fit into the GPU's memory, making it a useful function for restoring high-resolution images on a memory-constrained GPU.
import os import matplotlib # Check if running on remote SSH and use appropriate backend for matplotlib remote_ssh = "SSH_CONNECTION" in os.environ matplotlib.use('Agg' if remote_ssh else 'TkAgg') import matplotlib.pyplot as plt def fcn_set_gpu_id(gpu_visible_devices: str = "0") -> None: os.environ["CUDA_DEVICE_ORDER"] = "PCI_BUS_ID" os.environ['CUDA_VISIBLE_DEVICES'] = gpu_visible_devices fcn_set_gpu_id("0") from tk_r_em import load_network, load_hrstem_exp_test_data def fcn_inference(): """ Perform inference on test data using a pre-trained model and visualize the results. """ # select one of the available networks from [sfr_hrsem, sfr_lrsem, sfr_hrstem, sfr_lrstem, sfr_hrtem, sfr_lrtem] net_name = 'sfr_hrstem' # load experimental hrstem data x = load_hrstem_exp_test_data('sgl_exp_hrstem') # load its corresponding model r_em_nn = load_network(net_name) r_em_nn.summary() # run inference y_p = r_em_nn.predict_patch_based(x, patch_size=256, stride=128, batch_size=16) fig, axs = plt.subplots(1, 2, figsize=(48, 6)) ir = 0 axs[ir].imshow(x, cmap='hot') axs[ir].set_xticks([]) axs[ir].set_yticks([]) axs[ir].grid(False) axs[ir].set_title(f"Experimental {net_name} image", fontsize=14, ) ir = 1 axs[ir].imshow(y_p, cmap='hot') axs[ir].set_xticks([]) axs[ir].set_yticks([]) axs[ir].grid(False) axs[ir].set_title(f"Restored {net_name} image", fontsize=14) fig.subplots_adjust(hspace=2, wspace=10) fig.tight_layout() if remote_ssh: plt.savefig(f"restored_{net_name}.png", format='png') else: fig.show() print('Done') if __name__ == '__main__': fcn_inference()
Figure 5 showcases an HRSTEM experimental image and its restored version obtained by using the window-patch-based prediction method. The left column displays the original experimental HRSTEM image taken by Zezhong Zang, while the right column shows the restored image.
All models of tk_r_em have been optimized to run efficiently on a standard desktop computer. However, for faster performance, you can utilize GPU acceleration.
Please cite tk_r_em in your publications if it helps your research:
@article{Lobato2024,
author = {I. Lobato and T. Friedrich and S. Van Aert},
doi = {10.1038/s41524-023-01188-0},
issn = {2057-3960},
issue = {1},
journal = {npj Computational Materials 2024 10:1},
keywords = {Imaging techniques,Transmission electron microscopy},
month = {1},
pages = {1-19},
publisher = {Nature Publishing Group},
title = {Deep convolutional neural networks to restore single-shot electron microscopy images},
volume = {10},
url = {https://www.nature.com/articles/s41524-023-01188-0},
year = {2024},
}