Skip to content

Commit

Permalink
- Implemented Eizenga's M!=0 formula to allow any match score
Browse files Browse the repository at this point in the history
- Refactor debug report
- Updated README
  • Loading branch information
smarco committed Jul 6, 2022
1 parent 3506562 commit bdb09fa
Show file tree
Hide file tree
Showing 30 changed files with 569 additions and 266 deletions.
25 changes: 18 additions & 7 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@

### 1.1 What is WFA?

The wavefront alignment (WFA) algorithm is an **exact** gap-affine algorithm that takes advantage of homologous regions between the sequences to accelerate the alignment process. Unlike to traditional dynamic programming algorithms that run in quadratic time, the WFA runs in time `O(ns+s^2)`, proportional to the sequence length `n` and the alignment score `s`, using `O(s^2)` memory. Moreover, the WFA algorithm exhibits simple computational patterns that the modern compilers can automatically vectorize for different architectures without adapting the code. To intuitively illustrate why the WFA algorithm is so interesting, look at the following figure. The left panel shows the cells computed by a classical dynamic programming based algorithm (like Smith-Waterman or Needleman Wunsch). In contrast, the right panel shows the cells computed by the WFA algorithm to obtain the same result (i.e., the optimal alignment).
The wavefront alignment (WFA) algorithm is an **exact** gap-affine algorithm that takes advantage of homologous regions between the sequences to accelerate the alignment process. Unlike to traditional dynamic programming algorithms that run in quadratic time, the WFA runs in time `O(ns+s^2)`, proportional to the sequence length `n` and the alignment score `s`, using `O(s^2)` memory (or `O(s)` using the ultralow/BiWFA mode). Moreover, the WFA algorithm exhibits simple computational patterns that the modern compilers can automatically vectorize for different architectures without adapting the code. To intuitively illustrate why the WFA algorithm is so interesting, look at the following figure. The left panel shows the cells computed by a classical dynamic programming based algorithm (like Smith-Waterman or Needleman Wunsch). In contrast, the right panel shows the cells computed by the WFA algorithm to obtain the same result (i.e., the optimal alignment).

