1. Core Idea of Position Sensing

For a neutron-sensitive tube read out at both ends (Side A and Side B), the principle is:

\[ \text{Position} = \frac{P_{sideA}}{P_{sideA} + P_{sideB}} \quad\text{and}\quad P_{\text{sum}} = P_{sideA} + P_{sideB}. \]

  • \(P_{sideA}\) and \(P_{sideB}\) are the peak amplitudes (or integrated charges) measured at each end of the tube when a particle interacts somewhere along its length.
  • If the two amplitudes are equal (i.e., \(P_{sideA} = P_{sideB}\)), the position is 0.5, meaning the interaction occurred in the center of the tube.
  • If \(P_{sideA} \gg P_{sideB}\), the event is closer to side A; if \(P_{sideB} \gg P_{sideA}\), the event is closer to side B.

The PSA block:

  1. Performs baseline corrections and filters on each of the two input channels (A and B).
  2. Detects the peak amplitude for each channel and for the sum \(P_{\text{sum}}\).
  3. Provides triggers, flags (pile-up, overflow) and final amplitude outputs for each side, plus the sum.

2. Digital Processing Stages

Each channel (A or B) is processed by a chain of up to five sub-blocks (all implemented in fixed-point arithmetic):

  1. Baseline Correction (Block 1)
  2. Pole-Zero (PZ) Compensation
  3. First second-order Gaussian Filter Section
  4. Second second-order Gaussian Filter Section
  5. Final Baseline Correction (Block 2)

At the end of this chain, the signal is shaped approximately like a “quasi-Gaussian” pulse, with zero baseline and minimal low-frequency drift.

2.1 Baseline Correction

The baseline correction aims to remove DC offset and ensure the signal baseline is around zero. One typical approach is:

  1. Break the data stream into chunks of length \(N\).
  2. In each chunk, find the minimum sample value, denoted \(i[n]\).
  3. Filter this minimum value with a first-order low-pass:

\[ w[n] = \frac{1}{k} \, i[n] \;+\; \Bigl(1 - \frac{1}{k}\Bigr) \, w[n-1] \]

where \(k\) is typically a power of two (e.g., 32).
4. Subtract this estimate \(w[n]\) from the samples in the next chunk to shift the baseline:

\[ u[N \cdot n + m] \;=\; i[N \cdot n + m] \;-\; w[n], \quad 0 \le m < N. \]

This “running minimum” approach can slightly under-correct in the presence of noise, but ensures that pulses remain well above zero.

2.2 Pole-Zero Compensation

A pole-zero compensation corrects for a large decay-time constant \(\tau_l\) in the incoming signal—often due to a charge amplifier or a preamplifier with a slow exponential decay. We want effectively a much shorter decay time \(\tau_s\).

The continuous-time PZ transfer function is often written as:

\[ H(s) \;=\; \frac{s + \tfrac{1}{\tau_l}}{s + \tfrac{1}{\tau_s}}, \]

where \(\tau_s\) is chosen to be significantly shorter than \(\tau_l\). We can rewrite:

\[H(s) = \frac{s + \tfrac{1}{\tau_l}}{s + \tfrac{1}{\tau_s}}= 1 \;+\;\frac{\tfrac{1}{\tau_l} \; -\; \tfrac{1}{\tau_s}}{s + \tfrac{1}{\tau_s}}. \]

When applying a discrete-time approximation (for instance via an impulse-invariance or matched-\(z\) transform), one arrives at a two-stage approach:

  1. A first-order IIR filter to approximate \([1/(1 + s\tau_s)]\).
  2. A direct feed-through plus a weighted version of that IIR to implement the difference \(\frac{1/\tau_l - 1/\tau_s}{s + 1/\tau_s}\).

In discrete form, one can approximate \(e^{-T_s/\tau_s} \approx 1 - T_s/\tau_s\) (assuming \(\tau_s \gg T_s\)). Then the equations become something like:

\[ w[n] \;=\; i[n] \;+\; \Bigl(1 - \frac{T_s}{\tau_s}\Bigr)\, w[n-1], \] \[ u[n] \;=\; i[n] \;+\; \Bigl(\frac{T_s}{\tau_l} \; -\; \frac{T_s}{\tau_s}\Bigr)\, w[n]. \]

