Step 1: Multi-exponential regression at global average CW-OSL curve"

# Changelog:
# * 2020-May  , DM: First reasonable version
#   ... 
# * 2022-05-04, DM: Revised handling in case kableExtra is not installed
# * 2022-05-04, DM: Changed style to have better match between text font and formulas
#
# ToDo:
# * Get No. of record_type info and the global_average data directly from #
#   sum_OSLcurves output to save computing time
# * Create some kind of overview diagram to set residual signals into better relation (see idea in code)
#
# NOTE: Add the following lines to the YAML header to enable offline display of the equations:
#   mathjax: local
#   self_contained: false

kable_extra <- require(kableExtra, quietly = TRUE)
library(knitr)
library(ggplot2)
library(gridExtra)
library(scales)

plot_theme <- theme_classic()
image_file <- NULL

knitr::opts_chunk$set(fig.width=8,
                      fig.asp=.6,
                      results = "asis",
                      warning=FALSE,
                      message=FALSE,
                      error=FALSE,
                      echo=FALSE,
                      cache=FALSE)

record_type <- fit_data$parameters$record_type
n.curves <- 0
for (j in 1:length(data_set)) {
  n.curves <- n.curves + sum(names(data_set[[j]]) == record_type)}


This report was automatically created at r Sys.time() by the function RLum.OSL_global_fitting() of the R package OSLdecomposition (r packageVersion("OSLdecomposition")). Visit luminescence.de to learn more.


Results

Step 1 of a component-resolved dose calculation is the identification of the components occurring in the data set. Constant signal component decay rates $\lambda$ over all measurements were assumed and one global average CW-OSL curve was calculated. A series of multi-exponential decay model fittings with an increasing number of components $K$ was performed. The resulting CW-OSL models can be compared by their photoionisation cross sections $\sigma$ and their fitting quality to select the optimum set of signal components.

OSL plots {.tabset .tabset-pills}

Global average

if (!is.null(image_format)) image_file <- paste0(image_path, "step1_fig1_global_average.", image_format)

sum_OSLcurves(data_set,
              record_type = record_type,
              output.plot = TRUE,
              theme.set = plot_theme,
              plot.first = FALSE,
              title = NULL,
              verbose = FALSE,
              filename = image_file)

cat(paste0("<br><i><b>Figure 1.1:</b> <u>Red line</u>: Global average curve built from the arithmetic mean of the individual curves. <br><u>Grey points</u>: Data points of all measurements.</i><br><br>"))
# Define dynamic Rmarkdown code
dynamic_code <- c(
    "### K = {{i}}\n",
    "```r}}\n",
    "if (!is.null(image_format)) image_file <- paste0(image_path, \"step1_fig1_K={{i}}_fitting.\", image_format)\n",
    "plot_OSLcurve(fit_data$curve, 
                fit_data$component.tables[[{{i}}]], 
                title = NULL,
                show.legend = TRUE,
                show.crosssec = TRUE,
                show.initial = FALSE,
                theme.set = plot_theme,
                filename = image_file)\n",
    "cat(paste0(\"<br><i><b>Figure 1.{{i+1}}:</b> Fit model with K = {{i}} signal components<br>\", \"<u>Upper left</u>: CW-OSL plot of global average curve and fitting model.<br>\", \"<u>Upper right</u>: pseudoLM-OSL transformed plot<br>\",\"<u>Lower left</u>: Residual curve between fitting and global average curve<br>\",\"<u>Lower right</u>: Estimated type of quartz OSL component (colored), photoionisation cross sections $\\\\sigma$, decay rates $\\\\lambda$ and signal intensity $n$.</i><br><br>\"))",
    "```\n"
  )

tabs <- lapply(as.list(1:nrow(fit_data$F.test)),
                function(i) knitr::knit_expand(text = dynamic_code))


# Now knit the dynamic code. This has to be in a separate chunk

