Merge pull request #927 from aaru-dps/fakeshemp/kryoflux

Read kryoflux flux data
This commit is contained in:
2026-04-05 12:45:09 +01:00
committed by GitHub
5 changed files with 670 additions and 62 deletions

View File

@@ -47,4 +47,12 @@ public sealed partial class KryoFlux
const string KF_HW_RV = "hwrv";
const string KF_SCK = "sck";
const string KF_ICK = "ick";
// Per KryoFlux spec: Clock frequencies for current hardware
// mck = Master Clock Frequency = ((18432000 * 73) / 14) / 2
// sck = Sample Frequency = mck / 2
// ick = Index Frequency = mck / 16
const double MCK = (18432000.0 * 73.0 / 14.0) / 2.0;
const double SCK = MCK / 2.0;
const double ICK = MCK / 16.0;
}

View File

@@ -0,0 +1,76 @@
// /***************************************************************************
// Aaru Data Preservation Suite
// ----------------------------------------------------------------------------
//
// Filename : Helpers.cs
// Author(s) : Natalia Portillo <claunia@claunia.com>
//
// Component : Disk image plugins.
//
// --[ Description ] ----------------------------------------------------------
//
// Contains helpers for KryoFlux STREAM images.
//
// --[ License ] --------------------------------------------------------------
//
// This library is free software; you can redistribute it and/or modify
// it under the terms of the GNU Lesser General Public License as
// published by the Free Software Foundation; either version 2.1 of the
// License, or (at your option) any later version.
//
// This library is distributed in the hope that it will be useful, but
// WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
// Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public
// License along with this library; if not, see <http://www.gnu.org/licenses/>.
//
// ----------------------------------------------------------------------------
// Copyright © 2011-2026 Natalia Portillo
// ****************************************************************************/
using System;
using System.Diagnostics.CodeAnalysis;
namespace Aaru.Images;
[SuppressMessage("ReSharper", "UnusedType.Global")]
public sealed partial class KryoFlux
{
/// <summary>
/// Converts a uint32 cell value to Aaru's flux representation format.
/// Format: byte array where 255 = overflow, remainder = value
/// </summary>
/// <param name="ticks">The cell value in clock cycles</param>
/// <returns>Flux representation as byte array</returns>
static byte[] UInt32ToFluxRepresentation(uint ticks)
{
uint over = ticks / 255;
if(over == 0) return [(byte)ticks];
var expanded = new byte[over + 1];
Array.Fill(expanded, (byte)255, 0, (int)over);
expanded[^1] = (byte)(ticks % 255);
return expanded;
}
/// <summary>
/// Calculates resolution in picoseconds from sample clock frequency.
/// Resolution = (1 / sck) * 1e12 picoseconds
/// </summary>
/// <param name="sck">Sample clock frequency in Hz</param>
/// <returns>Resolution in picoseconds</returns>
static ulong CalculateResolution(double sck)
{
if(sck <= 0) return 0;
double periodSeconds = 1.0 / sck;
double periodPicoseconds = periodSeconds * 1e12;
return (ulong)periodPicoseconds;
}
}

View File