These filters typically work in fixed point by scaling the coefficients (\(T_s / \tau_s\), etc.) by a power of two (e.g., \(2^{15}\)).

2.3 Gaussian Filtering

Each side is then put through a 4th-order Gaussian filter, implemented in two sequential 2nd-order IIR stages. The analog Gaussian filter’s transfer function can be approximated by:

\[ H(s) \;=\; \frac{4.899}{ 4.899 \; +\;11.42\,s\; +\;10.87\,s^2\; +\;5.073\,s^3\; +\;s^4 }, \] with poles at \(-1.35536 \pm j\,0.32795\) and \(-1.18108 \pm j\,1.0604\).

For a specified shaping time \(\tau\), one must divide these poles by \(2 \pi \tau\). To bring that into the discrete domain, a common approach is the matched-\(z\) transform (or impulse invariance for single exponentials). Each pair of complex-conjugate poles yields a 2nd-order IIR section:

\[ (1 - a\,z^{-1} + b\,z^{-2}), \]

where:

\[ a_1 = 2\, e^{-\alpha_1\,T} \,\cos(\beta_1\,T), \quad b_1 = e^{-2\alpha_1\,T}, \] and similarly for \(a_2, b_2\). Here:

  • \(\alpha_1,\beta_1\) come from the real and imaginary parts of the first pair of poles, \(-\alpha_1 \pm j\beta_1\).
  • \(\alpha_2,\beta_2\) come from the second pair.

If we let \(T = \tfrac{T_s}{2\pi\tau}\) with \(T_s\) the sampling period (e.g., \(T_s = 1/F_{\text{ADC}}\)):

\[ a_1 = 2\, e^{-1.355\,T}\,\cos(0.3279\,T), \quad b_1 = e^{-2.711\,T}, \] \[ a_2 = 2\, e^{-1.181\,T}\,\cos(1.0604\,T), \quad b_2 = e^{-2.362\,T}. \]

One then implements the recursive difference equation (per 2nd-order section):

\[ u[n] \;=\; i[n] \;+\; a_k\,u[n-1] \;-\; b_k\,u[n-2], \] where \((a_k, b_k)\) correspond to \((a_1, b_1)\) or \((a_2, b_2)\). After the first 2nd-order stage, the output is fed into the second 2nd-order stage to achieve the 4th-order response.

Note: Because we skip the explicit finite zeros that might appear in the full partial fraction expansion, the implementation is simpler (fewer multiplications), but still yields a near-Gaussian impulse response for typical shaping times.


3. Trigger and Peak Detection

Once the shaped signals from Side A and Side B are generated, the PSA logic can:

  1. Sum the two channels:
    \[ P_{\text{sum}}(n) = \text{Filter}_A(n) + \text{Filter}_B(n). \]
  2. Compare \(P_{\text{sum}}\) (or another selected trigger source) with a threshold (\(\text{TRG\_THRES}\)).
  3. When the threshold is crossed, it starts a window for peak finding:
    • The system searches for the global maximum in \(P_{\text{sum}}(n)\) within the gate window.
    • At that same time index (where \(P_{\text{sum}}(n)\) is max), it records the side A and side B shaped amplitudes:

\[ P_{sideA} = \text{Filter}_A(n{\text{max}}) \]

\[ P_{sideB} = \text{Filter}_B(n{\text{max}}) \]

  • The ratio \(\frac{P_{sideA}}{P_{sideA} + P_{sideB}}\) then gives the position.

Further logic checks for pile-up (another pulse arriving within the same gate), overflow (signal saturations), etc.


4. Practical Steps in Simulation

  1. Signal Generator Settings

    • Two channels, A0 and A1, each with a certain amplitude, decay time, and offset.
    • For example, if A0 = A1 in amplitude, you expect the measured position to be ~0.5.
  2. Filtering and PZ

    • In the simulation, the baseline correction removes the DC offset.
    • The pole-zero stage compensates for preamp decay times if needed.
    • The 4th-order Gaussian filter shapes the pulse to a clean “bell-like” shape, easier to measure its peak.
  3. Changing Amplitudes

    • By adjusting the amplitude of A0 or A1, one can see how \(P_{sideA}\) vs. \(P_{sideB}\) changes and, correspondingly, how \(\tfrac{P_{sideA}}{P_{\text{sum}}}\) shifts the detected interaction point.
  4. Outputs

    • The final shaped peaks for side A and side B are latched at the moment the sum is at maximum.
    • The simulation or hardware readout then delivers \(P_{sideA}\), \(P_{sideB}\), \(P_{\text{sum}}\), plus flags and time stamps.