r knitr::knit(text = unlist(tabs))

Photoionisation cross sections

Under the assumption that the provided optical stimulation parameters ($\lambda_{stim}=$ r fit_data$parameters$stimulation.wavelength $\mathrm{nm}, \Phi_{stim}=$ r fit_data$parameters$stimulation.intensity $\mathrm{\frac{mW}{cm^2}}$) are correct, the found decay rates were translated into photoionisation cross section values and set into relation with quartz OSL literature values:

if (!is.null(image_format)) image_file <- paste0(image_path, "step1_fig2_cross-sections.", image_format)

plot_PhotoCrosssections(fit_data,
                        stimulation.intensity = fit_data$parameters$stimulation.intensity,
                        stimulation.wavelength = fit_data$parameters$stimulation.wavelength,
                        filename = image_file)

  cat("<br><i><b>Figure 2:</b> Photoionisation cross section comparison assuming a stimulation light intensity of", fit_data$parameters$stimulation.intensity, "mW/cm² and a stimulation light wavelength of", fit_data$parameters$stimulation.wavelength,"nm.</i><br><br>")

Systematic shifts in the photoionisation cross sections are likely when compared to literature values. These are caused by false estimates of the effective stimulation intensity. To manually correct for these shifts, apply the following rules when setting RLum.OSL_global_fitting(stimulation_intensity = ?):

  • Increase stimulation intensity to shift the cross sections towards lower absolute values
  • Decrease stimulation intensity to shift the cross sections towards higher absolute values

Fitting quality

Selecting a fitting model with the physically correct number of components $K$ is not trivial. According to Bailey et al. (1997), Mittelstraß (2019) and others, most quartz samples are sufficiently described by three CW-OSL components. Thus, the $K = 3$ fitting is chosen as default model. However, a fourth components might contribute a significant signal. Likewise, a measurement may contain just two components. An indication provides an $F$-test as suggested by Bluszcz & Adamiec (2006). The test criterion $F_K$ measures the improvement in the fitting quality by an additional component, see section F-test for details. An $F_K$ below the preset value of F~threshold~ = r fit_data$parameters$F.threshold indicates that the last added component did not improve the model significantly.

F_table <- fit_data$F.test.print
row_index <- c(1:nrow(F_table))
F_table <- cbind(data.frame(K = row_index), F_table)

# Extra 
# if (fit_data$parameters$background.fitting == TRUE) {
# colnames(F_table) <- c("  $K$  ", paste0("$\\lambda_", row_index,"$ $(s^{-1})$"), "background", "RSS", "$F_K$")

# Default model with K = 3
K.default <- 3
K.default.text <- ""
if (kable_extra) {
  K.default.text <- "<u>Blue stripe</u>: Algorithm selected model.<br>"
  if (fit_data$K.selected <= K.default) {
    K.default <- fit_data$K.selected
  } else {
    K.default.text <- paste0("<u>Green stripe</u>: Default CW-OSL model for quartz (recommended). ", K.default.text)
  }
}

# Table caption:
cat(paste0("<br><i><b>Table 2:</b> Decay rates and fitting quality parameters as function of component number $K$. <br>$RSS$: Residual square sum. $F_K$:  Measure of fitting improvement.</i><br>", K.default.text))

# print F-test table
colnames(F_table) <- c("  $K$  ", paste0("$\\lambda_", row_index,"$ $(s^{-1})$"), "$RSS$","$F_K$")
if (kable_extra) {
  kable(F_table, escape = TRUE, align = "c") %>% 
  kable_styling(bootstrap_options = c("condensed")) %>%  
  row_spec(K.default, background = "lightgreen") %>%
  row_spec(fit_data$K.selected, background = "lightblue")
}else{
  kable(F_table, escape = TRUE, align = "c") 
}
### Signal bin overdispersion idea