@@ -37,16 +37,18 @@ using Aaru.CommonTypes.Structs;
namespace Aaru.Images;
/// <inheritdoc cref="Aaru.CommonTypes.Interfaces.IMediaImage" />
/// <inheritdoc cref="IFluxImage" />
/// <summary>Implements reading KryoFlux flux images</summary>
[SuppressMessage("ReSharper", "InconsistentNaming")]
public sealed partial class KryoFlux : IMediaImage, IVerifiableSectorsImage
public sealed partial class KryoFlux : IVerifiableSectorsImage, IFluxImage
{
const string MODULE_NAME = "KryoFlux plugin";
// TODO: These variables have been made public so create-sidecar can access to this information until I define an API >4.0
public SortedDictionary<byte, IFilter> tracks;
List<TrackCapture> _trackCaptures;
public KryoFlux() => _imageInfo = new ImageInfo
{
ReadableSectorTags = [],

View File

@@ -1,4 +1,4 @@
// /***************************************************************************
// /***************************************************************************
// Aaru Data Preservation Suite
// ----------------------------------------------------------------------------
//
@@ -37,9 +37,11 @@ using System.IO;
using System.Linq;
using Aaru.CommonTypes.Enums;
using Aaru.CommonTypes.Interfaces;
using Aaru.CommonTypes.Structs;
using Aaru.Filters;
using Aaru.Helpers;
using Aaru.Logging;
using Aaru.CommonTypes;
namespace Aaru.Images;
@@ -76,6 +78,7 @@ public sealed partial class KryoFlux
// TODO: This is supposing NoFilter, shouldn't
tracks = new SortedDictionary<byte, IFilter>();
_trackCaptures = [];
byte step = 1;
byte heads = 2;
var topHead = false;
@@ -131,46 +134,87 @@ public sealed partial class KryoFlux
_imageInfo.CreationTime = DateTime.MaxValue;
_imageInfo.LastModificationTime = DateTime.MinValue;
ErrorNumber processError = ProcessTrackFile(trackfile, (uint)head, (ushort)cylinder, trackFilter);
if(processError != ErrorNumber.NoError) return processError;
tracks.Add(t, trackFilter);
}
_imageInfo.Heads = heads;
_imageInfo.Cylinders = (uint)(tracks.Count / heads);
// TODO: Find a way to determine the media type from the track data.
_imageInfo.MediaType = MediaType.DOS_35_HD;
return ErrorNumber.NoError;
}
ErrorNumber ProcessTrackFile(string trackfile, uint head, ushort track, IFilter trackFilter)
{
if(!File.Exists(trackfile)) return ErrorNumber.NoSuchFile;
AaruLogging.Debug(MODULE_NAME, "Processing track file: {0} (head {1}, track {2})", trackfile, head, track);
Stream trackStream = trackFilter.GetDataForkStream();
trackStream.Seek(0, SeekOrigin.Begin);
while(trackStream.Position < trackStream.Length)
byte[] fileData = new byte[trackStream.Length];
trackStream.EnsureRead(fileData, 0, (int)trackStream.Length);
int fileSize = fileData.Length;
uint[] cellValues = new uint[fileSize];
uint[] cellStreamPositions = new uint[fileSize];
uint cellAccumulator = 0;
int streamOfs = 0;
uint streamPos = 0;
int cellPos = 0;
// Sample Frequency
double sck = SCK;
List<(uint streamPosition, uint timer, uint sysTime)> indexEvents = [];
int oobHeaderSize = Marshal.SizeOf<OobBlock>();
while(streamOfs < fileSize)
{
var blockId = (byte)trackStream.ReadByte();
byte curOp = fileData[streamOfs];
int curOpLen;
switch(blockId)
if(curOp == (byte)BlockIds.Oob)
{
case (byte)BlockIds.Oob:
{
trackStream.Position--;
if(fileSize - streamOfs < oobHeaderSize)
return ErrorNumber.InvalidArgument;
var oob = new byte[Marshal.SizeOf<OobBlock>()];
trackStream.EnsureRead(oob, 0, Marshal.SizeOf<OobBlock>());
OobBlock oobBlk = Marshal.ByteArrayToStructureLittleEndian<OobBlock>(oob);
byte[] oobBytes = new byte[oobHeaderSize];
Array.Copy(fileData, streamOfs, oobBytes, 0, oobHeaderSize);
OobBlock oobBlk = Marshal.ByteArrayToStructureLittleEndian<OobBlock>(oobBytes);
if(oobBlk.blockType == OobTypes.EOF)
{
trackStream.Position = trackStream.Length;
break;
}
if(oobBlk.blockType != OobTypes.KFInfo)
curOpLen = oobHeaderSize + oobBlk.length;
if(fileSize - streamOfs < curOpLen)
return ErrorNumber.InvalidArgument;
int oobDataStart = streamOfs + oobHeaderSize;
switch(oobBlk.blockType)
{
trackStream.Position += oobBlk.length;
case OobTypes.KFInfo:
{
byte[] kfinfo = new byte[oobBlk.length];
Array.Copy(fileData, oobDataStart, kfinfo, 0, oobBlk.length);
break;
}
var kfinfo = new byte[oobBlk.length];
trackStream.EnsureRead(kfinfo, 0, oobBlk.length);
string kfinfoStr = StringHandlers.CToString(kfinfo);
string[] lines = kfinfoStr.Split([','], StringSplitOptions.RemoveEmptyEntries);
DateTime blockDate = DateTime.Now;
DateTime blockTime = DateTime.Now;
var foundDate = false;
bool foundDate = false;
foreach(string[] kvp in lines.Select(static line => line.Split('='))
.Where(static kvp => kvp.Length == 2))
@@ -205,13 +249,26 @@ public sealed partial class KryoFlux
case KF_VERSION:
_imageInfo.ApplicationVersion = kvp[1];
break;
case KF_SCK:
if(double.TryParse(kvp[1], NumberStyles.Float, CultureInfo.InvariantCulture,
out double parsedSck))
sck = parsedSck;
break;
case KF_ICK:
// Parse KF_ICK for validation purposes; parsed value is not currently used.
// We use sample frequency space instead.
_ = double.TryParse(kvp[1], NumberStyles.Float, CultureInfo.InvariantCulture,
out _);
break;
}
}
if(foundDate)
{
var blockTimestamp = new DateTime(blockDate.Year,
DateTime blockTimestamp = new DateTime(blockDate.Year,
blockDate.Month,
blockDate.Day,
blockTime.Hour,
@@ -228,37 +285,199 @@ public sealed partial class KryoFlux
break;
}
case (byte)BlockIds.Flux2:
case (byte)BlockIds.Flux2_1:
case (byte)BlockIds.Flux2_2:
case (byte)BlockIds.Flux2_3:
case (byte)BlockIds.Flux2_4:
case (byte)BlockIds.Flux2_5:
case (byte)BlockIds.Flux2_6:
case (byte)BlockIds.Flux2_7:
case OobTypes.StreamInfo:
{
if(oobDataStart + Marshal.SizeOf<OobStreamRead>() <= fileSize)
{
byte[] streamReadBytes = new byte[Marshal.SizeOf<OobStreamRead>()];
Array.Copy(fileData, oobDataStart, streamReadBytes, 0,
Marshal.SizeOf<OobStreamRead>());
OobStreamRead oobStreamRead =
Marshal.ByteArrayToStructureLittleEndian<OobStreamRead>(streamReadBytes);
AaruLogging.Debug(MODULE_NAME,
"Stream Read at position {0}, elapsed time {1} ms",
oobStreamRead.streamPosition,
oobStreamRead.trTime);
}
break;
}
case OobTypes.Index:
{
if(oobDataStart + Marshal.SizeOf<OobIndex>() <= fileSize)
{
byte[] indexBytes = new byte[Marshal.SizeOf<OobIndex>()];
Array.Copy(fileData, oobDataStart, indexBytes, 0, Marshal.SizeOf<OobIndex>());
OobIndex oobIndex = Marshal.ByteArrayToStructureLittleEndian<OobIndex>(indexBytes);
AaruLogging.Debug(MODULE_NAME,
"Index signal at stream position {0}, timer {1}, sysTime {2}",
oobIndex.streamPosition,
oobIndex.timer,
oobIndex.sysTime);
indexEvents.Add((oobIndex.streamPosition, oobIndex.timer, oobIndex.sysTime));
}
break;
}
case OobTypes.StreamEnd:
{
if(oobDataStart + Marshal.SizeOf<OobStreamEnd>() <= fileSize)
{
byte[] streamEndBytes = new byte[Marshal.SizeOf<OobStreamEnd>()];
Array.Copy(fileData, oobDataStart, streamEndBytes, 0,
Marshal.SizeOf<OobStreamEnd>());
OobStreamEnd oobStreamEnd =
Marshal.ByteArrayToStructureLittleEndian<OobStreamEnd>(streamEndBytes);
AaruLogging.Debug(MODULE_NAME,
"Stream End at position {0}, result {1}",
oobStreamEnd.streamPosition,
oobStreamEnd.result);
}
break;
}
}
streamOfs += curOpLen;
continue;
}
// Non-OOB: determine operation length and decode flux data
bool newCell = false;
switch(curOp)
{
case (byte)BlockIds.Nop1:
curOpLen = 1;
break;
case (byte)BlockIds.Nop2:
trackStream.Position++;
curOpLen = 2;
continue;
break;
case (byte)BlockIds.Nop3:
curOpLen = 3;
break;
case (byte)BlockIds.Ovl16:
curOpLen = 1;
cellAccumulator += 0x10000;
break;
case (byte)BlockIds.Flux3:
trackStream.Position += 2;
curOpLen = 3;
cellAccumulator += (uint)((fileData[streamOfs + 1] << 8) | fileData[streamOfs + 2]);
newCell = true;
continue;
break;
default:
continue;
if(curOp >= 0x0E)
{
curOpLen = 1;
cellAccumulator += curOp;
newCell = true;
}
else if((curOp & 0xF8) == 0)
{
curOpLen = 2;
cellAccumulator += ((uint)curOp << 8) | fileData[streamOfs + 1];
newCell = true;
}
else
return ErrorNumber.InvalidArgument;
break;
}
if(fileSize - streamOfs < curOpLen)
return ErrorNumber.InvalidArgument;
if(newCell)
{
cellValues[cellPos] = cellAccumulator;
cellStreamPositions[cellPos] = streamPos;
cellPos++;
cellAccumulator = 0;
}
streamPos += (uint)curOpLen;
streamOfs += curOpLen;
}
// Store final partial cell for index resolution boundary
cellValues[cellPos] = cellAccumulator;
cellStreamPositions[cellPos] = streamPos;
int totalCells = cellPos;
// Resolve index stream positions to cell positions
List<uint> indexPositions = [];
if(indexEvents.Count > 0)
{
int nextIndex = 0;
uint nextIndexStreamPos = indexEvents[nextIndex].streamPosition;
for(int i = 0; i < totalCells; i++)
{
if(nextIndex >= indexEvents.Count)
break;
int nextCellPos = i + 1;
if(nextIndexStreamPos <= cellStreamPositions[nextCellPos])
{
if(i == 0 && cellStreamPositions[0] >= nextIndexStreamPos)
nextCellPos = 0;
indexPositions.Add((uint)nextCellPos);
AaruLogging.Debug(MODULE_NAME,
"Index {0} resolved to cell position {1}",
nextIndex, nextCellPos);
nextIndex++;
if(nextIndex < indexEvents.Count)
nextIndexStreamPos = indexEvents[nextIndex].streamPosition;
}
}
}
tracks.Add(t, trackFilter);
}
// Build flux pulses array
uint[] fluxPulses = new uint[totalCells];
Array.Copy(cellValues, fluxPulses, totalCells);
_imageInfo.Heads = heads;
_imageInfo.Cylinders = (uint)(tracks.Count / heads);
ulong resolution = CalculateResolution(sck);
AaruLogging.Error(Localization.Flux_decoding_is_not_yet_implemented);
AaruLogging.Debug(MODULE_NAME,
"Decoded {0} flux pulses, {1} index signals, resolution {2} ps",
fluxPulses.Length,
indexPositions.Count,
resolution);
return ErrorNumber.NotImplemented;
TrackCapture capture = new TrackCapture
{
head = head,
track = track,
resolution = resolution,
fluxPulses = fluxPulses,
indexPositions = indexPositions.ToArray()
};
_trackCaptures.Add(capture);
return ErrorNumber.NoError;
}
/// <inheritdoc />
@@ -319,5 +538,232 @@ public sealed partial class KryoFlux
return ErrorNumber.NotImplemented;
}
#endregion
#region IFluxImage Members
/// <inheritdoc />
public ErrorNumber CapturesLength(uint head, ushort track, byte subTrack, out uint length)
{
length = 0;
if(_trackCaptures == null) return ErrorNumber.NotOpened;
// KryoFlux doesn't support subtracks - only subTrack 0 is valid
if(subTrack != 0) return ErrorNumber.OutOfRange;
// KryoFlux has one file per track/head, which results in exactly one capture
bool hasCapture = _trackCaptures.Any(c => c.head == head && c.track == track);
length = hasCapture ? 1u : 0u;
return ErrorNumber.NoError;
}
/// <inheritdoc />
public ErrorNumber ReadFluxIndexResolution(uint head, ushort track, byte subTrack, uint captureIndex,
out ulong resolution)
{
resolution = 0;
// KryoFlux doesn't support subtracks - only subTrack 0 is valid
if(subTrack != 0) return ErrorNumber.OutOfRange;
// KryoFlux has one file per track/head, which results in exactly one capture (captureIndex 0)
if(captureIndex != 0) return ErrorNumber.OutOfRange;
TrackCapture capture = _trackCaptures.Find(c => c.head == head && c.track == track);
if(capture == null) return ErrorNumber.OutOfRange;
resolution = capture.resolution;
return ErrorNumber.NoError;
}
/// <inheritdoc />
public ErrorNumber ReadFluxDataResolution(uint head, ushort track, byte subTrack, uint captureIndex,
out ulong resolution)
{
resolution = 0;
// KryoFlux doesn't support subtracks - only subTrack 0 is valid
if(subTrack != 0) return ErrorNumber.OutOfRange;
// KryoFlux has one file per track/head, which results in exactly one capture (captureIndex 0)
if(captureIndex != 0) return ErrorNumber.OutOfRange;
TrackCapture capture = _trackCaptures.Find(c => c.head == head && c.track == track);
if(capture == null) return ErrorNumber.OutOfRange;
resolution = capture.resolution;
return ErrorNumber.NoError;
}
/// <inheritdoc />
public ErrorNumber ReadFluxResolution(uint head, ushort track, byte subTrack, uint captureIndex,
out ulong indexResolution, out ulong dataResolution)
{
indexResolution = dataResolution = 0;
// KryoFlux doesn't support subtracks - only subTrack 0 is valid
if(subTrack != 0) return ErrorNumber.OutOfRange;
// KryoFlux has one file per track/head, which results in exactly one capture (captureIndex 0)
if(captureIndex != 0) return ErrorNumber.OutOfRange;
TrackCapture capture = _trackCaptures.Find(c => c.head == head && c.track == track);
if(capture == null) return ErrorNumber.OutOfRange;
indexResolution = dataResolution = capture.resolution;
return ErrorNumber.NoError;
}
/// <inheritdoc />
public ErrorNumber ReadFluxCapture(uint head, ushort track, byte subTrack, uint captureIndex,
out ulong indexResolution, out ulong dataResolution, out byte[] indexBuffer,
out byte[] dataBuffer)
{
indexBuffer = dataBuffer = null;
indexResolution = dataResolution = 0;
// KryoFlux doesn't support subtracks - only subTrack 0 is valid
if(subTrack != 0) return ErrorNumber.OutOfRange;
// KryoFlux has one file per track/head, which results in exactly one capture (captureIndex 0)
if(captureIndex != 0) return ErrorNumber.OutOfRange;
ErrorNumber error = ReadFluxResolution(head, track, subTrack, captureIndex, out indexResolution,
out dataResolution);
if(error != ErrorNumber.NoError) return error;
error = ReadFluxDataCapture(head, track, subTrack, captureIndex, out dataBuffer);
if(error != ErrorNumber.NoError) return error;
error = ReadFluxIndexCapture(head, track, subTrack, captureIndex, out indexBuffer);
return error;
}
/// <inheritdoc />
public ErrorNumber ReadFluxIndexCapture(uint head, ushort track, byte subTrack, uint captureIndex,
out byte[] buffer)
{
buffer = null;
// KryoFlux doesn't support subtracks - only subTrack 0 is valid
if(subTrack != 0) return ErrorNumber.OutOfRange;
// KryoFlux has one file per track/head, which results in exactly one capture (captureIndex 0)
if(captureIndex != 0) return ErrorNumber.OutOfRange;
TrackCapture capture = _trackCaptures.Find(c => c.head == head && c.track == track);
if(capture == null) return ErrorNumber.OutOfRange;
var tmpBuffer = new List<byte>();
uint previousPosition = 0;
foreach(uint indexPos in capture.indexPositions)
{
// Calculate ticks from start to this index position
uint ticks = 0;
for(uint i = previousPosition; i < indexPos && i < capture.fluxPulses.Length; i++)
ticks += capture.fluxPulses[i];
uint deltaTicks = ticks;
tmpBuffer.AddRange(UInt32ToFluxRepresentation(deltaTicks));
previousPosition = indexPos;
}
buffer = tmpBuffer.ToArray();
return ErrorNumber.NoError;
}
/// <inheritdoc />
public ErrorNumber
ReadFluxDataCapture(uint head, ushort track, byte subTrack, uint captureIndex, out byte[] buffer)
{
buffer = null;
// KryoFlux doesn't support subtracks - only subTrack 0 is valid
if(subTrack != 0) return ErrorNumber.OutOfRange;
// KryoFlux has one file per track/head, which results in exactly one capture (captureIndex 0)
if(captureIndex != 0) return ErrorNumber.OutOfRange;
TrackCapture capture = _trackCaptures.Find(c => c.head == head && c.track == track);
if(capture == null) return ErrorNumber.OutOfRange;
var tmpBuffer = new List<byte>();
foreach(uint pulse in capture.fluxPulses) tmpBuffer.AddRange(UInt32ToFluxRepresentation(pulse));
buffer = tmpBuffer.ToArray();
return ErrorNumber.NoError;
}
/// <inheritdoc />
public ErrorNumber SubTrackLength(uint head, ushort track, out byte length)
{
length = 0;
if(_trackCaptures == null) return ErrorNumber.NotOpened;
// KryoFlux doesn't support subtracks - filenames only contain cylinder and head
// Check if any captures exist for this track
List<TrackCapture> captures = _trackCaptures.FindAll(c => c.head == head && c.track == track);
if(captures.Count <= 0) return ErrorNumber.OutOfRange;
// Always return 1 since KryoFlux doesn't support subtracks
length = 1;
return ErrorNumber.NoError;
}
/// <inheritdoc />
public ErrorNumber GetAllFluxCaptures(out List<FluxCapture> captures)
{
captures = [];
if(_trackCaptures is { Count: > 0 })
{
// Group captures by head/track to assign capture indices
// Note: KryoFlux doesn't support subtracks, so subTrack is always 0
var grouped = _trackCaptures.GroupBy(c => new { c.head, c.track })
.ToList();
foreach(var group in grouped)
{
uint captureIndex = 0;
foreach(TrackCapture trackCapture in group)
{
captures.Add(new FluxCapture
{
Head = trackCapture.head,
Track = trackCapture.track,
SubTrack = 0, // KryoFlux doesn't support subtracks
CaptureIndex = captureIndex++,
IndexResolution = trackCapture.resolution,
DataResolution = trackCapture.resolution
});
}
}
}
return ErrorNumber.NoError;
}
#endregion
}

