diff --git a/Waveform/WaveEventCnv/WaveByteStream/src/RawWaveformDecoderTool.cxx b/Waveform/WaveEventCnv/WaveByteStream/src/RawWaveformDecoderTool.cxx
index 2081244c434b40e789bf268284a7b56b13a89929..ec7a6641e3df94a8ac0b3b989e5c99303e3b62d7 100644
--- a/Waveform/WaveEventCnv/WaveByteStream/src/RawWaveformDecoderTool.cxx
+++ b/Waveform/WaveEventCnv/WaveByteStream/src/RawWaveformDecoderTool.cxx
@@ -20,31 +20,16 @@ RawWaveformDecoderTool::RawWaveformDecoderTool(const std::string& type,
 {
   declareInterface<RawWaveformDecoderTool>(this);
 
+  // These must be configured by job options
   declareProperty("CaloChannels", m_caloChannels);
-  m_caloChannels.push_back(0);
-  m_caloChannels.push_back(1);
-  m_caloChannels.push_back(2);
-  m_caloChannels.push_back(3);
-
   declareProperty("VetoChannels", m_vetoChannels);
-  m_vetoChannels.push_back(4);
-  m_vetoChannels.push_back(5);
-  m_vetoChannels.push_back(6);
-  m_vetoChannels.push_back(7);
-
   declareProperty("TriggerChannels", m_triggerChannels);
-  m_triggerChannels.push_back(8);
-  m_triggerChannels.push_back(9);
-  m_triggerChannels.push_back(10);
-  m_triggerChannels.push_back(11);
-
   declareProperty("PreshowerChannels", m_preshowerChannels);
-  m_preshowerChannels.push_back(12);
-  m_preshowerChannels.push_back(13);
 
+  // Test channels is provided for conveniene, not normally used
   declareProperty("TestChannels", m_testChannels);
-  m_testChannels.push_back(14);
 
+  // Clock should always be in channel 15
   declareProperty("ClockChannels", m_clockChannels);
   m_clockChannels.push_back(15);
 
@@ -58,6 +43,82 @@ StatusCode
 RawWaveformDecoderTool::initialize() 
 {
   ATH_MSG_DEBUG("RawWaveformDecoderTool::initialize()");
+
+  // Set up helpers
+  ATH_CHECK(detStore()->retrieve(m_ecalID, "EcalID"));
+  ATH_CHECK(detStore()->retrieve(m_vetoID, "VetoID"));
+  ATH_CHECK(detStore()->retrieve(m_triggerID, "TriggerID"));
+  ATH_CHECK(detStore()->retrieve(m_preshowerID, "PreshowerID"));
+
+  // Take each channel list and create identifiers
+
+  // Loop through digitizer channel lists creating Identifiers
+  m_identifierMap.clear();
+
+  // First, calorimeter.  Can either be 4 or 6 channels
+  // Bottom row first from left to right, then top row
+  int index=0;
+  int module=0;
+  int row=0;
+  int pmt=0;
+
+  int max_modules = m_caloChannels.size() / 2;
+  for (auto const& chan : m_caloChannels) {
+    row = index / max_modules;
+    module = index % max_modules;
+    index++;
+    // Only store in map if digitizer channel is valid
+    if (chan < 0) continue;
+    m_identifierMap[chan] = m_ecalID->pmt_id(row, module, pmt);
+    ATH_MSG_DEBUG("Mapped digitizer channel " << chan << " to calo ID: " << m_identifierMap[chan]);
+  }
+
+  // Next, veto detector.  Have station and plate here.
+  int station=0;
+  int plate=0;
+  pmt=0; 
+  index=0;
+
+  int max_stations=m_vetoChannels.size() / 2;
+  for (auto const& chan : m_vetoChannels) {
+    station = index / max_stations;
+    plate = index % max_stations;
+    index++;
+    // Only store in map if digitizer channel is valid
+    if (chan < 0) continue;
+    m_identifierMap[chan] = m_vetoID->pmt_id(station, plate, pmt);
+    ATH_MSG_DEBUG("Mapped digitizer channel " << chan << " to veto ID: " << m_identifierMap[chan]);
+  }
+
+  // Next, trigger detector.  Have pmt and plate.
+  pmt=0;
+  station=0;
+  index=0;
+  int max_plates=m_triggerChannels.size() / 2;
+  for (auto const& chan : m_triggerChannels) {
+    plate = index / max_plates;
+    pmt = index % max_plates;
+    index++;
+    // Only store in map if digitizer channel is valid
+    if (chan < 0) continue;
+    m_identifierMap[chan] = m_triggerID->pmt_id(station, plate, pmt);
+    ATH_MSG_DEBUG("Mapped dititizer channel " << chan << " to trigger ID: " << m_identifierMap[chan]);
+  }
+
+  // Finally, preshower detector.
+  pmt=0;
+  station=0;
+  plate=0;
+  index=0;
+  for (auto const& chan : m_preshowerChannels) {
+    plate = index;
+    index++;
+    // Only store in map if digitizer channel is valid
+    if (chan < 0) continue;
+    m_identifierMap[chan] = m_preshowerID->pmt_id(station, plate, pmt);
+    ATH_MSG_DEBUG("Mapped digitizer channel " << chan << " to preshower ID: " << m_identifierMap[chan]);
+  }
+
   return StatusCode::SUCCESS;
 }
 
@@ -111,7 +172,7 @@ RawWaveformDecoderTool::convert(const DAQFormats::EventFull* re,
     ATH_MSG_DEBUG("Found valid digitizer fragment");
   }
 
-  std::vector<unsigned int>* channelList;
+  std::vector<int>* channelList;
 
   if (key == std::string("CaloWaveforms")) {
     channelList = &m_caloChannels;
@@ -130,7 +191,7 @@ RawWaveformDecoderTool::convert(const DAQFormats::EventFull* re,
     return StatusCode::FAILURE;
   }
 
-  for (unsigned int channel: *channelList) {
+  for (int channel: *channelList) {
     ATH_MSG_DEBUG("Converting channel "+std::to_string(channel)+" for "+key);
 
     // Check if this has data
@@ -161,6 +222,11 @@ RawWaveformDecoderTool::convert(const DAQFormats::EventFull* re,
 		      << *frag);
     }
 
+    // Set ID if one exists (clock, for instance, doesn't have an identifier)
+    if (m_identifierMap.count(channel) == 1) {
+      wfm->setIdentifier(m_identifierMap[channel]);
+    }
+
     container->push_back(wfm);    
 
     // Sanity check
diff --git a/Waveform/WaveEventCnv/WaveByteStream/src/RawWaveformDecoderTool.h b/Waveform/WaveEventCnv/WaveByteStream/src/RawWaveformDecoderTool.h
index 252698c5fb975370945883d112a72c21ef7798a2..e2852f13772113d8e52a62c9655a18365cb80f61 100644
--- a/Waveform/WaveEventCnv/WaveByteStream/src/RawWaveformDecoderTool.h
+++ b/Waveform/WaveEventCnv/WaveByteStream/src/RawWaveformDecoderTool.h
@@ -13,6 +13,12 @@
 #include "EventFormats/DAQFormats.hpp"
 #include "WaveRawEvent/RawWaveformContainer.h"
 
+#include "Identifier/Identifier.h"
+#include "FaserCaloIdentifier/EcalID.h"
+#include "ScintIdentifier/VetoID.h"
+#include "ScintIdentifier/TriggerID.h"
+#include "ScintIdentifier/PreshowerID.h"
+
 // This class provides conversion between bytestream and Waveform objects
 
 class RawWaveformDecoderTool : public AthAlgTool {
@@ -32,13 +38,48 @@ class RawWaveformDecoderTool : public AthAlgTool {
 
 private:
   // List of channels to include in each container
-  std::vector<unsigned int> m_caloChannels;
-  std::vector<unsigned int> m_vetoChannels;
-  std::vector<unsigned int> m_triggerChannels;
-  std::vector<unsigned int> m_preshowerChannels;
-  std::vector<unsigned int> m_testChannels;
-  std::vector<unsigned int> m_clockChannels;
+  // List order must correspond to offline channel order
+  // All L/R designations refer to looking at the detector from
+  // the beam direction.
+  //
+  // In general, the ordering is layer (longitudinal), row (vertical), module (horizontal)
+  // Layers increase with longitudianl position downstream
+  // Rows increase from bottom to top
+  // Modules increase from right to left
+  //
+  // For all lists, use invalid channel (-1) to indicate detectors
+  // missing in sequence (i.e. 3 of 4 veto counters)
+  // 
+  // TI12 detector:
+  // Calorimeter order:
+  //   bottom right, bottom left, top right, top left
+  // Veto 
+  //   front to back.  
+  // Trigger
+  //   bottom right PMT, bottom left PMT, top right PMT, top left PMT
+  // Preshower
+  //   front to back
+  //   
+  // 2021 Testbeam detector:
+  // Calo order:
+  //   bottom right, bottom center, bottom left, top R, top C, top L
+  // All others are just in order front to back 
+
+  std::vector<int> m_caloChannels;
+  std::vector<int> m_vetoChannels;
+  std::vector<int> m_triggerChannels;
+  std::vector<int> m_preshowerChannels;
+  std::vector<int> m_testChannels;
+  std::vector<int> m_clockChannels;
+
+  // Identifiers keyed by digitizer channel
+  std::map<unsigned int, Identifier> m_identifierMap;
 
+  // ID helpers
+  const EcalID* m_ecalID{nullptr};
+  const VetoID* m_vetoID{nullptr};
+  const TriggerID* m_triggerID{nullptr};
+  const PreshowerID* m_preshowerID{nullptr};
 
 };
 
diff --git a/Waveform/WaveEventCnv/WaveEventAthenaPool/src/RawWaveformCnv_p0.cxx b/Waveform/WaveEventCnv/WaveEventAthenaPool/src/RawWaveformCnv_p0.cxx
index d298c0ff390393b6b8822bcc43167eee7c89ee9d..dedc71cbac77ccbdfa48dff63013cc6c16ab97f8 100644
--- a/Waveform/WaveEventCnv/WaveEventAthenaPool/src/RawWaveformCnv_p0.cxx
+++ b/Waveform/WaveEventCnv/WaveEventAthenaPool/src/RawWaveformCnv_p0.cxx
@@ -9,7 +9,8 @@ RawWaveformCnv_p0::persToTrans(const RawWaveform_p0* persObj, RawWaveform* trans
 
   // Just fill available data here
   // Rest of it patched up in RawWaveformContainerCnv_p0
-  transObj->setIdentifier(persObj->m_ID);
+  // Persistent object stores ID value, so instantiate Identifier here
+  transObj->setIdentifier(Identifier32(persObj->m_ID));
   transObj->setChannel(persObj->m_channel);
   transObj->setCounts(persObj->m_adc_counts);
 
@@ -23,7 +24,9 @@ RawWaveformCnv_p0::transToPers(const RawWaveform* transObj, RawWaveform_p0* pers
   // log << MSG::DEBUG << (*transObj) << endmsg;
   // log << MSG::DEBUG << "Persistent waveform (before):" << endmsg;
   // persObj->print();
-  persObj->m_ID = transObj->identify();
+
+  // Save actual ID number 
+  persObj->m_ID = transObj->identify32().get_compact();
   persObj->m_channel = transObj->channel();
 
   // Use copy here