# List of model selection helps:
# 1. Comparison of the photoionisation cross section with literature values
# 2. Evaluation of the fitting quality by an F-test and the *reduced Chi²*
# 3. (ToDo:) Evaluation of the decomposition quality by signal bin overdispersion

# The idea is the following: Incorrect decay rates lead to lopsided residual curves over the signal bin intervals in the decomposition process. These lopsided residuals can be detected by taking the signal bin variance in relation to the poisson distribution variance. The result is a kind of "overdispersion" value for each signal bin in each OSL record. The K = ? model where the median of these overdispersions is minimal, should be the best suiting model for the majority of CW-OSL curves



Methods

<i><u>Further content planned for this chapter:</u>

* picture with shematic overview
* deeper explanation of multi-exponential fitting
* Some sentences about the limitations and the behaviour of the F-test
* Explanation why global background-fitting is counter-productive

</i><br>

Basics {#basics}

The data analysis approach assumes that all CW-OSL signal curves in a data set can be sufficently described by a sum of exponential decays with first order kinetics:

$$I(t) = \sum_{i=1}^K n_ie^{-\lambda_it}$$

Here, $I(t)$ is the CW-OSL signal, $K$ is the number of signal components. Each component is defined by its intensity $n$ and its decay rate $\lambda$.

To achieve component-separated dose information, the data analysis process is divided into multiple steps:

Step 1 workflow {#Overview}

This report covers the Step 1 analysis, which can be outlined as following:

Detailed explanations on step 1.1 and step 1.2 can be found in Mittelstraß (2019). Practical advice can be found in the R documentations ?sum_OSLcurves and ?fit_OSLcurve.

Step 1.2 is based on the ideas of Bluszcz & Adamiec (2006). The decay parameter are evaluated through differential evolution (see Price et al. 2006), performed by the function DEoptim from the package DEoptim (Ardia et al. 2020). The function DEoptim deploys the function decompose_OSLcurve (see Step2) for the calculation of the signal intensities. An Levenberg-Marquardt fitting refines the resulting parameter and is performed by the function nlsLM from the package minpack.lm (Elzhov et al. 2016).

Photoionisation cross sections {#M3}

The signal decay rate $\lambda_{k}$ of each component is translated into the photoionisation cross section $\sigma_{k}$ of the associated defect state by:

$$\sigma_{k}=\lambda_{k} {hc \over \Phi_{stim}\lambda_{stim}}$$ Here, $h$ is the Planck constant, $c$ is the speed of light, $\Phi_{stim}$ is the stimulation light intensity and $\lambda_{stim}$ is the stimulation light wavelength. If this wavelength is about ~ 470 nm and the measured sample material is quartz, then the resulting photoionisation cross sections can be compared with literature values, see figure 2. For the automatic assignement of component names, an approximated 2-$\sigma$ bandwidth of the average literature value was used, see ?fit_OSLcurve for the exact definition.

Important note: The stimulation light intensity, displayed by OSL/TL reader control software, might differ from the effective stimulation light intensity. Degrading of stimulation light sources and filters as well as a dirty chamber window can significantly decrease the light output. In addition, the light flux in the bulk material depends on the reflectance and transmittance of the sample, its fixation and the sample carrier. Systematic shifts in the photoionisation cross section values are therefore likely.

F-test {#M4}

Bluszcz & Adamiec (2006) proposed the use of an F-test to compare CW-OSL models by their fitting quality. The test criterion $F_K$ measures the improvement caused by an additional component. It compares the residual square sum (RSS) of the $K-1$ model with the new $K$ model:

$$F_K = \frac{(RSS_{K-1} - RSS_K)/2}{RSS_K(N - 2K)} $$

Here, $N$ is the number data points (channels) of the global average curve and $K$ is the number of OSL components in the fitting model. If $F_K$ falls below a preset threshold value (here, $F_{threshold}$ = r fit_data$parameters$F.threshold), the new fitting model with one additional component is apparently not significantly better than the K - 1 model. Mittelstraß (2019) showed that reasonable values for $F_{threshold}$ are in the range of: $50 < F_{threshold} < 300$. Parametrized for the current analysis is $F_{threshold}$ = r fit_data$parameters$F.threshold.

However, CW-OSL components with non-first-order kinetics, variations in the decay rates and too long measurement duration can lead to the detection of imaginary signal components, as Mittelstraß (2019) discussed in more detail. Therefore, the F-test outcome should be taken with care as over-fitting is still likely.



General remarks {#disclaimer}

This report was automatically generated by functions of the R package OSLdecomposition written and maintained by Dirk Mittelstraß (dirk.mittelstrass@luminescence.de). This package is preferably used together with the R package Luminescence (link) by Kreutzer et al. (2012). For the dynamic creation of this HTML report, the R packages knitr and rmarkdown are used, see Xie (2015) and Xie et al. (2018). All diagrams are drawn with ggplot2 (Wickham 2016).

The data analysis method covered with this report was developed for CW-OSL SAR protocol measured quartz samples, see Murray and Wintle (2000). It might be also useful for the analysis of Al2O3 or feldspar samples measured with SAR-like protocols. You can use, share and publish this report and the containing results at will. But we demand to refer to the R package OSLdecomposition including its version number (r packageVersion("OSLdecomposition")) if you publish the results in a peer-reviewed journal. Please include the following reference to your publication:


Mittelstraß, D., Schmidt, C., Beyer, J., Heitmann, J. and Straessner, A.: R package OSLdecomposition: Automated identification and separation of quartz CW-OSL signal components, in preparation.


We allow and encourage you to add the HTML file of this report to the electronical supplement of your publication.



References {#ref}

Ardia, D., Mullen, K.M., Peterson, B.G., Ulrich, J., 2020. DEoptim: Differential Evolution in R.
https://CRAN.R-project.org/package=DEoptim

Bailey, R. M., Smith, B. W. and Rhodes, E. J., 1997. Partial bleaching and the decay form characteristics of quartz OSL, Radiation Measurements, 27(2), 123–136.
https://doi.org/10.1016/S1350-4487(96)00157-6

Bluszcz, A., Adamiec, G., 2006. Application of differential evolution to fitting OSL decay curves. Radiation Measurements 41, 886–891.
https://doi.org/10.1016/j.radmeas.2006.05.016

Durcan, J.A., Duller, G.A.T., 2011. The fast ratio: A rapid measure for testing the dominance of the fast component in the initial OSL signal from quartz. Radiation Measurements 46, 1065–1072.
https://doi.org/10.1016/j.radmeas.2011.07.016

Elzhov, T.V., Mullen, K.M., Spiess, A.-N., Bolker, B., 2016. minpack.lm: R Interface to the Levenberg-Marquardt Nonlinear Least-Squares Algorithm Found in MINPACK, Plus Support for Bounds.
https://CRAN.R-project.org/package=minpack.lm

Jain, M., Murray, A.S., Bøtter-Jensen, L., 2003. Characterisation of blue-light stimulated luminescence components in different quartz samples: implications for dose measurement. Radiation Measurements 37, 441–449.
https://doi.org/10.1016/S1350-4487(03)00052-0

Kreutzer, S., Schmidt, C., Fuchs, M.C., Dietze, M., Fuchs, M., 2012. Introducing an R package for luminescence dating analysis. Ancient TL 30.
http://ancienttl.org/ATL_30-1_2012/ATL_30-1_Kreutzer_p1-8.pdf

Mittelstraß, D., 2019. Decomposition of weak optically stimulated luminescence signals and its application in retrospective dosimetry at quartz, Master thesis, TU Dresden, Dresden.
https://iktp.tu-dresden.de/IKTP/pub/19/Dirk_Mittelstrass_Master.pdf

Mittelstraß D., Schmidt C., Beyer J., and Straessner A., 2019. Automated identification and separation of quartz CW-OSL signal components with R. talk presented at the Central European Conference on Luminescence and Trapped-Charge dating: DLED 2019, Bingen, Germany
http://luminescence.de/OSLdecomp_talk.pdf

Murray, A. S. and Wintle, A. G., 2000. Luminescence dating of quartz using an improved single-aliquot regenerative-dose protocol. Radiation Measurements 32
https://doi.org/10.1016/S1350-4487(99)00253-X

Price, K.V., Storn, R.M., Lampinen, J.A., 2006. Differential Evolution - A Practical Approach to Global Optimization, Natural Computing. Springer-Verlag.

Singarayer, J.S., Bailey, R.M., 2003. Further investigations of the quartz optically stimulated luminescence components using linear modulation. Radiation Measurements, Proceedings of the 10th international Conference on Luminescence and Electron-Spin Resonance Dating (LED 2002) 37, 451–458.
https://doi.org/10.1016/S1350-4487(03)00062-3

Wickham, H., 2016. ggplot2: elegant graphics for data analysis, Second edition. ed, Use R! Springer, Cham.
https://ggplot2.tidyverse.org/

Xie, Y., 2015. Dynamic documents with R and Knitr, Second edition. ed. CRC Press/Taylor & Francis, Boca Raton.
https://yihui.org/knitr/

Xie, Y., Allaire, J.J., Grolemund, G., 2018. R Markdown: the definitive guide. Taylor & Francis, CRC Press, Boca Raton.
https://bookdown.org/yihui/rmarkdown/



Script parameters

# Algorithm settings
para_table <- data.frame(n = "Analyzed data set", t = object_name)
para_table <- rbind(para_table, 
                    data.frame(n = "Analyzed record type", t = record_type),
                    data.frame(n = "Maximum allowed components ", t = paste0("*K*~max~ = ", fit_data$parameters$K.max)),
                    data.frame(n = "Threshold *F*-value", t = paste0("*F*~threshold~ = ", fit_data$parameters$F.threshold)))

# Photo-ionisation
para_table <- rbind(para_table, 
                    data.frame(n = "Stimulation wavelength", t = paste0(fit_data$parameters$stimulation.wavelength," nm")),
                    data.frame(n = "Stimulation intensity", t = paste0(fit_data$parameters$stimulation.intensity," mW/cm²")))

#  Global curve
ch.N <- length(fit_data$curve$time)
ch.W <- fit_data$curve$time[2]-fit_data$curve$time[1]
para_table <- rbind(para_table, 
                    data.frame(n = "No. of input measurements", t = format(n.curves)),
                    data.frame(n = "No. of input aliquots", t = format(length(data_set))),
                    data.frame(n = "Channel time ", t = paste0("$\\Delta$*t* = ", prettyNum(ch.W), " s")),
                    data.frame(n = "Channels ", t = paste0("*N* = ", ch.N)),
                    data.frame(n = "Global curve length ", t = paste0("*t* = ", prettyNum(ch.W*ch.N), " s")))

if (kable_extra) {
  colnames(para_table) <- c("","")
  kable(para_table, escape = TRUE, align = "l",) %>%
  kable_styling(bootstrap_options = c("condensed"), full_width = FALSE, position = "left") %>%
  pack_rows("Settings", 1, 4) %>%
  pack_rows("Photoionisation cross section parameter", 5, 6) %>%
  pack_rows("Global curve properties", 7, 11)
} else {
  colnames(para_table) <- c("Parameter", "Value")
  kable(para_table, escape = TRUE, align = "l")
 # kable(para_table)
 # cat("<br><b>Note:</b> Install package 'kableExtra' for improved table style in this report.")
}





Try the OSLdecomposition package in your browser

Any scripts or data that you put into this service are public.

OSLdecomposition documentation built on Sept. 1, 2025, 1:09 a.m.