5. Summary of Key Formulas

  1. Position: \[ \text{pos} = \frac{P_{sideA}}{P_{sideA} + P_{sideB}}. \]

  2. Pole-Zero (continuous): \[ H(s) = \frac{s + \tfrac{1}{\tau_l}}{s + \tfrac{1}{\tau_s}} \;=\; 1 \;+\; \frac{\tfrac{1}{\tau_l} - \tfrac{1}{\tau_s}}{s + \tfrac{1}{\tau_s}}. \] (Discrete approximation done via first-order IIR.)

  3. Gaussian Filter
    Two 2nd-order IIR sections, each of the form: \[ u[n] = i[n] \;+\; a\,u[n-1] \;-\; b\,u[n-2], \] with \[ a = 2\,e^{-\alpha\,T}\,\cos(\beta\,T), \quad b = e^{-2\alpha\,T}, \] for each conjugate pole pair \(-\alpha \pm j\beta\).

  4. Baseline Correction:

    • Min-based approach: \[ w[n] = \frac{1}{k}\, i[n] + \Bigl(1-\frac{1}{k}\Bigr)\, w[n-1], \quad \text{then} \quad u[n] = i[n] - w[\lfloor n/N\rfloor]. \]

6. Block Diagram Commentary and Simulation Guide:

Block diagram
Block diagram

In the block diagram, the two analog inputs (A0 and A1) feed into the Position Sense Detector (U0), which applies sequential stages—baseline correction, pole-zero compensation, multi-stage Gaussian filtering, and peak detection—to produce the outputs \(P_{sideA}\), \(P_{sideB}\), and their sum \(P_{\text{sum}}\).

To run the simulation, first configure the “A0” and “A1” signal generators by specifying amplitude, delay, and decay time (\(\tau\)) and click “Generate” to create the pulse events.

Signal generator stimulus for CH-A
Signal generator stimulus for CH-A

Signal generator stimulus for CH-B
Signal generator stimulus for CH-B

Then launch the simulation, where the shaped signals will appear as exponential decays.

Simulation output
Simulation output

When each pulse reaches its peak, U0 captures \(P_{sideA}\), \(P_{sideB}\), and \(P_{\text{sum}}\) numerically in the waveform viewer. By changing the amplitude on A1 (for example, raising the “Amplitude (ADC)” from 600 to a higher value), you will see \(P_{sideB}\) increase and the resulting position ratio \(\tfrac{P_{sideA}}{P_{sideA} + P_{sideB}}\) shift accordingly—moving the detected event closer to side B. In this way, you can verify how altering the pulses on A0 and A1 affects the simulated position along the detector, confirming the expected change in the peak values and output ratio.

Coefficient Calculation

Below is an example of how to compute the filter and pole-zero coefficients for a 1 µs decay time (\(\tau = 1\,\mu\mathrm{s}\)), 1 µs preamplifier decay, and a 125 MSPS (125 MHz) ADC clock. We define:

\[ \mathrm{CLOCK\_FREQUENCY} = 125000000.0 \quad,\quad \tau = 1\times 10^{-6} \quad,\quad pz = 0.1\times 10^{-6}. \]

First, compute the parameter \(TS\): \[ TS = \frac{2\,\pi}{\tau \,\times\, \mathrm{CLOCK\_FREQUENCY}}. \]

Then the four Gaussian filter coefficients are calculated by:

\[ \text{coeff11} = \Bigl\lfloor \exp\bigl(-2.71072\,TS\bigr)\;\times\;16384 \; +\; 0.5 \Bigr\rfloor, \] \[ \text{coeff12} = \Bigl\lfloor 2 \;\times\; \exp\bigl(-1.35536\,TS\bigr)\;\times\;\cos\bigl(0.327948\,TS\bigr)\;\times\;16384 \; +\; 0.5 \Bigr\rfloor, \] \[ \text{coeff21} = \Bigl\lfloor \exp\bigl(-2.36216\,TS\bigr)\;\times\;16384 \; +\; 0.5 \Bigr\rfloor, \] \[ \text{coeff22} = \Bigl\lfloor 2 \;\times\; \exp\bigl(-1.18108\,TS\bigr)\;\times\;\cos\bigl(1.06037\,TS\bigr)\;\times\;16384 \; +\; 0.5 \Bigr\rfloor. \]

Lastly, the pole-zero compensation coefficient is: \[ \text{pzcoeff2} = \Bigl\lfloor \frac{32768}{\mathrm{CLOCK\_FREQUENCY}\,\times\,pz} \Bigr\rfloor. \]

For the parameters specified above (i.e., 1 µs decay, 125 MHz sampling), these integer values typically evaluate to:

  • coeff11 = 0x37d9
  • coeff12 = 0x778e
  • coeff21 = 0x38d6
  • coeff22 = 0x7873
  • pzcoeff2 = 0x106

7 PSA I/O and settings user guide

Below is a summary of the main inputs, outputs, and configuration parameters for the PSA (Position Sense Amplitude) core. The signals are grouped by function (Inputs, Outputs, and Configuration). Where applicable, enumerated parameter values are shown in tables.


1. Inputs

Signal Size Description
AN_IN0 16 bits Analog data from ADC – Side A.
AN_IN1 16 bits Analog data from ADC – Side B.
CNTRL_RUN 1 bit When set to 1, enables the core.
TRIG_EXT 1 bit External trigger input (always enabled).
TRIG_VETO 1 bit Global trigger veto input.
COINC_IN 1 bit Coincidence-validation input. If coincidence is required (via register), this must go high for at least 1 clock cycle during the gate window.
TIMESTAMP 64 bits Free-running external timestamp input.
CLK 1 bit System clock (e.g., 80 MHz or 125 MHz, depending on hardware).
RESET 1 bit Resets the filtering process (global reset).

2. Outputs

Signal Size Description
AN_PROBE0 16 bits Analog probe for side A (selected by PRB_SEL).
AN_PROBE1 16 bits Analog probe for side B (selected by PRB_SEL).
D_PROBEx_0 16 bits Digital probe for side A/B (Trigger signal).
D_PROBEx_1 16 bits Digital probe for side A/B (Over threshold).
D_PROBEx_2 16 bits Digital probe for side A/B (Gate).
D_PROBEx_3 16 bits Digital probe for side A/B (Energy Ready).
OR_TRIGGER 1 bit Single-clock pulse triggered from side A OR side B.
TOTs 2 bits Over-threshold signals from trigger circuit.
TRIGGER 1 bit Trigger output for coincidence purposes (width set by TRG_OUT_W).
SIDE_A 16 bits Energy measured on side A (valid only when DV=1).
SIDE_B 16 bits Energy measured on side B (valid only when DV=1).
SIDE_B (sum) 16 bits Sum of side A + side B energies (valid only when DV=1). (Sometimes labeled SIDE_SUM.)
PILEUP_A 1 bit Pile-up flag on side A (a second trigger occurred during the gate window). Valid only when DV=1.
PILEUP_B 1 bit Pile-up flag on side B (a second trigger occurred during the gate window). Valid only when DV=1.
OVR_A 1 bit Overflow occurred on side A (invalid energy). Valid only when DV=1.
OVR_B 1 bit Overflow occurred on side B (invalid energy). Valid only when DV=1.
TIMESTAMP 64 bits Timestamp captured on trigger edge.
DATA 128 bits Output packet in “list format.”
DV 1 bit Data Valid indicator.

3. Configuration Parameters

These parameters control the internal filtering, trigger modes, coincidences, and readout. Most are 32-bit registers.

3.1 Polarity and Offset

Parameter Size Meaning
POL 32 bits Polarity of input signal. See table below.
OFFSET 32 bits Offset subtracted from the 12-bit input before polarity inversion.

POL (Polarity) Values:

Value Meaning
0 Positive
1 Negative

