Skip to content

Latest commit

 

History

History
211 lines (147 loc) · 16.3 KB

noise.org

File metadata and controls

211 lines (147 loc) · 16.3 KB

WCT Noise Modeling

Overview

WCT provides support for “noise” in various sub-packages:

  • WireCellUtil/Spectrum.h provides low level functions: rayleigh(), hermitian_mirror(), interp(), extrap(), alias(), resamples().
  • WireCellAux/NoiseTools.h provides a Collector class to form mean spectra of various flavors from noise-rich waveforms and Generator to produce waveforms given a $σ_k$ spectrum.
  • WireCellSigProc/NoiseModeler.h provides a flow graph node using a Collector intended to run on data rich in noise, collect mean spectra and convert the result to WCT noise spectra and output them to file. It makes use of a NoiseRanker to determine if any given waveform appears to be rich in noise. Also provided is the Undigitizer component which is effectively applies the inverse transform of Digitizer from gen and prepares voltage-level waves for input to the NoiseModeler.
  • WireCellGen/AddNoise.h provide flow graph nodes to add noise coherently by groups of channels and incoherently on a per-channel basis. I relies on one or more “noise models” to provide spectra. The GroupNoiseModel provides two interfaces, one for coherent and another for incoherent and both work on a channel-group basis by consuming WCT noise spectra and a WCT channel-groups data from file or configuration. The EmpiricalNoiseModel consumes WCT noise spectra and also supports runtime change in electronics shaping and gain as well as wire length binning.
  • Various Jsonnet job config and wire-cell-python canned plots.

See also this presentation: org, pdf

WCT Noise Spectra

WCT noise spectra defines a dataset describing mean spectral amplitudes and related metadata. They are provided to WCT C++ code as objects following a schema defined here. The datasets are provided via WCT configuration mechanism either directly as Jsonnet configuration data or by naming stand-alone data files (usually as compressed JSON but Jsonnet may also be provided).

WCT noise spectra are in the form of an array of spectrum objects. Each spectrum object follows a schema describing what attributes it must or may have. The attributes are categorized as required, optional or undefined. A spectrum object must contain all required attributes. A consumer of a WCT noise file must ignore undefined attributes. An optional attribute is one that is required for some consumers and undefined for others. Consumers of optional attributes may provide default values for use if the attribute is unspecified.

Required attributes

A WCT spectral object must provide these attributes:

nsamples
an integer number providing the size of (number of contiguous time series samples taken from) the waveforms used to form the mean spectral amplitude. The nsamples must not count any zero padding that may have been applied to a waveform prior to forming the waveform contribution to the mean spectrum (ie, prior to applying the DFT). The nsamples value is distinct from the size of the freqs and amps arrays. If using NoiseTools::Collector the value for nsamples is provided by the .nticks() method.
period
a floating-point number providing the original sample period (inverse of sampling frequency) of the waveform expressed in the WCT system of units for [time]. (eg, 0.5*us for MicroBooNE). Note, period is not necessarily related to the values provided by the freqs array.
freqs
an array of floating-point numbers providing frequency values expressed in the WCT system of units for [frequency]. Note, WCT’s base unit for [frequency] is not 1.0 Hz. The size of this array must be equal to the size of the amps array and an element of freqs must provide the frequency at which the corresponding element of the amps array was sampled. The freqs array may represent an irregular sampling and may be unordered. It should include samples at or near the zero and Nyquist frequencies. In particular it need not be a regular frequency sampling of 1/period nor extend beyond the Nyquist frequency.
amps
an array of floating-point numbers providing an estimate of a mean spectral amplitude in units of [voltage] in the WCT system of units. Note that here “amps” an abbreviation of “amplitude” and not “amperage”. The value of an element of amps may be derived from some sub-sampling or interpolation of an original distribution of DFT coefficients. That is, an element of amps is the an average $\langle|X_k|\rangle,\ k∈ [0,N(fft)-1]$ with $N(fft) ≥$ nsamples, over some number of waveforms of size nsamples. The inequality is typically due to zero-padding of the waveform prior to taking the DFT. Note: in preparing amps the user is recommended to provide a number of waveforms approximately equal to nsamples in order to co-optimize spectral resolution and statistical stability. User is also recommended to utilize NoiseTools::Collector for low-level noise modeling code or execute a job using NoiseModeler for a high-level development.

Optional attributes required by EmpiricalNoiseModel

The EmpiricalNoiseModel component requires these optional attributes:

const
a floating point number expressed in the same units as amps and which provides an estimate of the mean white noise, and thus constant, spectral amplitude $\langle|X_w|\rangle$.
gain
a floating-point number giving the electronics gain from which the voltage waveforms originated. The value must be in WCT system of units for [voltage]/[charge] (eg 14.0*wc.mV/wc.fC as expressed in WCT Jsonnet configuration). Note, this is not a unit-less, relative gain.
shaping
a floating-point number giving the electronics shaping time from which the waveforms originated. The value must be expressed in WCT units for [time].
plane
an integer number giving the plane index counting from zero and in the direction of nominal drift. Ie, U=0, V=1, W=2. This value must indicate the plane in which a channel resides in order for the spectra to be applicable.
wirelen
a floating-point number giving a wire length expressed in the WCT system of units for [length]. This value should be representative of (eg, binned over) wires for which the associated spectrum applies.

Optional attributes required by GroupNoiseModel

The GroupNoiseModel provides a model interface for both coherent and incoherent noise where spectra are grouped in some manner. It requires this optional attribute:

group
an integer identifying an abstract group to which channels may be associated. The association to channels may be provided by a WCT channel groups array. The use of groupID as this attribute name is deprecated.

WCT channel groups

The GroupNoiseModel and potentially other components require information on how to collect channels into distinct groups. The user provides this information in the form of WCT channel-groups data structures. These are in the form of an array of WCT channel-group objects, each of which has these required attributes:

group
an integer identifying a group. Over one channel-groups set, the group values may be discontinuous and may be unordered. Each group value must be unique in the set.
channels
an array of integer values providing the channel ID numbers to associate as a group. The channel IDs are as used in the WCT wire object configuration provided and described elsewhere.

Providing the above data

WCT noise spectra and channel group datasets are sometimes highly structured, even algorithmically generated, and sometimes unstructured and voluminous such as when they are derived from some external analysis.

To accomodate the user, developers of WCT C++ IConfigurable components should define a configuration parameter which may accept these datasets in two forms: (1) a string giving the name of some file holding the dataset or (2) an object or array that is directly provided as configuration data.

When the user provides a file, it may be either in JSON or Jsonnet form and either may be compressed. A user wanting to provide datasets as Jsonnet is suggested to look at test-noise-roundtrip.jsonnet and the other test-noise-*.jsonnet which it imports for examples.

Developers of WCT components can provide the user this flexibility with just a few additional lines in the configure() method of their C++ component. For example, to retrieve a channel group dataset:

#include "WireCellUtil/Persist.h"
void MyClass::configure(const WireCell::Configuration& cfg)
{
    auto jgroups = cfg["groups"];
    if (jgroups.isString()) {
        jgroups = Persist::load(map_file);
    }
    // ... code using jgroups ...
}

Round-trip Validation

The WCT noise code supports both modeling and simulating noise. Each is effectively the inverses of the other and so we may check that we get out what we put in. The “round-trip” check consists of these steps:

  • A set of fictional noise spectra and channel groups are defined.
  • We interpret the spectra as both coherent and incoherent.
  • Each interpretation has a GroupedNoiseModel and for each a noise frame is generated.
  • Each of these are digitized to ADC and result saved to file.
  • Each ADC-frame continues and an Undigitizer restores voltage level.
  • Each V-frame is analyzed by a NoiseModeler
    • Traces are judged by a NoiseRanker
    • Survivors added to a NoiseCollector
    • Finally, the grouped spectra are saved to a WCT noise spectra file.
  • Plots are made .

Input spectra

The input spectra can be viewed with:

wirecell-sigproc plot-noise-spectra \
  gen/test/test-noise-spectra.jsonnet orig.pdf

This will consume spectra which are generated by the Jsonnet file which produces a function with the following signature:

function(ngrps=10, nsamples=4096, nsave=64, period=0.5*wc.us,
         fpeak=0.1, rms=1*wc.mV)

It’s arguments are as listed:

ngrps
number of spectral groups to generate. The spectrum from a group will have a fraction grpnum/ngrps of the given rms.
nsamples
number of waveform time samples (number of “ticks”) from which the spectrum is assumed to have originated.
nsave
number of sub-smampled points to produce. This may be chosen equal to nsamples however typical analyses result in far higher frequency resolution than statistical stability (ie, nsamples $\gg$ nwaves) and thus chosing a small nsave emulates the common case of sub-sampling the result.
period
the waveform sampling period (“tick”) from which the spectrum is assumed to have originated. This must be expressed as a [time] value in the WCT system-of-units.
peak
the location of the spectral peak expressed as a fraction of the Nyquist frequency (0.5/period).
rms
the expected RMS from waveforms generated from the returned mean spectral amplitude. This must be expressed as a [voltage] value in the WCT system-of-units.

See below for guidance on how to provide meaningful values for peak and rms.

Model details

The user requires some understanding of the noise spectral model that is used in this test in order to provide proper values. The spectral shape in the frequency domain is chosen to follow the Rayleigh distribution, \[R(x;σ) = \frac{x}{σ^2}e-x^2/(2σ^2),\ x ≥ 0\] We will include a constant scaling term so that the full spectral model function is: \[\langle|X_k|\rangle \triangleq S⋅ R_k,\] where we discretize $R_k \triangleq R(f_k,σ_s)$ and define the parameter $σ_s = \mathtt{peak} * FNyquist$. The $σ_s$ may be chosen to place the peak near that of some real world noise spectrum and the constant scaling term $S$ may be chosen so that waveforms generated from this mean spectrum will have some (mean) RMS near that of some set of real world noise waveforms.

