Simulation of elastic scattering in electron microscopy using the multislice algorithm. Simulations include various STEM and TEM techniques such as ADFSTEM, BFSTEM, ABFSTEM and CoM.
The code is written with readability and ease of understanding rather than speed. I've written the code such that it can be easily expanded upon. For example, to begin incorporating inelastic scattering one can calculate the transition potentials and reuse the multislice class. Further, addition elastic scattering techniques can be included by simply specifying the detector function.
This code has been tested with C++17 on Ubuntu, ensure that you place your project in a location which can access the header file of the technique you would like to use. You can clone the repo using the command
git clone https://github.com/Michael-MD/Electron-Microscopy-in-C.git
Next run
make
This will prepare executable a.out
.
The theory introduced here summarizes the key results of elastic scattering in electron microscopy. Further details can be found in Advanced computing in electron microscopy by Earl J. Kirkland which has become the go-to for an introduction to the field.
In electron microscopy, information about the strucutre and constituants of a specimen is inferred by observing the scattering and diffraction of electrons. Although the incident electrons have energies on the order of 10keV-100keV, the Schrödinger equation is employed with a relativistically corrected mass, wavelength and energy. Given a specimen with potential V(r), the Schrodinger equation reads
where
where
with formal operator solution given by
where
and is refered to as the projected potential.
Since
Now if
where
The techniques of electron microscopy can be catigoriezed into two primary modes. HR-TEM (High-Resolution Transmission Electron Microscopy) which uses incident place waves and STEM (Scanning Transmission Electron Mircsocopy) which uses a converged probe which scans across the specimenn surface.
In STEM, for each probe position, a diffraction pattern is obtained, the detector which measures the intensity pattern consitiues the various STEM techniques. The recorded signal at a given probe position is given by
where
To image a particular material, a .xyz file containing the crystal atomic species and locations for a single unit cell is to be placed in the structure directory.
This can then be loaded into the program by specifying the file name in the constructor of the technique of interest.
A few examples are provided in the repo. The file specifies general information about the unit cell at the beginning such as dimensions etc. and then specifies the atoms and locations on the subsequent lines. The template is as follows:
[name]
[unit cell length x] [unit cell length y] [unit cell length z] [basis angle alpha] [basis angle beta] [basis angle gamma]
[atomic number] [fractional x-coord] [fractional y-coord] [fractional z-coord] [occupancy] [mean-square displacement]
-1
The code currently doesn't make use of the basis angles, occupancy or msd.
The available techniques have the following headers:
ADFSTEM(float E, int px, int py, int px_p, int py_p, int tx, int ty, int tz, string filename, vector<float> &s,
float Cs, float deltaf, float alpha_max, float alpha_min_D, float alpha_max_D)
ABFSTEM(float E, int px, int py, int px_p, int py_p, int tx, int ty, int tz, string filename, vector<float> &s,
float Cs, float deltaf, float alpha_max, float alpha_min_D, float alpha_max_D)
BFSTEM(float E, int px, int py, int px_p, int py_p, int tx, int ty, int tz, string filename, vector<float> &s,
float Cs, float deltaf, float alpha_max, float alpha_max_D)
CSTEM(float E, int px, int py, int px_p, int py_p, int tx, int ty, int tz, string filename, vector<float> &s,
float Cs, float deltaf, float alpha_max)
HRTEM(float E, int px, int py, int tx, int ty, int tz, string filename, vector<float> &s,
float Cs, float deltaf, float alpha_max)
CoM(float E, int px_p, int py_p, int px, int py, int tx, int ty, int tz, string filename, vector<float> &s,
float Cs, float deltaf, float alpha_max, bool CoM_dir_x)
The inputs are as follows:
E :
Energy of input beam in eV
(px, py):
Dimensions of grid on which calculation is to occur in pixels i.e. the tranmission function, propagator etc. would be calculated on a grid of dimensions (px, py) pixels.
(px_p, py_p):
Resolution of STEM scan positions i.e. the number of scan positions in each direction.
(tx, ty, tz):
Tiling of unit cell.
filename:
File name of unit cell. Assumes files is located in directory strcutures/. Currently only .xyz files are supported.
s:
Vector of slice locations in fractions of unit cell thickness. For instance, if a unit cell is 3.905 Ang. then s = {.2,.8} will result in slices at .2
Cs:
Sperhical abberation of lens in [mm].
deltaf:
Lens defocus in Ang.
alpha_max:
Aperture opening angle in mrad.
alpha_min_D:
minimum detector angle in mrad.
alpha_max_D:
maximum detector angle in mrad.
New techniqes can be made by inheritying the multislice class and specifying the function which determines the detector. You may look at the wiki for details or simply look at one of the already existing techniqes as a template.
You may access any of the internal attributes such as the wave function or intensity and write these to a file by including "general_purpose_utils.h"
which includes a function writeToFile
which takes in the file name and 2D vector to be written to a file. For example,
#include "general_purpose_utils.h"
vector<float> s = {1};
CoM l(2e5, 64, 64, 128, 128, 1,1, 1, "fig511.xyz", s, 1.3, 500, 8.9, true);
writeToFile("I.txt", l.I);
All classes have an attribute layers
which stores the transmission function and propagators for the given layer. For example, continuing from the above example l.layers[0].t_re
is the real component of the transmission function of the first layer. The propagator can be accessed using P_re
or P_im
.
All classes have attribtues psi_re
and psi_im
while only STEM techniques have I
which stores the intensity.
Here are a few examples to get you up and running. The code to generate the file and the resulting image produced using a python script which I've inlcuded in the repo named visualize.py. STEM techniques will overwrite the wave function for each probe position so you will need to modify the code slightly to save the vector or otherwise. Secondly, HRTEM doesn't have access to a function which calcaultes the intensity but you may do so using the wave function.
#include "general_purpose_utils.h"
#include "HRTEM.h"
#include <vector>
using namespace std;
vector<float> s = {1};
HRTEM l(2e5, 512, 512, 8,8,1, "fig511.xyz", s, 1.3, 700, 10.37);
writeToFile("psi_re.txt", l.psi_re);
writeToFile("psi_im.txt", l.psi_im);
#include "general_purpose_utils.h"
#include "ADFSTEM.h"
#include <vector>
using namespace std;
vector<float> s = {1};
ADFSTEM l(2e5, 1024, 1024, 128, 128, 1,1, 1, "fig511.xyz", s, 1.3, 500, 8.9, 35, 200);
writeToFile("I.txt", l.I);
This is a linescan midway through the image obtained in python.
#include "general_purpose_utils.h"
#include "CoM.h"
#include <vector>
using namespace std;
vector<float> s = {1};
CoM l(2e5, 64, 64, 128, 128, 1,1, 1, "fig511.xyz", s, 1.3, 500, 8.9, true);
writeToFile("I.txt", l.I);