Parameter Size Meaning
TRG_SOURCE 32 bits Selects trigger data path (shaper output or raw input). See table below.
EN_COINC 32 bits Require external validation on COINC_IN.
TRG_MASK 32 bits Mask each side’s internal trigger.
TRG_MODE 32 bits Internal trigger mode (sum, no trigger, OR, AND). See table below.
TRG_THRES 32 bits Trigger threshold (0–65535).
TRG_HIST 32 bits Trigger hysteresis (subtracted on falling edge to avoid re-trigger on slow decay).
TRG_HOLD 32 bits Trigger holdoff time (in clock cycles), max 4095.
TRG_OUT_W 32 bits Width (in samples) of the output TRIGGER signal.

TRG_SOURCE:

Value Meaning
0 Use shaper output for trigger
1 Use input signal for trigger

EN_COINC:

Value Meaning
0 No external validation (DV always active)
1 External validation required (COINC_IN)

TRG_MASK:

Value Meaning
0 Both sides enabled
1 A side masked
2 B side masked
3 Both masked

TRG_MODE:

Value Meaning
0 Use sum (A + B) for trigger
1 No internal trigger
2 Use OR of side A & side B
3 Use AND of side A & side B

3.3 Filter Settings

Parameter Size Meaning
FLT_CFG 32 bits Bitfield configuration for filtering.
COEFF11 32 bits 1st Gaussian filter coefficient (see formula).
COEFF12 32 bits 1st Gaussian filter coefficient (see formula).
COEFF21 32 bits 2nd Gaussian filter coefficient (see formula).
COEFF22 32 bits 2nd Gaussian filter coefficient (see formula).
PZCOEFF 32 bits Pole-zero compensation coefficient (see formula).
PRB_SEL 32 bits Select analog monitor output (probe point).

FLT_CFG is typically started at 0x1F33 and includes bits for:

  • bits [3..0]: Attenuation (Section 1)
  • bits [7..4]: Attenuation (Section 2)
  • bit 8: Enable baseline correction (input)
  • bit 9: Enable pole-zero compensation
  • bit 10: Enable first second-order Gaussian filter section
  • bit 11: Enable second second-order Gaussian filter section
  • bit 12: Enable baseline correction (output)

PRB_SEL (Analog Monitor Selector):

Value Probed Signal
0 Signal with corrected offset and polarity
1 Signal with corrected offset only
2 Main filter output
3 Sum of both side main filter outputs
4 Peak detector output
5 Peak detector output on sum signal
6 Input baseline-correction algorithm output
7 Pole-zero compensation output
8 First filter algorithm output (divided by 2)
9 Second filter algorithm output (divided by 2)
A Output baseline algorithm (divided by 2)

3.4 Clock and Pole-Zero Formulas

  • COEFFxx are computed as: \[ \text{int}\bigl(\exp(\dots)\times 16384 + 0.5\bigr), \quad \text{int}\bigl(2\times \exp(\dots)\times\cos(\dots)\times 16384 + 0.5\bigr), \dots \] depending on the Gaussian poles and sample rate.

  • PZCOEFF is typically: \[ \text{int}\Bigl(\frac{32768}{(\mathrm{CLOCK\_FREQUENCY}\times pz)}\Bigr). \]


3.5 Notes on Use

  1. Configure FLT_CFG to enable/disable the relevant processing blocks (baseline, pole-zero, Gaussian).
  2. Set TRG_THRES and TRG_HIST carefully to avoid multiple triggers on a single pulse.
  3. Adjust offset and polarity if the incoming signal is negative.
  4. Monitor AN_PROBEx signals for diagnostic checks (to ensure no saturation occurs in the filter chain).

Conclusion

Using this chain (Baseline → PZ → Gaussian), each channel’s raw pulse is converted into a clean, peaked shape whose maximum is straightforward to measure. The system then captures the peak amplitudes simultaneously on both ends of the tube, and the charge (or amplitude) ratio provides the position of the incident neutron. By adjusting the amplitudes in the simulator (e.g., via A0, A1), one directly observes how \(P_{sideA}\) and \(P_{sideB}\) vary, and thus how the measured position shifts.

This approach is widely used for position-sensing He-3 or other gas-filled tubes and can be extended to multi-wire proportional chambers or other detectors requiring charge-division readout.