\(\newcommand{\AA}{\unicode{x212B}}\)
14.1. XAFS Functions: Overview and Naming Conventions¶
As with most Larch functions, each of the XAFS functions is designed to be able to act on arbitrary arrays of data to allow maximum flexibility in processing data. In addition, many of the Larch XAFS functions can write out several output values, including both scalars and arrays. While flexible, this could get rather cumbersome, and mean that you would generally have to keep track of a large set of related arrays.
14.1.1. Naming conventions for XAFS arrays¶
In Larch, it is most natural to put related data into a Group. This is often natural as data read in from a file is already held in a Group. If a set of XAFS data is held within a Group, then the Group members named energy and k and chi can be assumed to have the same standard meaning for all groups of XAFS Data. To make this most common usage easy, all of the XAFS functions follow a couple conventions for working more easily with Groups that can work on arbitrary arrays of data, but assume that they will write output values to a Group. In addition, the XAFS functions can usually just be given a Group that follows the expected XAFS naming convention. This is not rigidly enforced, and is not exclusive (that is, you can add data with other names), but following the expected naming conventions will make processing XAFS data much easier.
The naming convention define a set of expected names and meaning for data arrays and scalars within a Group. This is summarized in the Table of Conventional Names for an XAFS Group below.
Table of Conventional Names for an XAFS Group These are the standard names for arrays and scalars for various data associated with XAFS, including FEFF calculations. Given are the name, the physical quantity described, and the name of function that will generate this value.
name
type
meaning
created by
energy
array
\(E\) in eV
original data
mu
array
\(\mu\)
original data
e0
scalar
\(E_0\)
pre_edge()
,find_e0()
edge_step
scalar
\(\Delta \mu\)
dmude
array
\(d\mu/dE\)
norm
array
normalized \(\mu\)
flat
array
flattened \(\mu\)
pre_edge
array
pre-edge curve
post_edge
array
normalization curve
bkg
array
\(\mu_0(E)\)
chie
array
\(\chi(E)\)
k
array
\(k\)
chi
array
\(\chi(k)\)
kwin
array
\(\Omega(k)\)
r
array
\(R\)
chir
array
\(\chi(R)\) (complex)
chir_mag
array
\(|\chi(R)|\)
chir_re
array
\(\rm Re[\chi(R)]\)
chir_im
array
\(\rm Im[\chi(R)]\)
rwin
array
\(\Omega(R)\)
q
array
\(q\)
chiq
array
\(\chi(q)\) (complex)
chiq_mag
array
\(|\chi(q)|\)
chiq_re
array
\(\rm Re[\chi(q)]\)
chiq_im
array
\(\rm Im[\chi(q)]\)
where \(q\), \(\chi(q)\), and so on indicates back-transformed \(k\).
The XAFS functions encourage following this convention, in that they are consistent in wanting
\(\chi(k)\) to be represented by the two arrays GROUP.k
and GROUP.chi
14.1.1.1. group argument and _sys.xafsGroup
¶
The XAFS functions need to write outputs to some group – there are simply too many outputs to return and expect you to manage. To better accomodate this, all functions take a group argument, which is used as the group into which results are written. This gives a convenient way to manage the results of the different analysis steps, but gets tedious to provide this argument repeatedly when working with a particular data set.
For XAFS analysis, there is also a special group, _sys.xafsGroup
that
is used as the default group to write outputs to if no group argument is
supplied. In addition, when an explicit group argument is given,
_sys.xafsGroup
is set to this group. In short, the _sys.xafsGroup
will be used as the “current, default group”. This means that when working
with a set of XAFS data all contained within a single group (which is
expected to be the normal case), the group argument does not need to be
typed repeatedly.
Because this uses a global group in the Larch interpreter, this convention works from with the Larch language, but does not work from plain Python unless an instance of a Larch session is passed into the larch.xafs function using the _larch argument.
14.1.1.2. First Argument Group convention¶
Since the XAFS functions need to write outputs to some group and will
generally work with groups that contain data following Table of
Conventional Names for an XAFS Group, most of the XAFS
functions follow what is called the First Argument Group convention.
This convention gives a simple approach when working with groups of XAFS
data and it is worth understanding and using this for most work with the
XAFS work. This convention is built on the _sys.xafsGroup
convention
discussed above but is even easier to use.
While the XAFS functions can take arrays of data as the first two arguments
most work will have these arrays in a single group with array names that
follows the conventions above. As an example, the most general use of the
autobk()
function takes an array of energy as the first argument, an
array of mu as the second argument, and supplying an output group for
placing all the arrays and data calculated within the function. That is,
the most general use would look like:
autobk(energy, mu, group=dat, rbkg=1, ....)
Of course, most usage will actually want to use energy and mu arrays from the same group, and use that group as the output group, so that all data stays contained within the same group. That would make the call above look like:
autobk(dat.energy, dat.mu, group=dat, rbkg=1, ....)
where the group name dat is repeated three times.
The First Argument Group convention allows this to be written as:
autobk(dat, rbkg=1, ....)
That is, as long as the Group dat follows the XAFS naming conventions
(for autobk()
that it has an energy array named energy and
absorbance array named mu) the two forms above are equivalent. All the
XAFS functions follow this convention and use a consistent set of attribute
names (see Table of Conventional Names for an XAFS Group). This convention nearly makes the Larch XAFS routines
into object-oriented, or in this case Group oriented, set of functions
that interact in a coherent and predictable way on an XAFS dataset.
14.1.2. Plotting Macros for XAFS¶
XAFS analysis often uses several different standard views of the data arrays for \(\mu(E)\), \(\chi(k)\), and \(\chi(R)\). Larch’s plotting capabilities provide wide flexibility in how plots can be done. While that flexibility can be useful in general, within the narrow scope of plotting XAFS data, being able to easily create consistent plots with reasonable defaults produces results that are easier to digest and understand.
The macros described here attempt to provide that functionality of easy-to-use standard plotting macros. In particular, they automatically handle typesetting the labels for the plot axes in a consistent manner, and assign consistent labels to the different curves shown. The results are also easily extended, so that you can add curves, annotations, etc. Many of the examples in the following sections in this chapter make use of these macros.
14.1.2.1. plot_mu()
¶
- plot_mu(dgroup, norm=False, deriv=False, show_pre=False, show_post=False, show_e0=False, emin=None, emax=None, label=None, new=True, win=1)¶
Plot \(\mu(E)\) for an XAFS data group in various forms
- Parameters:
dgroup – group of XAFS data after
pre_edge()
results (see note below)norm – bool whether to show normalized data [
False
]deriv – bool whether to show derivative of XAFS data [
False
]show_pre – bool whether to show pre-edge curve [
False
]show_post – bool whether to show post-edge curve [
False
]show_e0 – bool whether to show E0 [
False
]show_deriv – bool whether to show deriv together with mu [
False
]emin – min energy to show, relative to E0 [
None
, start of data]emax – max energy to show, relative to E0 [
None
, end of data]label – string for label [
None
: ‘mu’, ‘dmu/dE’, or ‘mu norm’]new – bool whether to start a new plot [
True
]win – integer plot window to use [1]
The input data group must have the following attributes: energy, mu, norm, e0, pre_edge, edge_step, filename
14.1.2.2. plot_bkg()
¶
- plot_bkg(dgroup, norm=True, emin=None, emax=None, show_e0=False, label=None, new=True, win=1)¶
Plot \(\mu(E)\) and background \(\mu_0(E)\) for XAFS data group
- Parameters:
dgroup – group of XAFS data after autobk() results (see note below)
norm – bool whether to show normalized data [
True
]emin – min energy to show, relative to \(E_0\) [
None
, start of data]emax – max energy to show, relative to \(E_0\) [
None
, end of data]show_e0 – bool whether to show E0 [
False
]label – string for label [
None
: ‘mu’]new – bool whether to start a new plot [
True
]win – integer plot window to use [1]
The input data group must have the following attributes: energy, mu, bkg, norm, e0, pre_edge, edge_step, filename
14.1.2.3. plot_chik()
¶
- plot_chik(dgroup, kweight=None, kmax=None, show_window=True, label=None, new=True, win=1)¶
Plot k-weighted \(\chi(k)\) for XAFS data group
- Parameters:
dgroup – group of XAFS data after autobk() results (see note below)
kweight – k-weighting for plot [read from last
xftf()
, or 0]kmax – max k to show [
None
, end of data]show_window – bool whether to also plot k-window [
True
]label – string for label [
None
: ‘chi’]new – bool whether to start a new plot [
True
]win – integer plot window to use [1]
The input data group must have the following attributes: k, chi, kwin, filename.
14.1.2.4. plot_chir()
¶
- plot_chir(dgroup, show_mag=True, show_real=False, show_imag=False, rmax=None, label=True, new=True, win=1)¶
Plot \(\chi(R)\) for XAFS data group
- Parameters:
dgroup – group of XAFS data after xftf() results (see note below)
show_mag – bool whether to plot \(|\chi(R)|\) [
True
]show_real – bool whether to plot \(Re[\chi(R)]\) [
False
]show_imag – bool whether to plot \(Im[\chi(R)]\) [
False
]rmax – max R to show [
None
, end of data]label – string for label [
None
: ‘chir’]new – bool whether to start a new plot [
True
]win – integer plot window to use [1]
The input data group must have the following attributes: r, chir_mag, chir_im, chir_re, kweight, filename
14.1.2.5. plot_chifit()
¶
- plot_chifit(dataset, kmin=0, kmax=None, rmax=None, show_mag=True, show_real=False, show_imag=False, new=True, win=1)¶
Plot k-weighted \(\chi(k)\) and \(\chi(R)\) for fit to feffit dataset
- Parameters:
dataset – feffit dataset, after running
feffit()
.kmin – min k to show [0]
kmax – max k to show [
None
, end of data]rmax – max R to show [
None
, end of data]show_mag – bool whether to plot \(|chi(R)|\) [
True
]show_real – bool whether to plot \(Re[`chi(R)]\) [
False
]show_imag – bool whether to plot \(Im[\chi(R)]\) [
False
]new – bool whether to start a new plot [
True
]win – integer plot window to use [1]
14.1.2.6. plot_path_k()
¶
- plot_path_k(dataset, ipath, kmin=0, kmax=None, offset=0, label=None, new=False, win=1, **kws)¶
Plot k-weighted \(\chi(k)\) for a single Path of a feffit dataset
- Parameters:
dataset – feffit dataset, after running
feffit()
ipath – index of path, starting count at 0 [0]
kmin – min k to show [0]
kmax – max k to show [
None
, end of data]offset – vertical offset to use for plot [0]
label – path label [‘path I’]
new – bool whether to start a new plot [
True
]win – integer plot window to use [1]
kws – additional keyword arguments are passed to plot()
14.1.2.7. plot_path_r()
¶
- plot_path_r(dataset, ipath, rmax=None, offset=0, label=None, show_mag=True, show_real=False, show_imag=True, new=False, win=1, **kws)¶
Plot \(\chi(R)\) for a single Path of a feffit dataset
- Parameters:
dataset – feffit dataset, after running
feffit()
ipath – index of path, starting count at 0 [0]
kmax – max k to show [None, end of data]
offset – vertical offset to use for plot [0]
label – path label [‘path I’]
show_mag – bool whether to plot \(|\chi(R)|\) [
True
]show_real – bool whether to plot \(Re[\chi(R)]\) [
False
]show_imag – bool whether to plot \(Im[\chi(R)]\) [
False
]new – bool whether to start a new plot [
True
]win – integer plot window to use [1]
kws – additional keyword arguments are passed to plot()
14.1.2.8. plot_paths_k()
¶
- plot_paths_k(dataset, offset=-1, kmin=0, kmax=None, new=True, win=1, **kws):
Plot k-weighted chi(k) for model and all paths of a feffit dataset
- Parameters:
dataset – feffit dataset, after running
feffit()
kmin – min k to show [0]
kmax – max k to show [
None
, end of data]offset – vertical offset to use for paths for plot [-1]
new – bool whether to start a new plot [
True
]win – integer plot window to use [1]
kws – additional keyword arguments are passed to plot()
14.1.2.9. plot_paths_r()
¶
- plot_paths_r(dataset, offset=-0.5, rmax=None, show_mag=True, show_real=False, show_imag=False, new=True, win=1, **kws):
Plot \(\chi(R)\) for model and all paths of a feffit dataset
- Parameters:
dataset – feffit dataset, after running func:feffit
offset – vertical offset to use for paths for plot [-0.5]
rmax – max R to show [
None
, end of data]show_mag – bool whether to plot \(|\chi(R)|\) [T``rue``]
show_real – bool whether to plot \(Re[\chi(R)]\) [
False
]show_imag – bool whether to plot \(Im[\chi(R)]\) [
False
]new – bool whether to start a new plot [
True
]win – integer plot window to use [1]
kws – additional keyword arguments are passed to plot()
14.1.2.10. plot_prepeaks_baseline()
¶
- plot_prepeaks_baseline(dgroup, subtract_baseline=False, show_fitrange=True, show_peakrange=True, win=1, **kws):
Plot pre-edge peaks and baseline fit, as from
pre_edge_baseline()
or Larix.- Parameters:
dgroup – data group, after running
pre_edge_baseline()
subtract_baseline – bool whether to subtract baseline for plot
show_fitrange – bool whether to show fit range as vertical bars
show_peakrange – bool whether to show pre-edge peak range with markers
win – integer plot window to use [1]
kws – additional keyword arguments are passed to plot()
The dgroup group must have a prepeaks subgroup.
14.1.2.11. plot_prepeaks_fit()
¶
- plot_prepeaks_fit(dgroup, show_init=False, subtract_baseline=False, show_residual=False, win=1, **kws):
Plot pre-edge peaks and fit, as in the Larix GUI
- Parameters:
dgroup – data group, after running pre-edge peak fit.
show_init – bool whether to show initial model, before fitting
subtract_baseline – bool whether to subtract baseline for plot
show_residual – bool whether to show residual as a stacked plot.
win – integer plot window to use [1]
kws – additional keyword arguments are passed to plot()
The dgroup group must have a peakfit_history subgroup. Currently, this is automatically generated only using the Larix GUI or scripts written (and possibly altered) by it.
14.1.3. Utility Functions for XAFS¶
Listed here are some general purpose functions for XAFS.
14.1.3.1. ktoe()
and etok()
¶
- etok(energies)¶
Convert photo-electron energy in eV to wavenumber in \(\AA^{-1}\). energies can be a single number or array of numbers.
- ktoe(wavenumbers)¶
Convert photo-electron wavenumber in \(\AA^{-1}\) to energy in eV. wavenumber can be a single number or array of numbers.
An example use would be to print out a table of energies and \(k\) values:
larch> kvals = linspace(0, 25, 26)
larch> evals = ktoe(kvals)
larch> for k,e in zip(kvals, evals)):
larch> print " %5.1f 1/Ang -> %8.2f eV" %(k , e)
larch> endfor
0.0 1/Ang -> 0.00 eV
1.0 1/Ang -> 3.81 eV
2.0 1/Ang -> 15.24 eV
3.0 1/Ang -> 34.29 eV
4.0 1/Ang -> 60.96 eV
5.0 1/Ang -> 95.25 eV
6.0 1/Ang -> 137.16 eV
7.0 1/Ang -> 186.69 eV
8.0 1/Ang -> 243.84 eV
9.0 1/Ang -> 308.61 eV
10.0 1/Ang -> 381.00 eV
11.0 1/Ang -> 461.01 eV
12.0 1/Ang -> 548.64 eV
13.0 1/Ang -> 643.89 eV
14.0 1/Ang -> 746.76 eV
15.0 1/Ang -> 857.25 eV
16.0 1/Ang -> 975.36 eV
17.0 1/Ang -> 1101.08 eV
18.0 1/Ang -> 1234.43 eV
19.0 1/Ang -> 1375.40 eV
20.0 1/Ang -> 1523.99 eV
21.0 1/Ang -> 1680.20 eV
22.0 1/Ang -> 1844.03 eV
23.0 1/Ang -> 2015.48 eV
24.0 1/Ang -> 2194.55 eV
25.0 1/Ang -> 2381.24 eV
14.1.3.2. estimate_noise()
¶
- estimate_noise(k, chi=None, group=None, rmin=15, rmax=30, ....)¶
Automatically estimate the noise level in a \(\chi(k)\) spectrum.
- Parameters:
k – 1-d array of \(k\)
chi – 1-d array of \(\chi\)
group – output group.
rmin – minimum \(R\) value for noise estimate.
rmax – maximum \(R\) value for noise estimate.
kweight – exponent for weighting spectra by k**kweight [1]
kmin – starting k for FT Window [0]
kmax – ending k for FT Window [20]
dk – tapering parameter for FT Window [4]
dk2 – second tapering parameter for FT Window [None]
window – name of window type [‘kaiser’]
nfft – value to use for N_fft [2048].
kstep – value to use for delta_k ( Ang^-1) [0.05]
The method uses an XAFS Fourier transform, and many of arguments (kmin, kmax, etc) are identical to those of
xftf()
.This function follows the First Argument Group convention with arrays named k and chi. The following outputs are written to the supplied group (or _sys.xafsGroup if group is not supplied):
attribute
meaning
epsilon_k
estimated noise level in \(\chi(k)\).
epsilon_r
estimated noise level in \(\chi(R)\).
kmax_suggest
suggested highest \(k\) value for which \(|\chi(k)| > \epsilon_k\)
This method uses the high-R portion of \(\chi(R)\) (between rmin and rmax) as a measure of the noise level in the \(\chi(R)\) data and uses Parseval’s theorem to convert this noise level to that in \(\chi(k)\). This method implicitly assumes that there is no signal in the high-R portion of the spectrum, and that the noise in the spectrum is “white” (independent of \(R\)) . Each of these assumptions can be legitimately questioned. Then again, making the assertion that these assumptions are invalid and disregarding the estimated noise level here would require knowledge of the noise in an XAFS spectrum that most users do not have. At the very least, this estimate should be be interpreted as a minimal estimate of the noise level in \(\chi(k)\).
The estimate for the output value kmax_suggest has a tendency to be pessimistic in how far out the \(\chi(k)\) data goes before being dominated by noise, but has the advantage of being an impartial measure of data quality. It is particularly pessimistic for extremely good data. Then again, considering that the estimate for \(\epsilon\) is probably too small, the estimate may not be that bad.