The choice of the Rayleigh distribution for the spectral shape is motivated by the fact it roughly reproduces the shapes of real-world noise spectra. It is otherwise an ad-hoc choice and not motivated by any physics. In particular, the choice does not relate to the coincidental but important fact that each spectral bin $k$ is also distributed by a Rayleigh distribution. Its distribution is governed by Rayleigh parameter $σ_k,\ k∈[0,N-1]$ where $N$ is given by nsamples.

The fact that the spectral bin values are Rayleigh-distributed provides a very practical tool which allows relating the model parameters to the (mean) RMS in time. This is possible because the first two “raw” moments of the Rayleigh distribution are related through the $σ_k$ parameter as:

\[\langle|X_k|\rangle = \sqrt{\frac{π}{2}}σ_k\] and \[\langle|X_k|^2\rangle = 2σ_k^2.\] Solving gives: \[\langle|X_k|\rangle^2 = \frac{π}{4} \langle|X_k|^2\rangle,\ k∈[0,N-1].\] We may define RMS on a per-waveform basis in the time domain and relate that to a frequency-domain representation using Perseval/Rayleigh energy theorem: \[σrms^2 = ∑_k|x_k|^2/N = E/N = ∑_k|X_k|^2/N^2.\] And we may form an average $\langle\cdot\rangle$ over many waveforms, \[\langle σrms^2 \rangle = ∑_k \langle |X_k|^2 \rangle / N^2 = ∑_k \frac{4}{π} \langle |X_k| \rangle^2/N^2\] Putting the above all together, we are left to choose $σ_s$ and $S$ so that the above sum gives desired $\langle\sigmarms^2\rangle$. We expect to select $σ_s$ so that the model peak will be approximately the same as the peak of some real-world noise spectrum. Taking $σ_s$ as given, we are left to solve for $S$: \[S^2 = \frac{π N^2 \langle\sigmarms^2\rangle}{4 ∑_k R_k^2} \] Thus the rms parameter is identified as providing the desired value of the $\sqrt{\langle\sigmarms^2\rangle}$. An example with $N=6000,\ σrms=1\ \mathrm{mV},\ \mathtt{peak}=0.1,\ T=0.5\ μ\mathrm{s}$ and saving only 100 subsampled points is shown:

test-noise-spectra-in-0.png

See test-noise-roundtrip.sh for exact command. The commands to reproduce such plots are described next.

Visualize the model

The above plot was made with a command like the following:

wirecell-sigproc plot-noise-spectra \
  -A ngrps=1 \
  -A nsamples=6000 \
  -A nsave=100 \
  gen/test/test-noise-spectra.jsonnet \
  specta.pdf

As illustrated, novel values for parameters of the model may be set from the command line. This same Jsonnet file may be used from WCT job configuration. This provides an easy way to define noise where the Rayliegh shape is sufficient to model a desired noise spectrum.

For comparison, an example of a spectrum modeling real-world noise from the ProtoDUNE-SP (PDSP) detector is given:

protodune-noise-spectra-v10000.png

The noise in PDSP is about 4 ADC RMS and its 12 bit ADC sees voltage in the range of 200 to 1600 mV and so expects about 1.3 mV RMS of noise measured in voltage input to the ADC. As a reminder, the simple model above has rms = 1 mV and achieves a smilar peak of 200 mV in amplitude for similar peak and same nsamples.

Performing the round-trip

A main configuration file for wire-cell is provided that uses the same test-noise-spectra.jsonnet described above to provide the input to the round-trip. The round-trip job can be exercised with default parameters like:

$ wire-cell -c  gen/test/test-noise-roundtrip.jsonnet
$ ls -l test-noise-roundtrip-*{npz,json.bz2}

The job flow graph is:

test-noise-roundtrip-flow-graph.png

It produces output that represents a cross product of $(inco,cohe) ⊗ (adc,dac)$ where

inco
incoherent grouped noise (3 groups)
cohe
coherent grouped noise (10 groups)
adc
the simulated ADC
dac
the ADC rescaled back to voltage level

Each cross produces a .npz file and each noise type results in a .json.bz2 file of output spectra. These too can be visualized

wirecell-sigproc plot-noise-spectra \
  test-noise-roundtrip-inco-spectra.json.bz2 \
  inco-spectra.pdf

wirecell-sigproc plot-noise-spectra \
  test-noise-roundtrip-cohe-spectra.json.bz2 \
  cohe-spectra.pdf

Or, run it all together as:

aux/test/test-noise-roundtrip.sh