diff --git a/lightmotif-py/lightmotif/lib.rs b/lightmotif-py/lightmotif/lib.rs
index c0846ffd683d246947c4d84463981852f2285020..3ac92f9884b69cb5e997e589b2aa6787340687ea 100644
--- a/lightmotif-py/lightmotif/lib.rs
+++ b/lightmotif-py/lightmotif/lib.rs
@@ -570,7 +570,7 @@ impl From<lightmotif::ScoringMatrix<lightmotif::Dna>> for ScoringMatrix {
 #[pyclass(module = "lightmotif.lib", sequence)]
 #[derive(Clone, Debug)]
 pub struct StripedScores {
-    scores: lightmotif::scores::StripedScores<C>,
+    scores: lightmotif::scores::StripedScores<f32, C>,
     shape: [Py_ssize_t; 2],
     strides: [Py_ssize_t; 2],
 }
@@ -668,8 +668,8 @@ impl StripedScores {
     }
 }
 
-impl From<lightmotif::scores::StripedScores<C>> for StripedScores {
-    fn from(scores: lightmotif::scores::StripedScores<C>) -> Self {
+impl From<lightmotif::scores::StripedScores<f32, C>> for StripedScores {
+    fn from(scores: lightmotif::scores::StripedScores<f32, C>) -> Self {
         // assert_eq!(scores.range().start, 0);
         // extract the matrix shape
         let cols = scores.matrix().columns();
diff --git a/lightmotif/src/pli/dispatch.rs b/lightmotif/src/pli/dispatch.rs
index 821747489ecf72c6f7920db17b85ddd51cd2d5f4..b9d9a5b04338747fa49ae343560a69597937790a 100644
--- a/lightmotif/src/pli/dispatch.rs
+++ b/lightmotif/src/pli/dispatch.rs
@@ -78,7 +78,7 @@ impl Score<Dna, <Dispatch as Backend>::LANES> for Pipeline<Dna, Dispatch> {
         pssm: M,
         seq: S,
         rows: Range<usize>,
-        scores: &mut StripedScores<<Dispatch as Backend>::LANES>,
+        scores: &mut StripedScores<f32, <Dispatch as Backend>::LANES>,
     ) where
         S: AsRef<StripedSequence<Dna, <Dispatch as Backend>::LANES>>,
         M: AsRef<ScoringMatrix<Dna>>,
@@ -107,7 +107,7 @@ impl Score<Protein, <Dispatch as Backend>::LANES> for Pipeline<Protein, Dispatch
         pssm: M,
         seq: S,
         rows: Range<usize>,
-        scores: &mut StripedScores<<Dispatch as Backend>::LANES>,
+        scores: &mut StripedScores<f32, <Dispatch as Backend>::LANES>,
     ) where
         S: AsRef<StripedSequence<Protein, <Dispatch as Backend>::LANES>>,
         M: AsRef<ScoringMatrix<Protein>>,
@@ -149,7 +149,7 @@ impl<A: Alphabet> Stripe<A, <Dispatch as Backend>::LANES> for Pipeline<A, Dispat
 impl<A: Alphabet> Maximum<<Dispatch as Backend>::LANES> for Pipeline<A, Dispatch> {
     fn argmax(
         &self,
-        scores: &StripedScores<<Dispatch as Backend>::LANES>,
+        scores: &StripedScores<f32, <Dispatch as Backend>::LANES>,
     ) -> Option<MatrixCoordinates> {
         match self.backend {
             #[cfg(any(target_arch = "x86", target_arch = "x86_64"))]
diff --git a/lightmotif/src/pli/mod.rs b/lightmotif/src/pli/mod.rs
index 297c60a6d940d67d827f3c3f8f2ccaebe3f6121d..f9bd6be181a70782c2d3d45961bccc916cbd7896 100644
--- a/lightmotif/src/pli/mod.rs
+++ b/lightmotif/src/pli/mod.rs
@@ -73,7 +73,7 @@ pub trait Score<A: Alphabet, C: StrictlyPositive> {
         pssm: M,
         seq: S,
         rows: Range<usize>,
-        scores: &mut StripedScores<C>,
+        scores: &mut StripedScores<f32, C>,
     ) where
         S: AsRef<StripedSequence<A, C>>,
         M: AsRef<ScoringMatrix<A>>,
@@ -105,7 +105,7 @@ pub trait Score<A: Alphabet, C: StrictlyPositive> {
     }
 
     /// Compute the PSSM scores into the given striped score matrix.
-    fn score_into<S, M>(&self, pssm: M, seq: S, scores: &mut StripedScores<C>)
+    fn score_into<S, M>(&self, pssm: M, seq: S, scores: &mut StripedScores<f32, C>)
     where
         S: AsRef<StripedSequence<A, C>>,
         M: AsRef<ScoringMatrix<A>>,
@@ -117,7 +117,7 @@ pub trait Score<A: Alphabet, C: StrictlyPositive> {
     }
 
     /// Compute the PSSM scores for every sequence positions.
-    fn score<S, M>(&self, pssm: M, seq: S) -> StripedScores<C>
+    fn score<S, M>(&self, pssm: M, seq: S) -> StripedScores<f32, C>
     where
         S: AsRef<StripedSequence<A, C>>,
         M: AsRef<ScoringMatrix<A>>,
@@ -133,7 +133,7 @@ pub trait Score<A: Alphabet, C: StrictlyPositive> {
 /// Used for finding the highest scoring site in a striped score matrix.
 pub trait Maximum<C: StrictlyPositive> {
     /// Find the matrix coordinates with the highest score.
-    fn argmax(&self, scores: &StripedScores<C>) -> Option<MatrixCoordinates> {
+    fn argmax(&self, scores: &StripedScores<f32, C>) -> Option<MatrixCoordinates> {
         if scores.is_empty() {
             return None;
         }
@@ -156,7 +156,7 @@ pub trait Maximum<C: StrictlyPositive> {
     }
 
     /// Find the highest score.
-    fn max(&self, scores: &StripedScores<C>) -> Option<f32> {
+    fn max(&self, scores: &StripedScores<f32, C>) -> Option<f32> {
         self.argmax(scores).map(|c| scores.matrix()[c])
     }
 }
@@ -204,7 +204,7 @@ pub trait Threshold<C: StrictlyPositive> {
     /// # Note
     /// The indices are not be sorted, and the actual order depends on the
     /// implementation.
-    fn threshold(&self, scores: &StripedScores<C>, threshold: f32) -> Vec<MatrixCoordinates> {
+    fn threshold(&self, scores: &StripedScores<f32, C>, threshold: f32) -> Vec<MatrixCoordinates> {
         let mut positions = Vec::new();
         for (i, row) in scores.matrix().iter().enumerate() {
             for col in 0..C::USIZE {
@@ -323,7 +323,7 @@ where
         pssm: M,
         seq: S,
         rows: Range<usize>,
-        scores: &mut StripedScores<C>,
+        scores: &mut StripedScores<f32, C>,
     ) where
         S: AsRef<StripedSequence<A, C>>,
         M: AsRef<ScoringMatrix<A>>,
@@ -337,7 +337,7 @@ where
     A: Alphabet,
     C: StrictlyPositive + MultipleOf<U16>,
 {
-    fn argmax(&self, scores: &StripedScores<C>) -> Option<MatrixCoordinates> {
+    fn argmax(&self, scores: &StripedScores<f32, C>) -> Option<MatrixCoordinates> {
         Sse2::argmax(scores)
     }
 }
@@ -378,7 +378,7 @@ impl Score<Dna, <Avx2 as Backend>::LANES> for Pipeline<Dna, Avx2> {
         pssm: M,
         seq: S,
         rows: Range<usize>,
-        scores: &mut StripedScores<<Avx2 as Backend>::LANES>,
+        scores: &mut StripedScores<f32, <Avx2 as Backend>::LANES>,
     ) where
         S: AsRef<StripedSequence<Dna, <Avx2 as Backend>::LANES>>,
         M: AsRef<ScoringMatrix<Dna>>,
@@ -393,7 +393,7 @@ impl Score<Protein, <Avx2 as Backend>::LANES> for Pipeline<Protein, Avx2> {
         pssm: M,
         seq: S,
         rows: Range<usize>,
-        scores: &mut StripedScores<<Avx2 as Backend>::LANES>,
+        scores: &mut StripedScores<f32, <Avx2 as Backend>::LANES>,
     ) where
         S: AsRef<StripedSequence<Protein, <Avx2 as Backend>::LANES>>,
         M: AsRef<ScoringMatrix<Protein>>,
@@ -416,7 +416,7 @@ impl<A: Alphabet> Stripe<A, <Avx2 as Backend>::LANES> for Pipeline<A, Avx2> {
 impl<A: Alphabet> Maximum<<Avx2 as Backend>::LANES> for Pipeline<A, Avx2> {
     fn argmax(
         &self,
-        scores: &StripedScores<<Avx2 as Backend>::LANES>,
+        scores: &StripedScores<f32, <Avx2 as Backend>::LANES>,
     ) -> Option<MatrixCoordinates> {
         Avx2::argmax(scores)
     }
diff --git a/lightmotif/src/pli/platform/avx2.rs b/lightmotif/src/pli/platform/avx2.rs
index c7bff4b27ebea0bfaf14dfd2380e4acf2eb268c0..ccec78b1fbfde54ae6af79db51e9f9f9e349cd72 100644
--- a/lightmotif/src/pli/platform/avx2.rs
+++ b/lightmotif/src/pli/platform/avx2.rs
@@ -99,7 +99,7 @@ unsafe fn score_avx2_permute<A>(
     pssm: &ScoringMatrix<A>,
     seq: &StripedSequence<A, <Avx2 as Backend>::LANES>,
     rows: Range<usize>,
-    scores: &mut StripedScores<<Avx2 as Backend>::LANES>,
+    scores: &mut StripedScores<f32, <Avx2 as Backend>::LANES>,
 ) where
     A: Alphabet,
     <A as Alphabet>::K: IsLessOrEqual<U8>,
@@ -190,7 +190,7 @@ unsafe fn score_avx2_gather<A>(
     pssm: &ScoringMatrix<A>,
     seq: &StripedSequence<A, <Avx2 as Backend>::LANES>,
     rows: Range<usize>,
-    scores: &mut StripedScores<<Avx2 as Backend>::LANES>,
+    scores: &mut StripedScores<f32, <Avx2 as Backend>::LANES>,
 ) where
     A: Alphabet,
 {
@@ -272,7 +272,7 @@ unsafe fn score_avx2_gather<A>(
 #[cfg(any(target_arch = "x86", target_arch = "x86_64"))]
 #[target_feature(enable = "avx2")]
 unsafe fn argmax_avx2(
-    scores: &StripedScores<<Avx2 as Backend>::LANES>,
+    scores: &StripedScores<f32, <Avx2 as Backend>::LANES>,
 ) -> Option<MatrixCoordinates> {
     if scores.max_index() > u32::MAX as usize {
         panic!(
@@ -615,7 +615,7 @@ impl Avx2 {
         pssm: M,
         seq: S,
         rows: Range<usize>,
-        scores: &mut StripedScores<<Avx2 as Backend>::LANES>,
+        scores: &mut StripedScores<f32, <Avx2 as Backend>::LANES>,
     ) where
         A: Alphabet,
         <A as Alphabet>::K: IsLessOrEqual<U8>,
@@ -652,7 +652,7 @@ impl Avx2 {
         pssm: M,
         seq: S,
         rows: Range<usize>,
-        scores: &mut StripedScores<<Avx2 as Backend>::LANES>,
+        scores: &mut StripedScores<f32, <Avx2 as Backend>::LANES>,
     ) where
         A: Alphabet,
         S: AsRef<StripedSequence<A, <Avx2 as Backend>::LANES>>,
@@ -683,7 +683,9 @@ impl Avx2 {
     }
 
     #[allow(unused)]
-    pub fn argmax(scores: &StripedScores<<Avx2 as Backend>::LANES>) -> Option<MatrixCoordinates> {
+    pub fn argmax(
+        scores: &StripedScores<f32, <Avx2 as Backend>::LANES>,
+    ) -> Option<MatrixCoordinates> {
         #[cfg(any(target_arch = "x86", target_arch = "x86_64"))]
         unsafe {
             argmax_avx2(scores)
diff --git a/lightmotif/src/pli/platform/neon.rs b/lightmotif/src/pli/platform/neon.rs
index bc1eb3aba8ac32e225db9b886c364bc8f943e4a9..21cf142550b24b5629b30ef7e6185a9584373e6a 100644
--- a/lightmotif/src/pli/platform/neon.rs
+++ b/lightmotif/src/pli/platform/neon.rs
@@ -198,7 +198,7 @@ impl Neon {
         pssm: M,
         seq: S,
         rows: Range<usize>,
-        scores: &mut StripedScores<C>,
+        scores: &mut StripedScores<f32, C>,
     ) where
         A: Alphabet,
         C: MultipleOf<U16>,
diff --git a/lightmotif/src/pli/platform/sse2.rs b/lightmotif/src/pli/platform/sse2.rs
index 92339d50ead2bd4dfea5d38f44f17b57fafdeaac..39cc5f430b6f7495ea3b33a8a965f93ccec489e8 100644
--- a/lightmotif/src/pli/platform/sse2.rs
+++ b/lightmotif/src/pli/platform/sse2.rs
@@ -34,7 +34,7 @@ unsafe fn score_sse2<A: Alphabet, C: MultipleOf<<Sse2 as Backend>::LANES>>(
     pssm: &ScoringMatrix<A>,
     seq: &StripedSequence<A, C>,
     rows: Range<usize>,
-    scores: &mut StripedScores<C>,
+    scores: &mut StripedScores<f32, C>,
 ) {
     // mask vectors for broadcasting uint8x16_t to uint32x4_t to floatx4_t
     let zero = _mm_setzero_si128();
@@ -97,7 +97,7 @@ unsafe fn score_sse2<A: Alphabet, C: MultipleOf<<Sse2 as Backend>::LANES>>(
 #[cfg(any(target_arch = "x86", target_arch = "x86_64"))]
 #[target_feature(enable = "sse2")]
 unsafe fn argmax_sse2<C: MultipleOf<<Sse2 as Backend>::LANES>>(
-    scores: &StripedScores<C>,
+    scores: &StripedScores<f32, C>,
 ) -> Option<MatrixCoordinates> {
     if scores.max_index() > u32::MAX as usize {
         panic!(
@@ -188,7 +188,7 @@ impl Sse2 {
         pssm: M,
         seq: S,
         rows: Range<usize>,
-        scores: &mut StripedScores<C>,
+        scores: &mut StripedScores<f32, C>,
     ) where
         A: Alphabet,
         C: MultipleOf<<Sse2 as Backend>::LANES>,
@@ -221,7 +221,7 @@ impl Sse2 {
 
     #[allow(unused)]
     pub fn argmax<C: MultipleOf<<Sse2 as Backend>::LANES>>(
-        scores: &StripedScores<C>,
+        scores: &StripedScores<f32, C>,
     ) -> Option<MatrixCoordinates> {
         #[cfg(any(target_arch = "x86", target_arch = "x86_64"))]
         unsafe {
diff --git a/lightmotif/src/pwm.rs b/lightmotif/src/pwm.rs
index c4e3d5e9efa253e8fd9cb87870ef9262419d4a7b..93f8b65d4d9454ee404995f148d1201508a703ff 100644
--- a/lightmotif/src/pwm.rs
+++ b/lightmotif/src/pwm.rs
@@ -479,7 +479,7 @@ impl<A: Alphabet> ScoringMatrix<A> {
     ///
     /// # Note
     /// Uses platform-accelerated implementation when available.
-    pub fn score<S, C>(&self, seq: S) -> StripedScores<C>
+    pub fn score<S, C>(&self, seq: S) -> StripedScores<f32, C>
     where
         C: StrictlyPositive,
         S: AsRef<StripedSequence<A, C>>,
diff --git a/lightmotif/src/scan.rs b/lightmotif/src/scan.rs
index 629a59beea0b018306d93fce256100ade95730ca..bab4b3f2ec23fa5397fae3d23e7390608c187ac3 100644
--- a/lightmotif/src/scan.rs
+++ b/lightmotif/src/scan.rs
@@ -61,7 +61,7 @@ impl Hit {
 pub struct Scanner<'a, A: Alphabet> {
     pssm: &'a ScoringMatrix<A>,
     seq: &'a StripedSequence<A, C>,
-    scores: CowMut<'a, StripedScores<C>>,
+    scores: CowMut<'a, StripedScores<f32, C>>,
     threshold: f32,
     block_size: usize,
     row: usize,
@@ -85,7 +85,7 @@ impl<'a, A: Alphabet> Scanner<'a, A> {
     }
 
     /// Use the given `StripedScores` as a buffer.
-    pub fn scores(&mut self, scores: &'a mut StripedScores<C>) -> &mut Self {
+    pub fn scores(&mut self, scores: &'a mut StripedScores<f32, C>) -> &mut Self {
         self.scores = CowMut::Borrowed(scores);
         self
     }
diff --git a/lightmotif/src/scores.rs b/lightmotif/src/scores.rs
index 1f26b6b6ce372622b19c9c8a8054991244caa657..0471f0d9d8b33c3192e907f1da1da8fdffdaadaa 100644
--- a/lightmotif/src/scores.rs
+++ b/lightmotif/src/scores.rs
@@ -8,6 +8,7 @@ use std::ops::Range;
 use crate::abc::Dna;
 use crate::dense::DenseMatrix;
 use crate::dense::MatrixCoordinates;
+use crate::dense::MatrixElement;
 use crate::err::InvalidData;
 use crate::num::StrictlyPositive;
 use crate::pli::dispatch::Dispatch;
@@ -85,16 +86,16 @@ impl<T> From<Scores<T>> for Vec<T> {
 
 /// Striped matrix storing scores for an equally striped sequence.
 #[derive(Clone, Debug)]
-pub struct StripedScores<C: StrictlyPositive> {
+pub struct StripedScores<T: MatrixElement, C: StrictlyPositive> {
     /// The raw data matrix storing the scores.
-    data: DenseMatrix<f32, C>,
+    data: DenseMatrix<T, C>,
     /// The total length of the `StripedSequence` these scores were obtained from.
     max_index: usize,
 }
 
-impl<C: StrictlyPositive> StripedScores<C> {
+impl<T: MatrixElement, C: StrictlyPositive> StripedScores<T, C> {
     /// Create a new striped score matrix with the given length and data.
-    fn new(data: DenseMatrix<f32, C>, max_index: usize) -> Result<Self, InvalidData> {
+    fn new(data: DenseMatrix<T, C>, max_index: usize) -> Result<Self, InvalidData> {
         Ok(Self { data, max_index })
     }
 
@@ -114,12 +115,12 @@ impl<C: StrictlyPositive> StripedScores<C> {
     }
 
     /// Return a reference to the striped matrix storing the scores.
-    pub fn matrix(&self) -> &DenseMatrix<f32, C> {
+    pub fn matrix(&self) -> &DenseMatrix<T, C> {
         &self.data
     }
 
     /// Return a mutable reference to the striped matrix storing the scores.
-    pub fn matrix_mut(&mut self) -> &mut DenseMatrix<f32, C> {
+    pub fn matrix_mut(&mut self) -> &mut DenseMatrix<T, C> {
         &mut self.data
     }
 
@@ -137,18 +138,18 @@ impl<C: StrictlyPositive> StripedScores<C> {
 
     /// Iterate over scores of individual sequence positions.
     #[inline]
-    pub fn iter(&self) -> Iter<'_, C> {
+    pub fn iter(&self) -> Iter<'_, T, C> {
         Iter::new(self)
     }
 
     /// Convert the striped scores into an array.
     #[inline]
-    pub fn unstripe(&self) -> Scores<f32> {
-        self.iter().cloned().collect::<Vec<f32>>().into()
+    pub fn unstripe(&self) -> Scores<T> {
+        self.iter().cloned().collect::<Vec<T>>().into()
     }
 }
 
-impl<C: StrictlyPositive> StripedScores<C>
+impl<C: StrictlyPositive> StripedScores<f32, C>
 where
     Pipeline<Dna, Dispatch>: Maximum<C>,
 {
@@ -169,7 +170,7 @@ where
     }
 }
 
-impl<C: StrictlyPositive> StripedScores<C>
+impl<C: StrictlyPositive> StripedScores<f32, C>
 where
     Pipeline<Dna, Dispatch>: Threshold<C>,
 {
@@ -190,49 +191,49 @@ where
     }
 }
 
-impl<C: StrictlyPositive> AsRef<DenseMatrix<f32, C>> for StripedScores<C> {
-    fn as_ref(&self) -> &DenseMatrix<f32, C> {
+impl<T: MatrixElement, C: StrictlyPositive> AsRef<DenseMatrix<T, C>> for StripedScores<T, C> {
+    fn as_ref(&self) -> &DenseMatrix<T, C> {
         self.matrix()
     }
 }
 
-impl<C: StrictlyPositive> AsMut<DenseMatrix<f32, C>> for StripedScores<C> {
-    fn as_mut(&mut self) -> &mut DenseMatrix<f32, C> {
+impl<T: MatrixElement, C: StrictlyPositive> AsMut<DenseMatrix<T, C>> for StripedScores<T, C> {
+    fn as_mut(&mut self) -> &mut DenseMatrix<T, C> {
         self.matrix_mut()
     }
 }
 
-impl<C: StrictlyPositive> Default for StripedScores<C> {
+impl<T: MatrixElement, C: StrictlyPositive> Default for StripedScores<T, C> {
     fn default() -> Self {
         StripedScores::empty()
     }
 }
 
-impl<C: StrictlyPositive> Index<usize> for StripedScores<C> {
-    type Output = f32;
+impl<T: MatrixElement, C: StrictlyPositive> Index<usize> for StripedScores<T, C> {
+    type Output = T;
     #[inline]
-    fn index(&self, index: usize) -> &f32 {
+    fn index(&self, index: usize) -> &T {
         let col = index / self.data.rows();
         let row = index % self.data.rows();
         &self.data[row][col]
     }
 }
 
-impl<C: StrictlyPositive> From<StripedScores<C>> for Vec<f32> {
-    fn from(scores: StripedScores<C>) -> Self {
+impl<T: MatrixElement, C: StrictlyPositive> From<StripedScores<T, C>> for Vec<T> {
+    fn from(scores: StripedScores<T, C>) -> Self {
         scores.iter().cloned().collect()
     }
 }
 
 // --- Iter --------------------------------------------------------------------
 
-pub struct Iter<'a, C: StrictlyPositive> {
-    scores: &'a StripedScores<C>,
+pub struct Iter<'a, T: MatrixElement, C: StrictlyPositive> {
+    scores: &'a StripedScores<T, C>,
     indices: Range<usize>,
 }
 
-impl<'a, C: StrictlyPositive> Iter<'a, C> {
-    fn new(scores: &'a StripedScores<C>) -> Self {
+impl<'a, T: MatrixElement, C: StrictlyPositive> Iter<'a, T, C> {
+    fn new(scores: &'a StripedScores<T, C>) -> Self {
         // Compute the last index
         let end = scores
             .max_index
@@ -242,29 +243,29 @@ impl<'a, C: StrictlyPositive> Iter<'a, C> {
         Self { scores, indices }
     }
 
-    fn get(&self, i: usize) -> &'a f32 {
+    fn get(&self, i: usize) -> &'a T {
         let col = i / self.scores.data.rows();
         let row = i % self.scores.data.rows();
         &self.scores.data[row][col]
     }
 }
 
-impl<'a, C: StrictlyPositive> Iterator for Iter<'a, C> {
-    type Item = &'a f32;
+impl<'a, T: MatrixElement, C: StrictlyPositive> Iterator for Iter<'a, T, C> {
+    type Item = &'a T;
     fn next(&mut self) -> Option<Self::Item> {
         self.indices.next().map(|i| self.get(i))
     }
 }
 
-impl<'a, C: StrictlyPositive> ExactSizeIterator for Iter<'a, C> {
+impl<'a, T: MatrixElement, C: StrictlyPositive> ExactSizeIterator for Iter<'a, T, C> {
     fn len(&self) -> usize {
         self.indices.len()
     }
 }
 
-impl<'a, C: StrictlyPositive> FusedIterator for Iter<'a, C> {}
+impl<'a, T: MatrixElement, C: StrictlyPositive> FusedIterator for Iter<'a, T, C> {}
 
-impl<'a, C: StrictlyPositive> DoubleEndedIterator for Iter<'a, C> {
+impl<'a, T: MatrixElement, C: StrictlyPositive> DoubleEndedIterator for Iter<'a, T, C> {
     fn next_back(&mut self) -> Option<Self::Item> {
         self.indices.next_back().map(|i| self.get(i))
     }