View File

@@ -46,5 +46,81 @@ public sealed partial class KryoFlux
public readonly ushort length;
}
#endregion
#region Nested type: OobStreamRead
/// <summary>
/// Per KryoFlux spec: OOB Stream Read block structure (4 bytes after OOB header).
/// Contains stream position and elapsed time since last transfer.
/// </summary>
[StructLayout(LayoutKind.Sequential, Pack = 1)]
readonly struct OobStreamRead
{
public readonly uint streamPosition;
public readonly uint trTime;
}
#endregion
#region Nested type: OobIndex
/// <summary>
/// Per KryoFlux spec: OOB Index block structure (12 bytes after OOB header).
/// Contains stream position, timer value, and system time when index was detected.
/// </summary>
[StructLayout(LayoutKind.Sequential, Pack = 1)]
readonly struct OobIndex
{
public readonly uint streamPosition;
public readonly uint timer;
public readonly uint sysTime;
}
#endregion
#region Nested type: OobStreamEnd
/// <summary>
/// Per KryoFlux spec: OOB Stream End block structure (12 bytes after OOB header).
/// Contains stream position and result code.
/// </summary>
[StructLayout(LayoutKind.Sequential, Pack = 1)]
readonly struct OobStreamEnd
{
public readonly uint streamPosition;
// The specification says it's a ulong, but the actual data is a uint
public readonly uint result;
}
#endregion
#region Nested type: TrackCapture
/// <summary>
/// Internal structure representing a flux capture for a single track.
/// Contains decoded flux pulse data, index signal positions, and resolution information.
/// </summary>
public class TrackCapture
{
public uint head;
public ushort track;
/// <summary>
/// Resolution (sample rate) of the flux capture in picoseconds.
/// Calculated from KryoFlux clock frequencies (sck).
/// </summary>
public ulong resolution;
/// <summary>
/// Array of flux pulse intervals in clock cycles. Each value represents the time interval
/// between flux reversals, measured in KryoFlux clock cycles.
/// </summary>
public uint[] fluxPulses;
/// <summary>
/// Array of index positions. Each value is an index into the fluxPulses array
/// indicating where an index signal occurs.
/// </summary>
public uint[] indexPositions;
}
#endregion
}