<p align = "center">
<img src = "img/wfa.vs.swg.png" width="750px">
Expand Down Expand Up @@ -47,7 +47,7 @@ Section [WFA2-lib features](#wfa2.features) explores the most relevant options a
- The WFA algorithm is an **exact algorithm**. If no heuristic is applied (e.g., band or adaptive pruning), the core algorithm guarantees to always find the optimal solution (i.e., best alignment score). Since its first release, some authors have referenced the WFA as approximated or heuristic, which is NOT the case.


- Given two sequences of length `n`, traditional dynamic-programming (DP) based methods (like Smith-Waterman or Needleman-Wunsch) compute the optimal alignment in `O(n^2)` time, using `O(n^2)` memory. In contrast, the WFA algorithm requires `O(ns+s^2)` time and `O(s^2)` memory (being `s` the optimal alignment score). Therefore, **the memory consumption of the WFA algorithm is not intrinsically higher than that of other methods**. Most DP-based methods can use heuristics (like banded, X-drop, or Z-drop) to reduce the execution time and the memory usage at the expense of losing accuracy. Likewise, **the WFA algorithm can also use heuristics to reduce the execution time and memory usage**.
- Given two sequences of length `n`, traditional dynamic-programming (DP) based methods (like Smith-Waterman or Needleman-Wunsch) compute the optimal alignment in `O(n^2)` time, using `O(n^2)` memory. In contrast, the WFA algorithm requires `O(ns+s^2)` time and `O(s^2)` memory (being `s` the optimal alignment score). Therefore, **the memory consumption of the WFA algorithm is not intrinsically higher than that of other methods**. Most DP-based methods can use heuristics (like banded, X-drop, or Z-drop) to reduce the execution time and the memory usage at the expense of losing accuracy. Likewise, **the WFA algorithm can also use heuristics to reduce the execution time and memory usage**. Moreover, the memory mode `ultralow` uses the BiWFA algorithm to execute in `O(ns+s^2)` time and linear `O(s)` memory.


- **A note for the fierce competitors.** I can understand that science and publishing have become a fierce competition these days. Many researchers want their methods to be successful and popular, seeking funding, tenure, or even fame. If you are going to benchmark the WFA using the least favourable configuration, careless programming, and a disadvantageous setup, please, go ahead. But remember, researchers like you have put a lot of effort into developing the WFA. We all joined this "competition" because we sought to find better methods that could truly help other researchers. So, try to be nice, tone down the marketing, and produce fair evaluations and honest publications.
Expand Down Expand Up @@ -147,7 +147,7 @@ cout << "Alignment score " << aligner.getAlignmentScore() << endl;
* **Exact alignment** method that computes the optimal **alignment score** and/or **alignment CIGAR**.
* Supports **multiple distance metrics** (i.e., indel, edit, gap-lineal, gap-affine, and dual-gap gap-affine).
* Allows performing **end-to-end** (a.k.a. global) and **ends-free** (e.g., semi-global, extension, overlap) alignment.
* Implements **low-memory modes** to reduce and control memory consumption.
* Implements **low-memory modes** to reduce and control memory consumption (down to `O(s)` using the `ultralow` mode).
* Supports various **heuristic strategies** to use on top of the core WFA algorithm.
* WFA2-lib **operates with plain ASCII strings**. Although we mainly focus on aligning DNA/RNA sequences, the WFA algorithm and the WFA2-lib implementation work with any pair of strings. Moreover, these sequences do not have to be pre-processed (e.g., packed or profiled), nor any table must be precomputed (like the query profile, used within some Smith-Waterman implementations).
* Due to its simplicity, the WFA algorithm can be automatically vectorized for any SIMD-compliant CPU supported by the compiler. For this reason, **the WFA2-lib implementation is independent of any specific ISA or processor model**. Unlike other hardware-dependent libraries, we aim to offer a multiplatform pairwise-alignment library that can be executed on different processors and models (e.g., SSE, AVX2, AVX512, POWER-ISA, ARM, NEON, SVE, SVE2, RISCV-RVV, ...).
Expand Down Expand Up @@ -385,7 +385,7 @@ The WFA2 library allows computing alignments with different spans or shapes. Alt

### <a name="wfa2.mem"></a> 3.4 Memory modes

The WFA2 library implements various memory modes: `wavefront_memory_high`, `wavefront_memory_med`, `wavefront_memory_low`. These modes allow regulating the overall memory consumption at the expense of execution time. The standard WFA algorithm, which stores explicitly all wavefronts in memory, correspond to the mode `wavefront_memory_high`. The other methods progressively reduce memory usage at the expense of slightly larger alignment times. These memory modes can be used transparently with other alignment options and generate identical results. Note that this option does not affect the score-only alignment mode (it already uses a minimal memory footprint of `O(s)`).
The WFA2 library implements various memory modes: `wavefront_memory_high`, `wavefront_memory_med`, `wavefront_memory_low`, and , `wavefront_memory_ultralow`. These modes allow regulating the overall memory consumption at the expense of execution time. The standard WFA algorithm, which stores explicitly all wavefronts in memory, correspond to the mode `wavefront_memory_high`. The other methods progressively reduce memory usage at the expense of slightly larger alignment times. These memory modes can be used transparently with other alignment options and generate identical results. Note that this option does not affect the score-only alignment mode (it already uses a minimal memory footprint of `O(s)`).

```C
wavefront_aligner_attr_t attributes = wavefront_aligner_attr_default;
Expand All @@ -394,7 +394,7 @@ The WFA2 library implements various memory modes: `wavefront_memory_high`, `wave

### <a name="wfa2.heuristics"></a> 3.5 Heuristic modes

The WFA algorithm can be used combined with many heuristics to reduce the alignment time and memory used. As it happens to other alignment methods, heuristics can result in suboptimal solutions and loss of accuracy. Moreover, some heuristics may drop the alignment if the sequences exceed certain divergence thresholds (i.e., x-drop/z-drop). Due to the popularity and efficiency of these methods, the WFA2 library implements many of these heuristics. Note, **it is not about how little DP-matrix you compute, but about how good the resulting alignments are.**
The WFA algorithm can be used combined with many heuristics to reduce the alignment time and memory used. As it happens to other alignment methods, heuristics can result in suboptimal solutions and loss of accuracy. Moreover, some heuristics may drop the alignment if the sequences exceed certain divergence thresholds (i.e., x-drop/z-drop). Due to the popularity and efficiency of these methods, the WFA2 library implements many of these heuristics. Note, **it is not about how little DP-matrix you compute, but about how good/accurate the resulting alignments are.**

- **None (for comparison)**. If no heuristic is used, the WFA behaves exploring cells of the DP-matrix in increasing score order (increasing scores correspond to colours from blue to red).

Expand Down Expand Up @@ -500,11 +500,20 @@ The WFA algorithm can be used combined with many heuristics to reduce the alignm
attributes.heuristic.steps_between_cutoffs = 100;
```

<!-- ### <a name="wfa2.other"></a> 3.6 Other features -->
### <a name="wfa2.other.notes"></a> 3.6 Some technical notes

- Thanks to Eizenga's formulation, WFA2-lib can operate with any match score. Although, in practice, M=0 is still the most efficient choice as it takes advantage of matches between the sequences to accelerate the alignment computation.

- All WFA2-lib algorithms/variants are stable. That is, for alignments having the same score, the library always resolves ties (between M, X, I,and D) using the same criteria: M (highest prio) > X > D > I (lowest prio). Nevertheless, the memory mode `ultralow` (BiWFA) is optimal (always reports the best alignment) but not stable.

## <a name="wfa2.complains"></a> 4. REPORTING BUGS AND FEATURE REQUEST

Feedback and bug reporting is highly appreciated. Please report any issue or suggestion on github or email to the main developer ([email protected]).
Feedback and bug reporting is highly appreciated. Please report any issue or suggestion on github or email to the main developer ([email protected]). Don't hesitate to contact us
if:
- You experience any bug or crash.
- You want to request a feature or have any suggestion.
- Your application using the library is running slower than it should or you expected.
- Need help integrating the library into your tool.

## <a name="wfa2.licence"></a> 5. LICENSE

Expand All @@ -528,3 +537,5 @@ Miquel Moretó has contributed with fruitful technical discussions and tireless

**Santiago Marco-Sola, Juan Carlos Moure, Miquel Moreto, Antonio Espinosa**. ["Fast gap-affine pairwise alignment using the wavefront algorithm."](https://doi.org/10.1093/bioinformatics/btaa777) Bioinformatics, 2020.

**Santiago Marco-Sola, Jordan M Eizenga, Andrea Guarracino, Benedict Paten, Erik Garrison, Miquel Moreto**. Optimal gap-affine alignment in O(s) space. _bioRxiv_ (2022). DOI [2022.04.14.488380](https://doi.org/10.1101/2022.04.14.488380)

File renamed without changes.
42 changes: 42 additions & 0 deletions scripts/utests/wfa.utest.default.sh
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@
#!/bin/bash -x
# PROJECT: Wavefront Alignments Algorithms (Unitary Tests)
# LICENCE: MIT License
# AUTHOR(S): Santiago Marco-Sola <[email protected]>
# DESCRIPTION: WFA unitary tests (for performance & correcness)
# USAGE: ./wfa.utest.default.sh

# Clear
rm *.log *.alg

# Config
ALGORITHM="gap-affine-wfa"
REDUCTION="--wfa-heuristic=wfa-adaptive --wfa-heuristic-parameters 10,50,1"
LOWMEMORY="--wfa-memory-mode=med"

# Utest for length=100
\time -v ./bin/align_benchmark -a $ALGORITHM -i ../data/sim.l100.n100K.e2.seq -o sim.l100.e2.W.alg &> sim.l100.e2.W.log
\time -v ./bin/align_benchmark -a $ALGORITHM -i ../data/sim.l100.n100K.e2.seq -o sim.l100.e2.Wl.alg $LOWMEMORY &> sim.l100.e2.Wl.log
\time -v ./bin/align_benchmark -a $ALGORITHM -i ../data/sim.l100.n100K.e2.seq -o sim.l100.e2.Wa.alg $REDUCTION &> sim.l100.e2.Wa.log

# Utest for length=1K
\time -v ./bin/align_benchmark -a $ALGORITHM -i ../data/sim.l1K.n10K.e20.seq -o sim.l1K.e20.W.alg &> sim.l1K.e20.W.log
\time -v ./bin/align_benchmark -a $ALGORITHM -i ../data/sim.l1K.n10K.e20.seq -o sim.l1K.e20.Wl.alg $LOWMEMORY &> sim.l1K.e20.Wl.log
\time -v ./bin/align_benchmark -a $ALGORITHM -i ../data/sim.l1K.n10K.e20.seq -o sim.l1K.e20.Wa.alg $REDUCTION &> sim.l1K.e20.Wa.log

# Utest for length=10K
\time -v ./bin/align_benchmark -a $ALGORITHM -i ../data/sim.l10K.n1K.e20.seq -o sim.l10K.e20.W.alg &> sim.l10K.e20.W.log
\time -v ./bin/align_benchmark -a $ALGORITHM -i ../data/sim.l10K.n1K.e20.seq -o sim.l10K.e20.Wa.alg $REDUCTION &> sim.l10K.e20.Wa.log

# Utest for length=100K
\time -v ./bin/align_benchmark -a $ALGORITHM -i ../data/sim.l100K.n1.e10.seq -o sim.l100K.e10.Wl.alg $LOWMEMORY &> sim.l100K.e10.Wl.log

#
# Random tests
#
# INPUT="../data/sim.l1K.n10K.e20.seq -P 500"
# ./bin/align_benchmark -a indel-wfa -i $INPUT -c score --check-distance indel -v
# ./bin/align_benchmark -a edit-wfa -i $INPUT -c score --check-distance edit -v
# ./bin/align_benchmark -a gap-linear-wfa -i $INPUT -c score --check-distance linear -v
# ./bin/align_benchmark -a gap-linear-wfa -i $INPUT -c score --check-distance linear -v
# ./bin/align_benchmark -a gap-affine-wfa -i $INPUT -c score --check-distance affine -v
# ./bin/align_benchmark -a gap-affine-2p-wfa -i $INPUT -c score --check-distance affine2p -v
72 changes: 72 additions & 0 deletions scripts/utests/wfa.utest.extended.sh
Original file line number Diff line number Diff line change
@@ -0,0 +1,72 @@
#!/bin/bash -x
# PROJECT: Wavefront Alignments Algorithms (Unitary Tests)
# LICENCE: MIT License
# AUTHOR(S): Santiago Marco-Sola <[email protected]>
# DESCRIPTION: WFA unitary tests (extended for different use-cases)
# USAGE: ./wfa.utest.extended.sh

# Clear
rm -rf *.log *.alg utests
mkdir utests

# Config
BIN="/usr/bin/time -v ./bin/align_benchmark"
BIWFA="--wfa-memory-mode=ultralow"
XTRAS="--check correct"

# Base data
DATA100="../data/sim.l100.n100K.e2.seq"
DATA1K="../data/sim.l1K.n10K.e20.seq"
DATA10K="../data/sim.l10K.n1K.e20.seq"

ENDSFREE1K="../data/sim.endsfree.l1K.n10K.seq"
ENDSFREE10K="../data/sim.endsfree.l10K.n1K.seq"

LONG100K="../data/sim.l100K.n10.e20.seq"

# Base (edit)
$BIN -a edit-wfa -i $DATA100 -o utests/test.l100.aedit.alg $XTRAS &> utests/test.l100.aedit.log
$BIN -a edit-wfa -i $DATA1K -o utests/test.l1K.aedit.alg $XTRAS &> utests/test.l1K.aedit.log
$BIN -a edit-wfa -i $DATA10K -o utests/test.l10K.aedit.alg $BIWFA $XTRAS &> utests/test.l10K.aedit.log
$BIN -a edit-wfa -i $DATA100 -o utests/test.l100.sedit.alg --wfa-score-only $XTRAS &> utests/test.l100.sedit.log
$BIN -a edit-wfa -i $DATA1K -o utests/test.l1K.sedit.alg --wfa-score-only $XTRAS &> utests/test.l1K.sedit.log
$BIN -a edit-wfa -i $DATA10K -o utests/test.l10K.sedit.alg --wfa-score-only $XTRAS &> utests/test.l10K.sedit.log

# Base (gap-linear)
$BIN -a gap-linear-wfa -i $DATA100 -o utests/test.l100.alinear.alg $XTRAS &> utests/test.l100.alinear.log
$BIN -a gap-linear-wfa -i $DATA1K -o utests/test.l1K.alinear.alg $XTRAS &> utests/test.l1K.alinear.log
$BIN -a gap-linear-wfa -i $DATA10K -o utests/test.l10K.alinear.alg $BIWFA $XTRAS &> utests/test.l10K.alinear.log
$BIN -a gap-linear-wfa -i $DATA100 -o utests/test.l100.slinear.alg --wfa-score-only $XTRAS &> utests/test.l100.slinear.log
$BIN -a gap-linear-wfa -i $DATA1K -o utests/test.l1K.slinear.alg --wfa-score-only $XTRAS &> utests/test.l1K.slinear.log
$BIN -a gap-linear-wfa -i $DATA10K -o utests/test.l10K.slinear.alg --wfa-score-only $XTRAS &> utests/test.l10K.slinear.log

# Base (gap-affine)
$BIN -a gap-affine-wfa -i $DATA100 -o utests/test.l100.aaffine.alg $XTRAS &> utests/test.l100.aaffine.log
$BIN -a gap-affine-wfa -i $DATA1K -o utests/test.l1K.aaffine.alg $XTRAS &> utests/test.l1K.aaffine.log
#$BIN -a gap-affine-wfa -i $DATA10K -o utests/test.l10K.aaffine.alg $BIWFA $XTRAS &> utests/test.l10K.aaffine.log
$BIN -a gap-affine-wfa -i $DATA100 -o utests/test.l100.saffine.alg --wfa-score-only $XTRAS &> utests/test.l100.saffine.log
$BIN -a gap-affine-wfa -i $DATA1K -o utests/test.l1K.saffine.alg --wfa-score-only $XTRAS &> utests/test.l1K.saffine.log
#$BIN -a gap-affine-wfa -i $DATA10K -o utests/test.l10K.saffine.alg --wfa-score-only $XTRAS &> utests/test.l10K.saffine.log

# Base (gap-affine-2p)
$BIN -a gap-affine2p-wfa -i $DATA100 -o utests/test.l100.a2p.alg $XTRAS &> utests/test.l100.a2p.log
$BIN -a gap-affine2p-wfa -i $DATA1K -o utests/test.l1K.a2p.alg $XTRAS &> utests/test.l1K.a2p.log
#$BIN -a gap-affine2p-wfa -i $DATA10K -o utests/test.l10K.a2p.alg $BIWFA $XTRAS &> utests/test.l10K.a2p.log
$BIN -a gap-affine2p-wfa -i $DATA100 -o utests/test.l100.s2p.alg --wfa-score-only $XTRAS &> utests/test.l100.s2p.log
$BIN -a gap-affine2p-wfa -i $DATA1K -o utests/test.l1K.s2p.alg --wfa-score-only $XTRAS &> utests/test.l1K.s2p.log
#$BIN -a gap-affine2p-wfa -i $DATA10K -o utests/test.l10K.s2p.alg --wfa-score-only $XTRAS &> utests/test.l10K.s2p.log

# Heuristics (gap-affine-2p alignment)
HEURISTIC="--wfa-heuristic=wfa-adaptive --wfa-heuristic-parameters=10,50,1"
$BIN -a gap-affine2p-wfa -i $DATA100 -o utests/test.adaptive.l100.alg $HEURISTIC $XTRAS &> utests/test.adaptive.l100.log
$BIN -a gap-affine2p-wfa -i $DATA1K -o utests/test.adaptive.l1K.alg $HEURISTIC $XTRAS &> utests/test.adaptive.l1K.log
$BIN -a gap-affine2p-wfa -i $DATA10K -o utests/test.adaptive.l10K.alg $HEURISTIC $XTRAS &> utests/test.adaptive.l10K.log

# Ends-free (length-diff=20%,e=10%,indels=2,20%)
$BIN -a gap-affine2p-wfa -i $ENDSFREE1K -o utests/test.endsfree.l1K.alg $BIWFA --ends-free=200,200,200,200 $XTRAS &> utests/test.endsfree.l1K.log
$BIN -a gap-affine2p-wfa -i $ENDSFREE10K -o utests/test.endsfree.l10K.alg $BIWFA --ends-free=2000,2000,2000,2000 $XTRAS &> utests/test.endsfree.l10K.log

# Very long sequences (100K)
#$BIN -a gap-affine2p-wfa -i $LONG100K -o utests/test.long.l100K.alg $BIWFA $XTRAS &> utests/test.long.l100K.log


Loading

0 comments on commit bdb09fa

Please sign in to comment.